blob: a22841790abcf353a768bc3ed2c85999262ddd29 [file] [log] [blame]
/*
* Copyright (C) 2015 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.messaging.util;
import android.app.ActivityManager;
import android.content.ContentResolver;
import android.content.Context;
import android.database.Cursor;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.BitmapShader;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.PorterDuff;
import android.graphics.Rect;
import android.graphics.RectF;
import android.graphics.Shader.TileMode;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import android.provider.MediaStore;
import androidx.annotation.Nullable;
import android.text.TextUtils;
import android.view.View;
import com.android.messaging.Factory;
import com.android.messaging.datamodel.MediaScratchFileProvider;
import com.android.messaging.datamodel.MessagingContentProvider;
import com.android.messaging.datamodel.media.ImageRequest;
import com.android.messaging.util.Assert.DoesNotRunOnMainThread;
import com.android.messaging.util.exif.ExifInterface;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.io.Files;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.Charset;
import java.util.Arrays;
public class ImageUtils {
private static final String TAG = LogUtil.BUGLE_TAG;
private static final int MAX_OOM_COUNT = 1;
private static final byte[] GIF87_HEADER = "GIF87a".getBytes(Charset.forName("US-ASCII"));
private static final byte[] GIF89_HEADER = "GIF89a".getBytes(Charset.forName("US-ASCII"));
// Used for drawBitmapWithCircleOnCanvas.
// Default color is transparent for both circle background and stroke.
public static final int DEFAULT_CIRCLE_BACKGROUND_COLOR = 0;
public static final int DEFAULT_CIRCLE_STROKE_COLOR = 0;
private static volatile ImageUtils sInstance;
public static ImageUtils get() {
if (sInstance == null) {
synchronized (ImageUtils.class) {
if (sInstance == null) {
sInstance = new ImageUtils();
}
}
}
return sInstance;
}
@VisibleForTesting
public static void set(final ImageUtils imageUtils) {
sInstance = imageUtils;
}
/**
* Transforms a bitmap into a byte array.
*
* @param quality Value between 0 and 100 that the compressor uses to discern what quality the
* resulting bytes should be
* @param bitmap Bitmap to convert into bytes
* @return byte array of bitmap
*/
public static byte[] bitmapToBytes(final Bitmap bitmap, final int quality)
throws OutOfMemoryError {
boolean done = false;
int oomCount = 0;
byte[] imageBytes = null;
while (!done) {
try {
final ByteArrayOutputStream os = new ByteArrayOutputStream();
bitmap.compress(Bitmap.CompressFormat.JPEG, quality, os);
imageBytes = os.toByteArray();
done = true;
} catch (final OutOfMemoryError e) {
LogUtil.w(TAG, "OutOfMemory converting bitmap to bytes.");
oomCount++;
if (oomCount <= MAX_OOM_COUNT) {
Factory.get().reclaimMemory();
} else {
done = true;
LogUtil.w(TAG, "Failed to convert bitmap to bytes. Out of Memory.");
}
throw e;
}
}
return imageBytes;
}
/**
* Given the source bitmap and a canvas, draws the bitmap through a circular
* mask. Only draws a circle with diameter equal to the destination width.
*
* @param bitmap The source bitmap to draw.
* @param canvas The canvas to draw it on.
* @param source The source bound of the bitmap.
* @param dest The destination bound on the canvas.
* @param bitmapPaint Optional Paint object for the bitmap
* @param fillBackground when set, fill the circle with backgroundColor
* @param strokeColor draw a border outside the circle with strokeColor
*/
public static void drawBitmapWithCircleOnCanvas(final Bitmap bitmap, final Canvas canvas,
final RectF source, final RectF dest, @Nullable Paint bitmapPaint,
final boolean fillBackground, final int backgroundColor, int strokeColor) {
// Draw bitmap through shader first.
final BitmapShader shader = new BitmapShader(bitmap, TileMode.CLAMP, TileMode.CLAMP);
final Matrix matrix = new Matrix();
// Fit bitmap to bounds.
matrix.setRectToRect(source, dest, Matrix.ScaleToFit.CENTER);
shader.setLocalMatrix(matrix);
if (bitmapPaint == null) {
bitmapPaint = new Paint();
}
bitmapPaint.setAntiAlias(true);
if (fillBackground) {
bitmapPaint.setColor(backgroundColor);
canvas.drawCircle(dest.centerX(), dest.centerX(), dest.width() / 2f, bitmapPaint);
}
bitmapPaint.setShader(shader);
canvas.drawCircle(dest.centerX(), dest.centerX(), dest.width() / 2f, bitmapPaint);
bitmapPaint.setShader(null);
if (strokeColor != 0) {
final Paint stroke = new Paint();
stroke.setAntiAlias(true);
stroke.setColor(strokeColor);
stroke.setStyle(Paint.Style.STROKE);
final float strokeWidth = 6f;
stroke.setStrokeWidth(strokeWidth);
canvas.drawCircle(dest.centerX(),
dest.centerX(),
dest.width() / 2f - stroke.getStrokeWidth() / 2f,
stroke);
}
}
/**
* Sets a drawable to the background of a view. setBackgroundDrawable() is deprecated since
* JB and replaced by setBackground().
*/
@SuppressWarnings("deprecation")
public static void setBackgroundDrawableOnView(final View view, final Drawable drawable) {
if (OsUtil.isAtLeastJB()) {
view.setBackground(drawable);
} else {
view.setBackgroundDrawable(drawable);
}
}
/**
* Based on the input bitmap bounds given by BitmapFactory.Options, compute the required
* sub-sampling size for loading a scaled down version of the bitmap to the required size
* @param options a BitmapFactory.Options instance containing the bounds info of the bitmap
* @param reqWidth the desired width of the bitmap. Can be ImageRequest.UNSPECIFIED_SIZE.
* @param reqHeight the desired height of the bitmap. Can be ImageRequest.UNSPECIFIED_SIZE.
* @return
*/
public int calculateInSampleSize(
final BitmapFactory.Options options, final int reqWidth, final int reqHeight) {
// Raw height and width of image
final int height = options.outHeight;
final int width = options.outWidth;
int inSampleSize = 1;
final boolean checkHeight = reqHeight != ImageRequest.UNSPECIFIED_SIZE;
final boolean checkWidth = reqWidth != ImageRequest.UNSPECIFIED_SIZE;
if ((checkHeight && height > reqHeight) ||
(checkWidth && width > reqWidth)) {
final int halfHeight = height / 2;
final int halfWidth = width / 2;
// Calculate the largest inSampleSize value that is a power of 2 and keeps both
// height and width larger than the requested height and width.
while ((!checkHeight || (halfHeight / inSampleSize) > reqHeight)
&& (!checkWidth || (halfWidth / inSampleSize) > reqWidth)) {
inSampleSize *= 2;
}
}
return inSampleSize;
}
private static final String[] MEDIA_CONTENT_PROJECTION = new String[] {
MediaStore.MediaColumns.MIME_TYPE
};
private static final int INDEX_CONTENT_TYPE = 0;
@DoesNotRunOnMainThread
public static String getContentType(final ContentResolver cr, final Uri uri) {
// Figure out the content type of media.
String contentType = null;
Cursor cursor = null;
if (UriUtil.isMediaStoreUri(uri)) {
try {
cursor = cr.query(uri, MEDIA_CONTENT_PROJECTION, null, null, null);
if (cursor != null && cursor.moveToFirst()) {
contentType = cursor.getString(INDEX_CONTENT_TYPE);
}
} finally {
if (cursor != null) {
cursor.close();
}
}
}
if (contentType == null) {
// Last ditch effort to get the content type. Look at the file extension.
contentType = ContentType.getContentTypeFromExtension(uri.toString(),
ContentType.IMAGE_UNSPECIFIED);
}
return contentType;
}
/**
* @param context Android context
* @param uri Uri to the image data
* @return The exif orientation value for the image in the specified uri
*/
public static int getOrientation(final Context context, final Uri uri) {
try {
return getOrientation(context.getContentResolver().openInputStream(uri));
} catch (FileNotFoundException e) {
LogUtil.e(TAG, "getOrientation couldn't open: " + uri, e);
}
return android.media.ExifInterface.ORIENTATION_UNDEFINED;
}
/**
* @param inputStream The stream to the image file. Closed on completion
* @return The exif orientation value for the image in the specified stream
*/
public static int getOrientation(final InputStream inputStream) {
int orientation = android.media.ExifInterface.ORIENTATION_UNDEFINED;
if (inputStream != null) {
try {
final ExifInterface exifInterface = new ExifInterface();
exifInterface.readExif(inputStream);
final Integer orientationValue =
exifInterface.getTagIntValue(ExifInterface.TAG_ORIENTATION);
if (orientationValue != null) {
orientation = orientationValue.intValue();
}
} catch (IOException e) {
// If the image if GIF, PNG, or missing exif header, just use the defaults
} finally {
try {
if (inputStream != null) {
inputStream.close();
}
} catch (IOException e) {
LogUtil.e(TAG, "getOrientation error closing input stream", e);
}
}
}
return orientation;
}
/**
* Returns whether the resource is a GIF image.
*/
public static boolean isGif(String contentType, Uri contentUri) {
if (TextUtils.equals(contentType, ContentType.IMAGE_GIF)) {
return true;
}
if (ContentType.isImageType(contentType)) {
try {
ContentResolver contentResolver = Factory.get().getApplicationContext()
.getContentResolver();
InputStream inputStream = contentResolver.openInputStream(contentUri);
return ImageUtils.isGif(inputStream);
} catch (Exception e) {
LogUtil.w(TAG, "Could not open GIF input stream", e);
}
}
// Assume anything with a non-image content type is not a GIF
return false;
}
/**
* @param inputStream The stream to the image file. Closed on completion
* @return Whether the image stream represents a GIF
*/
public static boolean isGif(InputStream inputStream) {
if (inputStream != null) {
try {
byte[] gifHeaderBytes = new byte[6];
int value = inputStream.read(gifHeaderBytes, 0, 6);
if (value == 6) {
return Arrays.equals(gifHeaderBytes, GIF87_HEADER)
|| Arrays.equals(gifHeaderBytes, GIF89_HEADER);
}
} catch (IOException e) {
return false;
} finally {
try {
inputStream.close();
} catch (IOException e) {
// Ignore
}
}
}
return false;
}
/**
* Read an image and compress it to particular max dimensions and size.
* Used to ensure images can fit in an MMS.
* TODO: This uses memory very inefficiently as it processes the whole image as a unit
* (rather than slice by slice) but system JPEG functions do not support slicing and dicing.
*/
public static class ImageResizer {
/**
* The quality parameter which is used to compress JPEG images.
*/
private static final int IMAGE_COMPRESSION_QUALITY = 95;
/**
* The minimum quality parameter which is used to compress JPEG images.
*/
private static final int MINIMUM_IMAGE_COMPRESSION_QUALITY = 50;
/**
* Minimum factor to reduce quality value
*/
private static final double QUALITY_SCALE_DOWN_RATIO = 0.85f;
/**
* Maximum passes through the resize loop before failing permanently
*/
private static final int NUMBER_OF_RESIZE_ATTEMPTS = 6;
/**
* Amount to scale down the picture when it doesn't fit
*/
private static final float MIN_SCALE_DOWN_RATIO = 0.75f;
/**
* When computing sampleSize target scaling of no more than this ratio
*/
private static final float MAX_TARGET_SCALE_FACTOR = 1.5f;
// Current sample size for subsampling image during initial decode
private int mSampleSize;
// Current bitmap holding initial decoded source image
private Bitmap mDecoded;
// If scaling is needed this holds the scaled bitmap (else should equal mDecoded)
private Bitmap mScaled;
// Current JPEG compression quality to use when compressing image
private int mQuality;
// Current factor to scale down decoded image before compressing
private float mScaleFactor;
// Flag keeping track of whether cache memory has been reclaimed
private boolean mHasReclaimedMemory;
// Initial size of the image (typically provided but can be UNSPECIFIED_SIZE)
private int mWidth;
private int mHeight;
// Orientation params of image as read from EXIF data
private final ExifInterface.OrientationParams mOrientationParams;
// Matrix to undo orientation and scale at the same time
private final Matrix mMatrix;
// Size limit as provided by MMS library
private final int mWidthLimit;
private final int mHeightLimit;
private final int mByteLimit;
// Uri from which to read source image
private final Uri mUri;
// Application context
private final Context mContext;
// Cached value of bitmap factory options
private final BitmapFactory.Options mOptions;
private final String mContentType;
private final int mMemoryClass;
/**
* Return resized (compressed) image (else null)
*
* @param width The width of the image (if known)
* @param height The height of the image (if known)
* @param orientation The orientation of the image as an ExifInterface constant
* @param widthLimit The width limit, in pixels
* @param heightLimit The height limit, in pixels
* @param byteLimit The binary size limit, in bytes
* @param uri Uri to the image data
* @param context Needed to open the image
* @param contentType of image
* @return encoded image meeting size requirements else null
*/
public static byte[] getResizedImageData(final int width, final int height,
final int orientation, final int widthLimit, final int heightLimit,
final int byteLimit, final Uri uri, final Context context,
final String contentType) {
final ImageResizer resizer = new ImageResizer(width, height, orientation,
widthLimit, heightLimit, byteLimit, uri, context, contentType);
return resizer.resize();
}
/**
* Create and initialize an image resizer
*/
private ImageResizer(final int width, final int height, final int orientation,
final int widthLimit, final int heightLimit, final int byteLimit, final Uri uri,
final Context context, final String contentType) {
mWidth = width;
mHeight = height;
mOrientationParams = ExifInterface.getOrientationParams(orientation);
mMatrix = new Matrix();
mWidthLimit = widthLimit;
mHeightLimit = heightLimit;
mByteLimit = byteLimit;
mUri = uri;
mWidth = width;
mContext = context;
mQuality = IMAGE_COMPRESSION_QUALITY;
mScaleFactor = 1.0f;
mHasReclaimedMemory = false;
mOptions = new BitmapFactory.Options();
mOptions.inScaled = false;
mOptions.inDensity = 0;
mOptions.inTargetDensity = 0;
mOptions.inSampleSize = 1;
mOptions.inJustDecodeBounds = false;
mOptions.inMutable = false;
final ActivityManager am =
(ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
mMemoryClass = Math.max(16, am.getMemoryClass());
mContentType = contentType;
}
/**
* Try to compress the image
*
* @return encoded image meeting size requirements else null
*/
private byte[] resize() {
return ImageUtils.isGif(mContentType, mUri) ? resizeGifImage() : resizeStaticImage();
}
private byte[] resizeGifImage() {
byte[] bytesToReturn = null;
final String inputFilePath;
if (MediaScratchFileProvider.isMediaScratchSpaceUri(mUri)) {
inputFilePath = MediaScratchFileProvider.getFileFromUri(mUri).getAbsolutePath();
} else {
if (!TextUtils.equals(mUri.getScheme(), ContentResolver.SCHEME_FILE)) {
Assert.fail("Expected a GIF file uri, but actual uri = " + mUri.toString());
}
inputFilePath = mUri.getPath();
}
if (GifTranscoder.canBeTranscoded(mWidth, mHeight)) {
// Needed to perform the transcoding so that the gif can continue to play in the
// conversation while the sending is taking place
final Uri tmpUri = MediaScratchFileProvider.buildMediaScratchSpaceUri("gif");
final File outputFile = MediaScratchFileProvider.getFileFromUri(tmpUri);
final String outputFilePath = outputFile.getAbsolutePath();
final boolean success =
GifTranscoder.transcode(mContext, inputFilePath, outputFilePath);
if (success) {
try {
bytesToReturn = Files.toByteArray(outputFile);
} catch (IOException e) {
LogUtil.e(TAG, "Could not create FileInputStream with path of "
+ outputFilePath, e);
}
}
// Need to clean up the new file created to compress the gif
mContext.getContentResolver().delete(tmpUri, null, null);
} else {
// We don't want to transcode the gif because its image dimensions would be too
// small so just return the bytes of the original gif
try {
bytesToReturn = Files.toByteArray(new File(inputFilePath));
} catch (IOException e) {
LogUtil.e(TAG,
"Could not create FileInputStream with path of " + inputFilePath, e);
}
}
return bytesToReturn;
}
private byte[] resizeStaticImage() {
if (!ensureImageSizeSet()) {
// Cannot read image size
return null;
}
// Find incoming image size
if (!canBeCompressed()) {
return null;
}
// Decode image - if out of memory - reclaim memory and retry
try {
for (int attempts = 0; attempts < NUMBER_OF_RESIZE_ATTEMPTS; attempts++) {
final byte[] encoded = recodeImage(attempts);
// Only return data within the limit
if (encoded != null && encoded.length <= mByteLimit) {
return encoded;
} else {
final int currentSize = (encoded == null ? 0 : encoded.length);
updateRecodeParameters(currentSize);
}
}
} catch (final FileNotFoundException e) {
LogUtil.e(TAG, "File disappeared during resizing");
} finally {
// Release all bitmaps
if (mScaled != null && mScaled != mDecoded) {
mScaled.recycle();
}
if (mDecoded != null) {
mDecoded.recycle();
}
}
return null;
}
/**
* Ensure that the width and height of the source image are known
* @return flag indicating whether size is known
*/
private boolean ensureImageSizeSet() {
if (mWidth == MessagingContentProvider.UNSPECIFIED_SIZE ||
mHeight == MessagingContentProvider.UNSPECIFIED_SIZE) {
// First get the image data (compressed)
final ContentResolver cr = mContext.getContentResolver();
InputStream inputStream = null;
// Find incoming image size
try {
mOptions.inJustDecodeBounds = true;
inputStream = cr.openInputStream(mUri);
BitmapFactory.decodeStream(inputStream, null, mOptions);
mWidth = mOptions.outWidth;
mHeight = mOptions.outHeight;
mOptions.inJustDecodeBounds = false;
return true;
} catch (final FileNotFoundException e) {
LogUtil.e(TAG, "Could not open file corresponding to uri " + mUri, e);
} catch (final NullPointerException e) {
LogUtil.e(TAG, "NPE trying to open the uri " + mUri, e);
} finally {
if (inputStream != null) {
try {
inputStream.close();
} catch (final IOException e) {
// Nothing to do
}
}
}
return false;
}
return true;
}
/**
* Choose an initial subsamplesize that ensures the decoded image is no more than
* MAX_TARGET_SCALE_FACTOR bigger than largest supported image and that it is likely to
* compress to smaller than the target size (assuming compression down to 1 bit per pixel).
* @return whether the image can be down subsampled
*/
private boolean canBeCompressed() {
final boolean logv = LogUtil.isLoggable(LogUtil.BUGLE_IMAGE_TAG, LogUtil.VERBOSE);
int imageHeight = mHeight;
int imageWidth = mWidth;
// Assume can use half working memory to decode the initial image (4 bytes per pixel)
final int workingMemoryPixelLimit = (mMemoryClass * 1024 * 1024 / 8);
// Target 1 bits per pixel in final compressed image
final int finalSizePixelLimit = mByteLimit * 8;
// When choosing to halve the resolution - only do so the image will still be too big
// after scaling by MAX_TARGET_SCALE_FACTOR
final int heightLimitWithSlop = (int) (mHeightLimit * MAX_TARGET_SCALE_FACTOR);
final int widthLimitWithSlop = (int) (mWidthLimit * MAX_TARGET_SCALE_FACTOR);
final int pixelLimitWithSlop = (int) (finalSizePixelLimit *
MAX_TARGET_SCALE_FACTOR * MAX_TARGET_SCALE_FACTOR);
final int pixelLimit = Math.min(pixelLimitWithSlop, workingMemoryPixelLimit);
int sampleSize = 1;
boolean fits = (imageHeight < heightLimitWithSlop &&
imageWidth < widthLimitWithSlop &&
imageHeight * imageWidth < pixelLimit);
// Compare sizes to compute sub-sampling needed
while (!fits) {
sampleSize = sampleSize * 2;
// Note that recodeImage may try using mSampleSize * 2. Hence we use the factor of 4
if (sampleSize >= (Integer.MAX_VALUE / 4)) {
LogUtil.w(LogUtil.BUGLE_IMAGE_TAG, String.format(
"Cannot resize image: widthLimit=%d heightLimit=%d byteLimit=%d " +
"imageWidth=%d imageHeight=%d", mWidthLimit, mHeightLimit, mByteLimit,
mWidth, mHeight));
Assert.fail("Image cannot be resized"); // http://b/18926934
return false;
}
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"computeInitialSampleSize: Increasing sampleSize to " + sampleSize
+ " as h=" + imageHeight + " vs " + heightLimitWithSlop
+ " w=" + imageWidth + " vs " + widthLimitWithSlop
+ " p=" + imageHeight * imageWidth + " vs " + pixelLimit);
}
imageHeight = mHeight / sampleSize;
imageWidth = mWidth / sampleSize;
fits = (imageHeight < heightLimitWithSlop &&
imageWidth < widthLimitWithSlop &&
imageHeight * imageWidth < pixelLimit);
}
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"computeInitialSampleSize: Initial sampleSize " + sampleSize
+ " for h=" + imageHeight + " vs " + heightLimitWithSlop
+ " w=" + imageWidth + " vs " + widthLimitWithSlop
+ " p=" + imageHeight * imageWidth + " vs " + pixelLimit);
}
mSampleSize = sampleSize;
return true;
}
/**
* Recode the image from initial Uri to encoded JPEG
* @param attempt Attempt number
* @return encoded image
*/
private byte[] recodeImage(final int attempt) throws FileNotFoundException {
byte[] encoded = null;
try {
final ContentResolver cr = mContext.getContentResolver();
final boolean logv = LogUtil.isLoggable(LogUtil.BUGLE_IMAGE_TAG, LogUtil.VERBOSE);
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG, "getResizedImageData: attempt=" + attempt
+ " limit (w=" + mWidthLimit + " h=" + mHeightLimit + ") quality="
+ mQuality + " scale=" + mScaleFactor + " sampleSize=" + mSampleSize);
}
if (mScaled == null) {
if (mDecoded == null) {
mOptions.inSampleSize = mSampleSize;
try (final InputStream inputStream = cr.openInputStream(mUri)) {
mDecoded = BitmapFactory.decodeStream(inputStream, null, mOptions);
} catch (IOException e) {
// Ignore
}
if (mDecoded == null) {
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: got empty decoded bitmap");
}
return null;
}
}
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG, "getResizedImageData: decoded w,h="
+ mDecoded.getWidth() + "," + mDecoded.getHeight());
}
// Make sure to scale the decoded image if dimension is not within limit
final int decodedWidth = mDecoded.getWidth();
final int decodedHeight = mDecoded.getHeight();
if (decodedWidth > mWidthLimit || decodedHeight > mHeightLimit) {
final float minScaleFactor = Math.max(
mWidthLimit == 0 ? 1.0f :
(float) decodedWidth / (float) mWidthLimit,
mHeightLimit == 0 ? 1.0f :
(float) decodedHeight / (float) mHeightLimit);
if (mScaleFactor < minScaleFactor) {
mScaleFactor = minScaleFactor;
}
}
if (mScaleFactor > 1.0 || mOrientationParams.rotation != 0) {
mMatrix.reset();
mMatrix.postRotate(mOrientationParams.rotation);
mMatrix.postScale(mOrientationParams.scaleX / mScaleFactor,
mOrientationParams.scaleY / mScaleFactor);
mScaled = Bitmap.createBitmap(mDecoded, 0, 0, decodedWidth, decodedHeight,
mMatrix, false /* filter */);
if (mScaled == null) {
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: got empty scaled bitmap");
}
return null;
}
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG, "getResizedImageData: scaled w,h="
+ mScaled.getWidth() + "," + mScaled.getHeight());
}
} else {
mScaled = mDecoded;
}
}
// Now encode it at current quality
encoded = ImageUtils.bitmapToBytes(mScaled, mQuality);
if (encoded != null && logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: Encoded down to " + encoded.length + "@"
+ mScaled.getWidth() + "/" + mScaled.getHeight() + "~"
+ mQuality);
}
} catch (final OutOfMemoryError e) {
LogUtil.w(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData - image too big (OutOfMemoryError), will try "
+ " with smaller scale factor");
// fall through and keep trying with more compression
}
return encoded;
}
/**
* When image recode fails this method updates compression parameters for the next attempt
* @param currentSize encoded image size (will be 0 if OOM)
*/
private void updateRecodeParameters(final int currentSize) {
final boolean logv = LogUtil.isLoggable(LogUtil.BUGLE_IMAGE_TAG, LogUtil.VERBOSE);
// Only return data within the limit
if (currentSize > 0 &&
mQuality > MINIMUM_IMAGE_COMPRESSION_QUALITY) {
// First if everything succeeded but failed to hit target size
// Try quality proportioned to sqrt of size over size limit
mQuality = Math.max(MINIMUM_IMAGE_COMPRESSION_QUALITY,
Math.min((int) (mQuality * Math.sqrt((1.0 * mByteLimit) / currentSize)),
(int) (mQuality * QUALITY_SCALE_DOWN_RATIO)));
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: Retrying at quality " + mQuality);
}
} else if (currentSize > 0 &&
mScaleFactor < 2.0 * MIN_SCALE_DOWN_RATIO * MIN_SCALE_DOWN_RATIO) {
// JPEG compression failed to hit target size - need smaller image
// First try scaling by a little (< factor of 2) just so long resulting scale down
// ratio is still significantly bigger than next subsampling step
// i.e. mScaleFactor/MIN_SCALE_DOWN_RATIO (new scaling factor) <
// 2.0 / MIN_SCALE_DOWN_RATIO (arbitrary limit)
mQuality = IMAGE_COMPRESSION_QUALITY;
mScaleFactor = mScaleFactor / MIN_SCALE_DOWN_RATIO;
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: Retrying at scale " + mScaleFactor);
}
// Release scaled bitmap to trigger rescaling
if (mScaled != null && mScaled != mDecoded) {
mScaled.recycle();
}
mScaled = null;
} else if (currentSize <= 0 && !mHasReclaimedMemory) {
// Then before we subsample try cleaning up our cached memory
Factory.get().reclaimMemory();
mHasReclaimedMemory = true;
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: Retrying after reclaiming memory ");
}
} else {
// Last resort - subsample image by another factor of 2 and try again
mSampleSize = mSampleSize * 2;
mQuality = IMAGE_COMPRESSION_QUALITY;
mScaleFactor = 1.0f;
if (logv) {
LogUtil.v(LogUtil.BUGLE_IMAGE_TAG,
"getResizedImageData: Retrying at sampleSize " + mSampleSize);
}
// Release all bitmaps to trigger subsampling
if (mScaled != null && mScaled != mDecoded) {
mScaled.recycle();
}
mScaled = null;
if (mDecoded != null) {
mDecoded.recycle();
mDecoded = null;
}
}
}
}
/**
* Scales and center-crops a bitmap to the size passed in and returns the new bitmap.
*
* @param source Bitmap to scale and center-crop
* @param newWidth destination width
* @param newHeight destination height
* @return Bitmap scaled and center-cropped bitmap
*/
public static Bitmap scaleCenterCrop(final Bitmap source, final int newWidth,
final int newHeight) {
final int sourceWidth = source.getWidth();
final int sourceHeight = source.getHeight();
// Compute the scaling factors to fit the new height and width, respectively.
// To cover the final image, the final scaling will be the bigger
// of these two.
final float xScale = (float) newWidth / sourceWidth;
final float yScale = (float) newHeight / sourceHeight;
final float scale = Math.max(xScale, yScale);
// Now get the size of the source bitmap when scaled
final float scaledWidth = scale * sourceWidth;
final float scaledHeight = scale * sourceHeight;
// Let's find out the upper left coordinates if the scaled bitmap
// should be centered in the new size give by the parameters
final float left = (newWidth - scaledWidth) / 2;
final float top = (newHeight - scaledHeight) / 2;
// The target rectangle for the new, scaled version of the source bitmap will now
// be
final RectF targetRect = new RectF(left, top, left + scaledWidth, top + scaledHeight);
// Finally, we create a new bitmap of the specified size and draw our new,
// scaled bitmap onto it.
final Bitmap dest = Bitmap.createBitmap(newWidth, newHeight, source.getConfig());
final Canvas canvas = new Canvas(dest);
canvas.drawBitmap(source, null, targetRect, null);
return dest;
}
/**
* The drawable can be a Nine-Patch. If we directly use the same drawable instance for each
* drawable of different sizes, then the drawable sizes would interfere with each other. The
* solution here is to create a new drawable instance for every time with the SAME
* ConstantState (i.e. sharing the same common state such as the bitmap, so that we don't have
* to recreate the bitmap resource), and apply the different properties on top (nine-patch
* size and color tint).
*
* TODO: we are creating new drawable instances here, but there are optimizations that
* can be made. For example, message bubbles shouldn't need the mutate() call and the
* play/pause buttons shouldn't need to create new drawable from the constant state.
*/
public static Drawable getTintedDrawable(final Context context, final Drawable drawable,
final int color) {
// For some reason occassionally drawables on JB has a null constant state
final Drawable.ConstantState constantStateDrawable = drawable.getConstantState();
final Drawable retDrawable = (constantStateDrawable != null)
? constantStateDrawable.newDrawable(context.getResources()).mutate()
: drawable;
retDrawable.setColorFilter(color, PorterDuff.Mode.SRC_ATOP);
return retDrawable;
}
/**
* Decodes image resource header and returns the image size.
*/
public static Rect decodeImageBounds(final Context context, final Uri imageUri) {
final ContentResolver cr = context.getContentResolver();
try {
final InputStream inputStream = cr.openInputStream(imageUri);
if (inputStream != null) {
try {
BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeStream(inputStream, null, options);
return new Rect(0, 0, options.outWidth, options.outHeight);
} finally {
try {
inputStream.close();
} catch (IOException e) {
// Do nothing.
}
}
}
} catch (FileNotFoundException e) {
LogUtil.e(TAG, "Couldn't open input stream for uri = " + imageUri);
}
return new Rect(0, 0, ImageRequest.UNSPECIFIED_SIZE, ImageRequest.UNSPECIFIED_SIZE);
}
}