summaryrefslogtreecommitdiff
path: root/src/de/danoeh/antennapod/asynctask/ImageLoader.java
blob: a4a9bc8239d5c9b29794ed02b7d7f1b67c5556f0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
package de.danoeh.antennapod.asynctask;

import java.io.InputStream;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;

import android.annotation.SuppressLint;
import android.app.ActivityManager;
import android.content.Context;
import android.content.res.TypedArray;
import android.os.Handler;
import android.support.v4.util.LruCache;
import android.util.Log;
import android.widget.ImageView;
import de.danoeh.antennapod.AppConfig;
import de.danoeh.antennapod.PodcastApp;
import de.danoeh.antennapod.R;

/** Caches and loads FeedImage bitmaps in the background */
public class ImageLoader {
	private static final String TAG = "ImageLoader";
	private static ImageLoader singleton;

	public static final int IMAGE_TYPE_THUMBNAIL = 0;
	public static final int IMAGE_TYPE_COVER = 1;

	private Handler handler;
	private ExecutorService executor;

	/**
	 * Stores references to loaded bitmaps. Bitmaps can be accessed by the id of
	 * the FeedImage the bitmap belongs to.
	 */

	final int memClass = ((ActivityManager) PodcastApp.getInstance()
			.getSystemService(Context.ACTIVITY_SERVICE)).getMemoryClass();

	// Use 1/8th of the available memory for this memory cache.
	final int thumbnailCacheSize = 1024 * 1024 * memClass / 8;

	private LruCache<String, CachedBitmap> coverCache;
	private LruCache<String, CachedBitmap> thumbnailCache;

	private ImageLoader() {
		handler = new Handler();
		executor = createExecutor();

		coverCache = new LruCache<String, CachedBitmap>(1);

		thumbnailCache = new LruCache<String, CachedBitmap>(thumbnailCacheSize) {

			@SuppressLint("NewApi")
			@Override
			protected int sizeOf(String key, CachedBitmap value) {
				if (Integer.valueOf(android.os.Build.VERSION.SDK_INT) >= 12)
					return value.getBitmap().getByteCount();
				else
					return (value.getBitmap().getRowBytes() * value.getBitmap()
							.getHeight());

			}

		};
	}

	private ExecutorService createExecutor() {
		return Executors.newFixedThreadPool(Runtime.getRuntime()
				.availableProcessors(), new ThreadFactory() {

			@Override
			public Thread newThread(Runnable r) {
				Thread t = new Thread(r);
				t.setPriority(Thread.MIN_PRIORITY);
				return t;
			}
		});
	}

	public static synchronized ImageLoader getInstance() {
		if (singleton == null) {
			singleton = new ImageLoader();
		}
		return singleton;
	}

	/**
	 * Load a bitmap from the cover cache. If the bitmap is not in the cache, it
	 * will be loaded from the disk. This method should either be called if the
	 * ImageView's size has already been set or inside a Runnable which is
	 * posted to the ImageView's message queue.
	 */
	public void loadCoverBitmap(ImageWorkerTaskResource source, ImageView target) {
		loadCoverBitmap(source, target, target.getHeight());
	}

	/**
	 * Load a bitmap from the cover cache. If the bitmap is not in the cache, it
	 * will be loaded from the disk. This method should either be called if the
	 * ImageView's size has already been set or inside a Runnable which is
	 * posted to the ImageView's message queue.
	 */
	public void loadCoverBitmap(ImageWorkerTaskResource source,
			ImageView target, int length) {
		final int defaultCoverResource = getDefaultCoverResource(target
				.getContext());

		if (source != null && source.getImageLoaderCacheKey() != null) {
            target.setTag(R.id.imageloader_key, source.getImageLoaderCacheKey());
            CachedBitmap cBitmap = getBitmapFromCoverCache(source.getImageLoaderCacheKey());
			if (cBitmap != null && cBitmap.getLength() >= length) {
				target.setImageBitmap(cBitmap.getBitmap());
			} else {
				target.setImageResource(defaultCoverResource);
				BitmapDecodeWorkerTask worker = new BitmapDecodeWorkerTask(
						handler, target, source, length, IMAGE_TYPE_COVER);
				executor.submit(worker);
			}
		} else {
			target.setImageResource(defaultCoverResource);
		}
	}

