blob: c620f448b3b7d8d3d47eb96353253ee1b6b961d4 [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.drawable.Drawable;
import android.net.Uri;
import android.os.Bundle;
import android.os.Parcelable;
import android.os.SystemClock;
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.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();
// Timeout for loading images from ImageCache when calling from UI thread
private static final long MAX_UI_THREAD_TIMEOUT_MS = 100L;
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 && !isLowRam();
}
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);
}
/**
* 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, or null if unable to load.
*/
Drawable resolveImage(Uri uri) {
try {
return LocalImageResolver.resolveImage(uri, mContext, mMaxImageWidth, mMaxImageHeight);
} catch (Exception ex) {
// Catch general Exception because ContentResolver can re-throw arbitrary Exception
// from remote process as a RuntimeException. See: Parcel#readException
Log.d(TAG, "resolveImage: Can't load image from " + uri, ex);
}
return null;
}
/**
* Loads an image from the Uri.
* This method is synchronous and is usually called from the Main thread.
* It will time-out after MAX_UI_THREAD_TIMEOUT_MS.
*
* @param uri Uri of the target image.
* @return drawable of the image, null if loading failed/timeout
*/
@Override
public Drawable loadImage(Uri uri) {
return hasCache() ? loadImageFromCache(uri, MAX_UI_THREAD_TIMEOUT_MS) : resolveImage(uri);
}
private Drawable loadImageFromCache(Uri uri, long timeoutMs) {
// if the uri isn't currently cached, try caching it first
if (!mImageCache.hasEntry(uri)) {
mImageCache.preload((uri));
}
return mImageCache.get(uri, timeoutMs);
}
/**
* 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;
}
/**
* Wait for a maximum timeout for images to finish preloading
* @param timeoutMs total timeout time
*/
void waitForPreloadedImages(long timeoutMs) {
if (!hasCache()) {
return;
}
Set<Uri> preloadedUris = getWantedUriSet();
if (preloadedUris != null) {
// Decrement remaining timeout after each image check
long endTimeMs = SystemClock.elapsedRealtime() + timeoutMs;
preloadedUris.forEach(
uri -> loadImageFromCache(uri, endTimeMs - SystemClock.elapsedRealtime()));
}
}
void cancelRunningTasks() {
if (!hasCache()) {
return;
}
mImageCache.cancelRunningTasks();
}
/**
* 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, long timeoutMs);
/**
* 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();
/**
* Cancel all unfinished image loading tasks
*/
void cancelRunningTasks();
}
}