blob: 52f7c2cfee96c284c0ce12fdbacad0634f30fbbd [file] [log] [blame]
/*
* Copyright (C) 2018 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.systemui.statusbar.notification.row;
import android.app.ActivityManager;
import android.app.Notification;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.Icon;
import android.net.Uri;
import android.os.Bundle;
import android.os.Parcelable;
import android.util.Log;
import com.android.internal.R;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.widget.ImageResolver;
import com.android.internal.widget.LocalImageResolver;
import com.android.internal.widget.MessagingMessage;
import java.io.IOException;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
/**
* Custom resolver with built-in image cache for image messages.
*
* If the URL points to a bitmap that's larger than the maximum width or height, the bitmap
* will be resized down to that maximum size before being cached. See {@link #getMaxImageWidth()},
* {@link #getMaxImageHeight()}, and {@link #resolveImage(Uri)} for the downscaling implementation.
*/
public class NotificationInlineImageResolver implements ImageResolver {
private static final String TAG = NotificationInlineImageResolver.class.getSimpleName();
private final Context mContext;
private final ImageCache mImageCache;
private Set<Uri> mWantedUriSet;
// max allowed bitmap width, in pixels
@VisibleForTesting
protected int mMaxImageWidth;
// max allowed bitmap height, in pixels
@VisibleForTesting
protected int mMaxImageHeight;
/**
* Constructor.
* @param context Context.
* @param imageCache The implementation of internal cache.
*/
public NotificationInlineImageResolver(Context context, ImageCache imageCache) {
mContext = context.getApplicationContext();
mImageCache = imageCache;
if (mImageCache != null) {
mImageCache.setImageResolver(this);
}
updateMaxImageSizes();
}
/**
* Check if this resolver has its internal cache implementation.
* @return True if has its internal cache, false otherwise.
*/
public boolean hasCache() {
return mImageCache != null && !ActivityManager.isLowRamDeviceStatic();
}
private boolean isLowRam() {
return ActivityManager.isLowRamDeviceStatic();
}
/**
* Update the maximum width and height allowed for bitmaps, ex. after a configuration change.
*/
public void updateMaxImageSizes() {
mMaxImageWidth = getMaxImageWidth();
mMaxImageHeight = getMaxImageHeight();
}
@VisibleForTesting
protected int getMaxImageWidth() {
return mContext.getResources().getDimensionPixelSize(isLowRam()
? R.dimen.notification_custom_view_max_image_width_low_ram
: R.dimen.notification_custom_view_max_image_width);
}
@VisibleForTesting
protected int getMaxImageHeight() {
return mContext.getResources().getDimensionPixelSize(isLowRam()
? R.dimen.notification_custom_view_max_image_height_low_ram
: R.dimen.notification_custom_view_max_image_height);
}
@VisibleForTesting
protected BitmapDrawable resolveImageInternal(Uri uri) throws IOException {
return (BitmapDrawable) LocalImageResolver.resolveImage(uri, mContext);
}
/**
* To resolve image from specified uri directly. If the resulting image is larger than the
* maximum allowed size, scale it down.
* @param uri Uri of the image.
* @return Drawable of the image.
* @throws IOException Throws if failed at resolving the image.
*/
Drawable resolveImage(Uri uri) throws IOException {
BitmapDrawable image = resolveImageInternal(uri);
Bitmap bitmap = image.getBitmap();
image.setBitmap(Icon.scaleDownIfNecessary(bitmap, mMaxImageWidth, mMaxImageHeight));
return image;
}
@Override
public Drawable loadImage(Uri uri) {
Drawable result = null;
try {
if (hasCache()) {
// if the uri isn't currently cached, try caching it first
if (!mImageCache.hasEntry(uri)) {
mImageCache.preload((uri));
}
result = mImageCache.get(uri);
} else {
result = resolveImage(uri);
}
} catch (IOException | SecurityException ex) {
Log.d(TAG, "loadImage: Can't load image from " + uri, ex);
}
return result;
}
/**
* Resolve the message list from specified notification and
* refresh internal cache according to the result.
* @param notification The Notification to be resolved.
*/
public void preloadImages(Notification notification) {
if (!hasCache()) {
return;
}
retrieveWantedUriSet(notification);
Set<Uri> wantedSet = getWantedUriSet();
wantedSet.forEach(uri -> {
if (!mImageCache.hasEntry(uri)) {
// The uri is not in the cache, we need trigger a loading task for it.
mImageCache.preload(uri);
}
});
}
/**
* Try to purge unnecessary cache entries.
*/
public void purgeCache() {
if (!hasCache()) {
return;
}
mImageCache.purge();
}
private void retrieveWantedUriSet(Notification notification) {
Parcelable[] messages;
Parcelable[] historicMessages;
List<Notification.MessagingStyle.Message> messageList;
List<Notification.MessagingStyle.Message> historicList;
Set<Uri> result = new HashSet<>();
Bundle extras = notification.extras;
if (extras == null) {
return;
}
messages = extras.getParcelableArray(Notification.EXTRA_MESSAGES);
messageList = messages == null ? null :
Notification.MessagingStyle.Message.getMessagesFromBundleArray(messages);
if (messageList != null) {
for (Notification.MessagingStyle.Message message : messageList) {
if (MessagingMessage.hasImage(message)) {
result.add(message.getDataUri());
}
}
}
historicMessages = extras.getParcelableArray(Notification.EXTRA_HISTORIC_MESSAGES);
historicList = historicMessages == null ? null :
Notification.MessagingStyle.Message.getMessagesFromBundleArray(historicMessages);
if (historicList != null) {
for (Notification.MessagingStyle.Message historic : historicList) {
if (MessagingMessage.hasImage(historic)) {
result.add(historic.getDataUri());
}
}
}
mWantedUriSet = result;
}
Set<Uri> getWantedUriSet() {
return mWantedUriSet;
}
/**
* A interface for internal cache implementation of this resolver.
*/
interface ImageCache {
/**
* Load the image from cache first then resolve from uri if missed the cache.
* @param uri The uri of the image.
* @return Drawable of the image.
*/
Drawable get(Uri uri);
/**
* Set the image resolver that actually resolves image from specified uri.
* @param resolver The resolver implementation that resolves image from specified uri.
*/
void setImageResolver(NotificationInlineImageResolver resolver);
/**
* Check if the uri is in the cache no matter it is loading or loaded.
* @param uri The uri to check.
* @return True if it is already in the cache; false otherwise.
*/
boolean hasEntry(Uri uri);
/**
* Start a new loading task for the target uri.
* @param uri The target to load.
*/
void preload(Uri uri);
/**
* Purge unnecessary entries in the cache.
*/
void purge();
}
}