	/**
	 * Load a bitmap from the thumbnail cache. If the bitmap is not in the
	 * cache, it will be loaded from the disk. This method should either be
	 * called if the ImageView's size has already been set or inside a Runnable
	 * which is posted to the ImageView's message queue.
	 */
	public void loadThumbnailBitmap(ImageWorkerTaskResource source,
			ImageView target) {
		loadThumbnailBitmap(source, target, target.getHeight());
	}

	/**
	 * Load a bitmap from the thumbnail cache. If the bitmap is not in the
	 * cache, it will be loaded from the disk. This method should either be
	 * called if the ImageView's size has already been set or inside a Runnable
	 * which is posted to the ImageView's message queue.
	 */
	public void loadThumbnailBitmap(ImageWorkerTaskResource source,
			ImageView target, int length) {
		final int defaultCoverResource = getDefaultCoverResource(target
				.getContext());

		if (source != null && source.getImageLoaderCacheKey() != null) {
            target.setTag(R.id.imageloader_key, source.getImageLoaderCacheKey());
            CachedBitmap cBitmap = getBitmapFromThumbnailCache(source.getImageLoaderCacheKey());
			if (cBitmap != null && cBitmap.getLength() >= length) {
				target.setImageBitmap(cBitmap.getBitmap());
			} else {
				target.setImageResource(defaultCoverResource);
				BitmapDecodeWorkerTask worker = new BitmapDecodeWorkerTask(
						handler, target, source, length, IMAGE_TYPE_THUMBNAIL);
				executor.submit(worker);
			}
		} else {
			target.setImageResource(defaultCoverResource);
		}
	}

	public void clearExecutorQueue() {
		executor.shutdownNow();
		if (AppConfig.DEBUG)
			Log.d(TAG, "Executor was shut down.");
		executor = createExecutor();

	}

	public void wipeImageCache() {
		coverCache.evictAll();
		thumbnailCache.evictAll();
	}

	public boolean isInThumbnailCache(String fileUrl) {
		return thumbnailCache.get(fileUrl) != null;
	}

	private CachedBitmap getBitmapFromThumbnailCache(String key) {
		return thumbnailCache.get(key);
	}

	public void addBitmapToThumbnailCache(String key, CachedBitmap bitmap) {
		thumbnailCache.put(key, bitmap);
	}

	public boolean isInCoverCache(String fileUrl) {
		return coverCache.get(fileUrl) != null;
	}

	private CachedBitmap getBitmapFromCoverCache(String key) {
		return coverCache.get(key);
	}

	public void addBitmapToCoverCache(String key, CachedBitmap bitmap) {
		coverCache.put(key, bitmap);
	}

	private int getDefaultCoverResource(Context context) {
		return android.R.color.transparent;
	}

	/**
	 * Used by the BitmapDecodeWorker task to retrieve the source of the bitmap.
	 */
	public interface ImageWorkerTaskResource {
		/**
		 * Opens a new InputStream that can be decoded as a bitmap by the
		 * BitmapFactory.
		 */
		public InputStream openImageInputStream();

		/**
		 * Returns an InputStream that points to the beginning of the image
		 * resource. Implementations can either create a new InputStream or
		 * reset the existing one, depending on their implementation of
		 * openInputStream. If a new InputStream is returned, the one given as a
		 * parameter MUST be closed.
		 * @param input The input stream that was returned by openImageInputStream()
		 * */
		public InputStream reopenImageInputStream(InputStream input);

		/**
		 * Returns a string that identifies the image resource. Example: file
		 * path of an image
		 */
		public String getImageLoaderCacheKey();
	}

}