Home | History | Annotate | Download | only in lettertiles
      1 /*
      2  * Copyright (C) 2013 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.common.lettertiles;
     18 
     19 import android.content.res.Resources;
     20 import android.content.res.TypedArray;
     21 import android.graphics.Bitmap;
     22 import android.graphics.BitmapFactory;
     23 import android.graphics.Canvas;
     24 import android.graphics.ColorFilter;
     25 import android.graphics.Paint;
     26 import android.graphics.Paint.Align;
     27 import android.graphics.Rect;
     28 import android.graphics.Typeface;
     29 import android.graphics.drawable.Drawable;
     30 import android.text.TextUtils;
     31 import android.util.Log;
     32 
     33 import com.android.contacts.common.R;
     34 import com.android.contacts.common.util.BitmapUtil;
     35 
     36 import junit.framework.Assert;
     37 
     38 /**
     39  * A drawable that encapsulates all the functionality needed to display a letter tile to
     40  * represent a contact image.
     41  */
     42 public class LetterTileDrawable extends Drawable {
     43 
     44     private final String TAG = LetterTileDrawable.class.getSimpleName();
     45 
     46     private final Paint mPaint;
     47 
     48     /** Letter tile */
     49     private static TypedArray sColors;
     50     private static int sDefaultColor;
     51     private static int sTileFontColor;
     52     private static float sLetterToTileRatio;
     53     private static Bitmap DEFAULT_PERSON_AVATAR;
     54     private static Bitmap DEFAULT_BUSINESS_AVATAR;
     55     private static Bitmap DEFAULT_VOICEMAIL_AVATAR;
     56 
     57     /** Reusable components to avoid new allocations */
     58     private static final Paint sPaint = new Paint();
     59     private static final Rect sRect = new Rect();
     60     private static final char[] sFirstChar = new char[1];
     61 
     62     /** Contact type constants */
     63     public static final int TYPE_PERSON = 1;
     64     public static final int TYPE_BUSINESS = 2;
     65     public static final int TYPE_VOICEMAIL = 3;
     66     public static final int TYPE_DEFAULT = TYPE_PERSON;
     67 
     68     private String mDisplayName;
     69     private String mIdentifier;
     70     private int mContactType = TYPE_DEFAULT;
     71     private float mScale = 1.0f;
     72     private float mOffset = 0.0f;
     73 
     74     /** This should match the total number of colors defined in colors.xml for letter_tile_color */
     75     private static final int NUM_OF_TILE_COLORS = 8;
     76 
     77     public LetterTileDrawable(final Resources res) {
     78         mPaint = new Paint();
     79         mPaint.setFilterBitmap(true);
     80         mPaint.setDither(true);
     81 
     82         if (sColors == null) {
     83             sColors = res.obtainTypedArray(R.array.letter_tile_colors);
     84             sDefaultColor = res.getColor(R.color.letter_tile_default_color);
     85             sTileFontColor = res.getColor(R.color.letter_tile_font_color);
     86             sLetterToTileRatio = res.getFraction(R.dimen.letter_to_tile_ratio, 1, 1);
     87             DEFAULT_PERSON_AVATAR = BitmapFactory.decodeResource(res,
     88                     R.drawable.ic_list_item_avatar);
     89             DEFAULT_BUSINESS_AVATAR = BitmapFactory.decodeResource(res,
     90                     R.drawable.ic_list_item_businessavatar);
     91             DEFAULT_VOICEMAIL_AVATAR = BitmapFactory.decodeResource(res,
     92                     R.drawable.ic_voicemail_avatar);
     93             sPaint.setTypeface(Typeface.create(
     94                     res.getString(R.string.letter_tile_letter_font_family), Typeface.NORMAL));
     95             sPaint.setTextAlign(Align.CENTER);
     96             sPaint.setAntiAlias(true);
     97         }
     98     }
     99 
    100     @Override
    101     public void draw(final Canvas canvas) {
    102         final Rect bounds = getBounds();
    103         if (!isVisible() || bounds.isEmpty()) {
    104             return;
    105         }
    106         // Draw letter tile.
    107         drawLetterTile(canvas);
    108     }
    109 
    110     /**
    111      * Draw the bitmap onto the canvas at the current bounds taking into account the current scale.
    112      */
    113     private void drawBitmap(final Bitmap bitmap, final int width, final int height,
    114             final Canvas canvas) {
    115         // The bitmap should be drawn in the middle of the canvas without changing its width to
    116         // height ratio.
    117         final Rect destRect = copyBounds();
    118 
    119         // Crop the destination bounds into a square, scaled and offset as appropriate
    120         final int halfLength = (int) (mScale * Math.min(destRect.width(), destRect.height()) / 2);
    121 
    122         destRect.set(destRect.centerX() - halfLength,
    123                 (int) (destRect.centerY() - halfLength + mOffset * destRect.height()),
    124                 destRect.centerX() + halfLength,
    125                 (int) (destRect.centerY() + halfLength + mOffset * destRect.height()));
    126 
    127         // Source rectangle remains the entire bounds of the source bitmap.
    128         sRect.set(0, 0, width, height);
    129 
    130         canvas.drawBitmap(bitmap, sRect, destRect, mPaint);
    131     }
    132 
    133     private void drawLetterTile(final Canvas canvas) {
    134         // Draw background color.
    135         sPaint.setColor(pickColor(mIdentifier));
    136 
    137         sPaint.setAlpha(mPaint.getAlpha());
    138         canvas.drawRect(getBounds(), sPaint);
    139 
    140         // Draw letter/digit only if the first character is an english letter
    141         if (mDisplayName != null && isEnglishLetter(mDisplayName.charAt(0))) {
    142             // Draw letter or digit.
    143             sFirstChar[0] = Character.toUpperCase(mDisplayName.charAt(0));
    144 
    145             // Scale text by canvas bounds and user selected scaling factor
    146             final int minDimension = Math.min(getBounds().width(), getBounds().height());
    147             sPaint.setTextSize(mScale * sLetterToTileRatio * minDimension);
    148             //sPaint.setTextSize(sTileLetterFontSize);
    149             sPaint.getTextBounds(sFirstChar, 0, 1, sRect);
    150             sPaint.setColor(sTileFontColor);
    151             final Rect bounds = getBounds();
    152 
    153             // Draw the letter in the canvas, vertically shifted up or down by the user-defined
    154             // offset
    155             canvas.drawText(sFirstChar, 0, 1, bounds.centerX(),
    156                     bounds.centerY() + mOffset * bounds.height() + sRect.height() / 2,
    157                     sPaint);
    158         } else {
    159             // Draw the default image if there is no letter/digit to be drawn
    160             final Bitmap bitmap = getBitmapForContactType(mContactType);
    161             drawBitmap(bitmap, bitmap.getWidth(), bitmap.getHeight(),
    162                         canvas);
    163         }
    164     }
    165 
    166     /**
    167      * Returns a deterministic color based on the provided contact identifier string.
    168      */
    169     private int pickColor(final String identifier) {
    170         if (TextUtils.isEmpty(identifier) || mContactType == TYPE_VOICEMAIL) {
    171             return sDefaultColor;
    172         }
    173         // String.hashCode() implementation is not supposed to change across java versions, so
    174         // this should guarantee the same email address always maps to the same color.
    175         // The email should already have been normalized by the ContactRequest.
    176         final int color = Math.abs(identifier.hashCode()) % NUM_OF_TILE_COLORS;
    177         return sColors.getColor(color, sDefaultColor);
    178     }
    179 
    180     private static Bitmap getBitmapForContactType(int contactType) {
    181         switch (contactType) {
    182             case TYPE_PERSON:
    183                 return DEFAULT_PERSON_AVATAR;
    184             case TYPE_BUSINESS:
    185                 return DEFAULT_BUSINESS_AVATAR;
    186             case TYPE_VOICEMAIL:
    187                 return DEFAULT_VOICEMAIL_AVATAR;
    188             default:
    189                 return DEFAULT_PERSON_AVATAR;
    190         }
    191     }
    192 
    193     private static boolean isEnglishLetter(final char c) {
    194         return ('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z');
    195     }
    196 
    197     @Override
    198     public void setAlpha(final int alpha) {
    199         mPaint.setAlpha(alpha);
    200     }
    201 
    202     @Override
    203     public void setColorFilter(final ColorFilter cf) {
    204         mPaint.setColorFilter(cf);
    205     }
    206 
    207     @Override
    208     public int getOpacity() {
    209         return android.graphics.PixelFormat.OPAQUE;
    210     }
    211 
    212     /**
    213      * Scale the drawn letter tile to a ratio of its default size
    214      *
    215      * @param scale The ratio the letter tile should be scaled to as a percentage of its default
    216      * size, from a scale of 0 to 2.0f. The default is 1.0f.
    217      */
    218     public void setScale(float scale) {
    219         mScale = scale;
    220     }
    221 
    222     /**
    223      * Assigns the vertical offset of the position of the letter tile to the ContactDrawable
    224      *
    225      * @param offset The provided offset must be within the range of -0.5f to 0.5f.
    226      * If set to -0.5f, the letter will be shifted upwards by 0.5 times the height of the canvas
    227      * it is being drawn on, which means it will be drawn with the center of the letter starting
    228      * at the top edge of the canvas.
    229      * If set to 0.5f, the letter will be shifted downwards by 0.5 times the height of the canvas
    230      * it is being drawn on, which means it will be drawn with the center of the letter starting
    231      * at the bottom edge of the canvas.
    232      * The default is 0.0f.
    233      */
    234     public void setOffset(float offset) {
    235         Assert.assertTrue(offset >= -0.5f && offset <= 0.5f);
    236         mOffset = offset;
    237     }
    238 
    239     public void setContactDetails(final String displayName, final String identifier) {
    240         mDisplayName = displayName;
    241         mIdentifier = identifier;
    242     }
    243 
    244     public void setContactType(int contactType) {
    245         mContactType = contactType;
    246     }
    247 }
    248