440 lines
15 KiB
Java
440 lines
15 KiB
Java
/*
|
|
* Copyright (C) 2013 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.dialer.lettertile;
|
|
|
|
import android.content.res.Resources;
|
|
import android.content.res.TypedArray;
|
|
import android.graphics.Bitmap;
|
|
import android.graphics.Bitmap.Config;
|
|
import android.graphics.Canvas;
|
|
import android.graphics.ColorFilter;
|
|
import android.graphics.Outline;
|
|
import android.graphics.Paint;
|
|
import android.graphics.Paint.Align;
|
|
import android.graphics.Rect;
|
|
import android.graphics.Typeface;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.support.annotation.IntDef;
|
|
import android.support.annotation.NonNull;
|
|
import android.support.annotation.Nullable;
|
|
import android.telecom.TelecomManager;
|
|
import android.text.TextUtils;
|
|
import com.android.dialer.common.Assert;
|
|
import java.lang.annotation.Retention;
|
|
import java.lang.annotation.RetentionPolicy;
|
|
|
|
/**
|
|
* A drawable that encapsulates all the functionality needed to display a letter tile to represent a
|
|
* contact image.
|
|
*/
|
|
public class LetterTileDrawable extends Drawable {
|
|
|
|
/**
|
|
* ContactType indicates the avatar type of the contact. For a person or for the default when no
|
|
* name is provided, it is {@link #TYPE_DEFAULT}, otherwise, for a business it is {@link
|
|
* #TYPE_BUSINESS}, and voicemail contacts should use {@link #TYPE_VOICEMAIL}.
|
|
*/
|
|
@Retention(RetentionPolicy.SOURCE)
|
|
@IntDef({TYPE_PERSON, TYPE_BUSINESS, TYPE_VOICEMAIL, TYPE_GENERIC_AVATAR, TYPE_SPAM})
|
|
public @interface ContactType {}
|
|
|
|
/** Contact type constants */
|
|
public static final int TYPE_PERSON = 1;
|
|
|
|
public static final int TYPE_BUSINESS = 2;
|
|
public static final int TYPE_VOICEMAIL = 3;
|
|
/**
|
|
* A generic avatar that features the default icon, default color, and no letter. Useful for
|
|
* situations where a contact is anonymous.
|
|
*/
|
|
public static final int TYPE_GENERIC_AVATAR = 4;
|
|
|
|
public static final int TYPE_SPAM = 5;
|
|
public static final int TYPE_CONFERENCE = 6;
|
|
@ContactType public static final int TYPE_DEFAULT = TYPE_PERSON;
|
|
|
|
/**
|
|
* Shape indicates the letter tile shape. It can be either a {@link #SHAPE_CIRCLE}, otherwise, it
|
|
* is a {@link #SHAPE_RECTANGLE}.
|
|
*/
|
|
@Retention(RetentionPolicy.SOURCE)
|
|
@IntDef({SHAPE_CIRCLE, SHAPE_RECTANGLE})
|
|
public @interface Shape {}
|
|
|
|
/** Shape constants */
|
|
public static final int SHAPE_CIRCLE = 1;
|
|
|
|
public static final int SHAPE_RECTANGLE = 2;
|
|
|
|
/** 54% opacity */
|
|
private static final int ALPHA = 138;
|
|
/** 100% opacity */
|
|
private static final int SPAM_ALPHA = 255;
|
|
/** Default icon scale for vector drawable. */
|
|
private static final float VECTOR_ICON_SCALE = 0.7f;
|
|
|
|
/** Reusable components to avoid new allocations */
|
|
private final Paint paint = new Paint();
|
|
|
|
private final Rect rect = new Rect();
|
|
private final char[] firstChar = new char[1];
|
|
|
|
/** Letter tile */
|
|
@NonNull private final TypedArray colors;
|
|
|
|
private final int spamColor;
|
|
private final int defaultColor;
|
|
private final int tileFontColor;
|
|
private final float letterToTileRatio;
|
|
@NonNull private final Drawable defaultPersonAvatar;
|
|
@NonNull private final Drawable defaultBusinessAvatar;
|
|
@NonNull private final Drawable defaultVoicemailAvatar;
|
|
@NonNull private final Drawable defaultSpamAvatar;
|
|
@NonNull private final Drawable defaultConferenceAvatar;
|
|
|
|
@ContactType private int contactType = TYPE_DEFAULT;
|
|
private float scale = 1.0f;
|
|
private float offset = 0.0f;
|
|
private boolean isCircle = false;
|
|
|
|
private int color;
|
|
private Character letter = null;
|
|
|
|
private String displayName;
|
|
|
|
public LetterTileDrawable(final Resources res) {
|
|
colors = res.obtainTypedArray(R.array.letter_tile_colors);
|
|
spamColor = res.getColor(R.color.spam_contact_background);
|
|
defaultColor = res.getColor(R.color.letter_tile_default_color);
|
|
tileFontColor = res.getColor(R.color.letter_tile_font_color);
|
|
letterToTileRatio = res.getFraction(R.dimen.letter_to_tile_ratio, 1, 1);
|
|
defaultPersonAvatar =
|
|
res.getDrawable(R.drawable.product_logo_avatar_anonymous_white_color_120, null);
|
|
defaultBusinessAvatar = res.getDrawable(R.drawable.quantum_ic_business_vd_theme_24, null);
|
|
defaultVoicemailAvatar = res.getDrawable(R.drawable.quantum_ic_voicemail_vd_theme_24, null);
|
|
defaultSpamAvatar = res.getDrawable(R.drawable.quantum_ic_report_vd_theme_24, null);
|
|
defaultConferenceAvatar = res.getDrawable(R.drawable.quantum_ic_group_vd_theme_24, null);
|
|
|
|
paint.setTypeface(Typeface.create("sans-serif-medium", Typeface.NORMAL));
|
|
paint.setTextAlign(Align.CENTER);
|
|
paint.setAntiAlias(true);
|
|
paint.setFilterBitmap(true);
|
|
paint.setDither(true);
|
|
color = defaultColor;
|
|
}
|
|
|
|
private Rect getScaledBounds(float scale, float offset) {
|
|
// The drawable should be drawn in the middle of the canvas without changing its width to
|
|
// height ratio.
|
|
final Rect destRect = copyBounds();
|
|
// Crop the destination bounds into a square, scaled and offset as appropriate
|
|
final int halfLength = (int) (scale * Math.min(destRect.width(), destRect.height()) / 2);
|
|
|
|
destRect.set(
|
|
destRect.centerX() - halfLength,
|
|
(int) (destRect.centerY() - halfLength + offset * destRect.height()),
|
|
destRect.centerX() + halfLength,
|
|
(int) (destRect.centerY() + halfLength + offset * destRect.height()));
|
|
return destRect;
|
|
}
|
|
|
|
private Drawable getDrawableForContactType(int contactType) {
|
|
switch (contactType) {
|
|
case TYPE_BUSINESS:
|
|
scale = VECTOR_ICON_SCALE;
|
|
return defaultBusinessAvatar;
|
|
case TYPE_VOICEMAIL:
|
|
scale = VECTOR_ICON_SCALE;
|
|
return defaultVoicemailAvatar;
|
|
case TYPE_SPAM:
|
|
scale = VECTOR_ICON_SCALE;
|
|
return defaultSpamAvatar;
|
|
case TYPE_CONFERENCE:
|
|
scale = VECTOR_ICON_SCALE;
|
|
return defaultConferenceAvatar;
|
|
case TYPE_PERSON:
|
|
case TYPE_GENERIC_AVATAR:
|
|
default:
|
|
return defaultPersonAvatar;
|
|
}
|
|
}
|
|
|
|
private static boolean isEnglishLetter(final char c) {
|
|
return ('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z');
|
|
}
|
|
|
|
@Override
|
|
public void draw(@NonNull final Canvas canvas) {
|
|
final Rect bounds = getBounds();
|
|
if (!isVisible() || bounds.isEmpty()) {
|
|
return;
|
|
}
|
|
// Draw letter tile.
|
|
drawLetterTile(canvas);
|
|
}
|
|
|
|
public Bitmap getBitmap(int width, int height) {
|
|
Bitmap bitmap = Bitmap.createBitmap(width, height, Config.ARGB_8888);
|
|
this.setBounds(0, 0, width, height);
|
|
Canvas canvas = new Canvas(bitmap);
|
|
this.draw(canvas);
|
|
return bitmap;
|
|
}
|
|
|
|
private void drawLetterTile(final Canvas canvas) {
|
|
// Draw background color.
|
|
paint.setColor(color);
|
|
|
|
final Rect bounds = getBounds();
|
|
final int minDimension = Math.min(bounds.width(), bounds.height());
|
|
|
|
if (isCircle) {
|
|
canvas.drawCircle(bounds.centerX(), bounds.centerY(), minDimension / 2, paint);
|
|
} else {
|
|
canvas.drawRect(bounds, paint);
|
|
}
|
|
|
|
// Draw letter/digit only if the first character is an english letter or there's a override
|
|
if (letter != null) {
|
|
// Draw letter or digit.
|
|
firstChar[0] = letter;
|
|
|
|
// Scale text by canvas bounds and user selected scaling factor
|
|
paint.setTextSize(scale * letterToTileRatio * minDimension);
|
|
paint.getTextBounds(firstChar, 0, 1, rect);
|
|
paint.setTypeface(Typeface.create("sans-serif", Typeface.NORMAL));
|
|
paint.setColor(tileFontColor);
|
|
paint.setAlpha(ALPHA);
|
|
|
|
// Draw the letter in the canvas, vertically shifted up or down by the user-defined
|
|
// offset
|
|
canvas.drawText(
|
|
firstChar,
|
|
0,
|
|
1,
|
|
bounds.centerX(),
|
|
bounds.centerY() + offset * bounds.height() - rect.exactCenterY(),
|
|
paint);
|
|
} else {
|
|
// Draw the default image if there is no letter/digit to be drawn
|
|
Drawable drawable = getDrawableForContactType(contactType);
|
|
if (drawable == null) {
|
|
throw Assert.createIllegalStateFailException(
|
|
"Unable to find drawable for contact type " + contactType);
|
|
}
|
|
|
|
drawable.setBounds(getScaledBounds(scale, offset));
|
|
drawable.setAlpha(drawable == defaultSpamAvatar ? SPAM_ALPHA : ALPHA);
|
|
drawable.draw(canvas);
|
|
}
|
|
}
|
|
|
|
public int getColor() {
|
|
return color;
|
|
}
|
|
|
|
public LetterTileDrawable setColor(int color) {
|
|
this.color = color;
|
|
return this;
|
|
}
|
|
|
|
/** Returns a deterministic color based on the provided contact identifier string. */
|
|
private int pickColor(final String identifier) {
|
|
if (contactType == TYPE_SPAM) {
|
|
return spamColor;
|
|
}
|
|
|
|
if (contactType == TYPE_VOICEMAIL
|
|
|| contactType == TYPE_BUSINESS
|
|
|| TextUtils.isEmpty(identifier)) {
|
|
return defaultColor;
|
|
}
|
|
|
|
// String.hashCode() implementation is not supposed to change across java versions, so
|
|
// this should guarantee the same email address always maps to the same color.
|
|
// The email should already have been normalized by the ContactRequest.
|
|
final int color = Math.abs(identifier.hashCode()) % colors.length();
|
|
return colors.getColor(color, defaultColor);
|
|
}
|
|
|
|
@Override
|
|
public void setAlpha(final int alpha) {
|
|
paint.setAlpha(alpha);
|
|
}
|
|
|
|
@Override
|
|
public void setColorFilter(final ColorFilter cf) {
|
|
paint.setColorFilter(cf);
|
|
}
|
|
|
|
@Override
|
|
public int getOpacity() {
|
|
return android.graphics.PixelFormat.OPAQUE;
|
|
}
|
|
|
|
@Override
|
|
public void getOutline(Outline outline) {
|
|
if (isCircle) {
|
|
outline.setOval(getBounds());
|
|
} else {
|
|
outline.setRect(getBounds());
|
|
}
|
|
|
|
outline.setAlpha(1);
|
|
}
|
|
|
|
/**
|
|
* Scale the drawn letter tile to a ratio of its default size
|
|
*
|
|
* @param scale The ratio the letter tile should be scaled to as a percentage of its default size,
|
|
* from a scale of 0 to 2.0f. The default is 1.0f.
|
|
*/
|
|
public LetterTileDrawable setScale(float scale) {
|
|
this.scale = scale;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Assigns the vertical offset of the position of the letter tile to the ContactDrawable
|
|
*
|
|
* @param offset The provided offset must be within the range of -0.5f to 0.5f. If set to -0.5f,
|
|
* the letter will be shifted upwards by 0.5 times the height of the canvas it is being drawn
|
|
* on, which means it will be drawn with the center of the letter starting at the top edge of
|
|
* the canvas. If set to 0.5f, the letter will be shifted downwards by 0.5 times the height of
|
|
* the canvas it is being drawn on, which means it will be drawn with the center of the letter
|
|
* starting at the bottom edge of the canvas. The default is 0.0f.
|
|
*/
|
|
public LetterTileDrawable setOffset(float offset) {
|
|
Assert.checkArgument(offset >= -0.5f && offset <= 0.5f);
|
|
this.offset = offset;
|
|
return this;
|
|
}
|
|
|
|
public LetterTileDrawable setLetter(Character letter) {
|
|
this.letter = letter;
|
|
return this;
|
|
}
|
|
|
|
public Character getLetter() {
|
|
return this.letter;
|
|
}
|
|
|
|
private LetterTileDrawable setLetterAndColorFromContactDetails(
|
|
final String displayName, final String identifier) {
|
|
if (!TextUtils.isEmpty(displayName) && isEnglishLetter(displayName.charAt(0))) {
|
|
letter = Character.toUpperCase(displayName.charAt(0));
|
|
} else {
|
|
letter = null;
|
|
}
|
|
color = pickColor(identifier);
|
|
return this;
|
|
}
|
|
|
|
private LetterTileDrawable setContactType(@ContactType int contactType) {
|
|
this.contactType = contactType;
|
|
return this;
|
|
}
|
|
|
|
@ContactType
|
|
public int getContactType() {
|
|
return this.contactType;
|
|
}
|
|
|
|
public LetterTileDrawable setIsCircular(boolean isCircle) {
|
|
this.isCircle = isCircle;
|
|
return this;
|
|
}
|
|
|
|
public boolean tileIsCircular() {
|
|
return this.isCircle;
|
|
}
|
|
|
|
/**
|
|
* Creates a canonical letter tile for use across dialer fragments.
|
|
*
|
|
* @param displayName The display name to produce the letter in the tile. Null values or numbers
|
|
* yield no letter.
|
|
* @param identifierForTileColor The string used to produce the tile color.
|
|
* @param shape The shape of the tile.
|
|
* @param contactType The type of contact, e.g. TYPE_VOICEMAIL.
|
|
* @return this
|
|
*/
|
|
public LetterTileDrawable setCanonicalDialerLetterTileDetails(
|
|
@Nullable final String displayName,
|
|
@Nullable final String identifierForTileColor,
|
|
@Shape final int shape,
|
|
final int contactType) {
|
|
|
|
this.setIsCircular(shape == SHAPE_CIRCLE);
|
|
|
|
/**
|
|
* We return quickly under the following conditions: 1. We are asked to draw a default tile, and
|
|
* no coloring information is provided, meaning no further initialization is necessary OR 2.
|
|
* We've already invoked this method before, set mDisplayName, and found that it has not
|
|
* changed. This is useful during events like hangup, when we lose the call state for special
|
|
* types of contacts, like voicemail. We keep track of the special case until we encounter a new
|
|
* display name.
|
|
*/
|
|
if (contactType == TYPE_DEFAULT
|
|
&& ((displayName == null && identifierForTileColor == null)
|
|
|| (displayName != null && displayName.equals(this.displayName)))) {
|
|
return this;
|
|
}
|
|
|
|
this.displayName = displayName;
|
|
setContactType(contactType);
|
|
|
|
// Special contact types receive default color and no letter tile, but special iconography.
|
|
if (contactType != TYPE_PERSON) {
|
|
this.setLetterAndColorFromContactDetails(null, null);
|
|
} else {
|
|
if (identifierForTileColor != null) {
|
|
this.setLetterAndColorFromContactDetails(displayName, identifierForTileColor);
|
|
} else {
|
|
this.setLetterAndColorFromContactDetails(displayName, displayName);
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Returns the appropriate LetterTileDrawable.TYPE_ based on the given primitive conditions.
|
|
*
|
|
* <p>If no special state is detected, yields TYPE_DEFAULT
|
|
*/
|
|
public static @ContactType int getContactTypeFromPrimitives(
|
|
boolean isVoicemailNumber,
|
|
boolean isSpam,
|
|
boolean isBusiness,
|
|
int numberPresentation,
|
|
boolean isConference) {
|
|
if (isVoicemailNumber) {
|
|
return LetterTileDrawable.TYPE_VOICEMAIL;
|
|
} else if (isSpam) {
|
|
return LetterTileDrawable.TYPE_SPAM;
|
|
} else if (isBusiness) {
|
|
return LetterTileDrawable.TYPE_BUSINESS;
|
|
} else if (numberPresentation == TelecomManager.PRESENTATION_RESTRICTED) {
|
|
return LetterTileDrawable.TYPE_GENERIC_AVATAR;
|
|
} else if (isConference) {
|
|
return LetterTileDrawable.TYPE_CONFERENCE;
|
|
} else {
|
|
return LetterTileDrawable.TYPE_DEFAULT;
|
|
}
|
|
}
|
|
}
|