1 /* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 package com.android.dreams.phototable; 17 18 import android.content.ContentResolver; 19 import android.content.Context; 20 import android.content.SharedPreferences; 21 import android.content.res.Resources; 22 import android.database.Cursor; 23 import android.graphics.Bitmap; 24 import android.graphics.BitmapFactory; 25 import android.graphics.Matrix; 26 import android.net.Uri; 27 import android.util.Log; 28 29 import java.io.BufferedInputStream; 30 import java.io.FileNotFoundException; 31 import java.io.IOException; 32 import java.io.InputStream; 33 import java.util.Collection; 34 import java.util.Collections; 35 import java.util.HashMap; 36 import java.util.LinkedList; 37 import java.util.Random; 38 39 /** 40 * Picks a random image from a source of photos. 41 */ 42 public abstract class PhotoSource { 43 private static final String TAG = "PhotoTable.PhotoSource"; 44 private static final boolean DEBUG = false; 45 46 // This should be large enough for BitmapFactory to decode the header so 47 // that we can mark and reset the input stream to avoid duplicate network i/o 48 private static final int BUFFER_SIZE = 32 * 1024; 49 50 public class ImageData { 51 public String id; 52 public String url; 53 public int orientation; 54 55 protected String albumId; 56 protected Cursor cursor; 57 protected int position; 58 protected Uri uri; 59 getStream(int longSide)60 InputStream getStream(int longSide) { 61 return PhotoSource.this.getStream(this, longSide); 62 } naturalNext()63 ImageData naturalNext() { 64 return PhotoSource.this.naturalNext(this); 65 } naturalPrevious()66 ImageData naturalPrevious() { 67 return PhotoSource.this.naturalPrevious(this); 68 } donePaging()69 public void donePaging() { 70 PhotoSource.this.donePaging(this); 71 } 72 } 73 74 public class AlbumData { 75 public String id; 76 public String title; 77 public String thumbnailUrl; 78 public String account; 79 public long updated; 80 getType()81 public String getType() { 82 String type = PhotoSource.this.getClass().getName(); 83 log(TAG, "type is " + type); 84 return type; 85 } 86 } 87 88 private final LinkedList<ImageData> mImageQueue; 89 private final int mMaxQueueSize; 90 private final float mMaxCropRatio; 91 private final int mBadImageSkipLimit; 92 private final PhotoSource mFallbackSource; 93 private final HashMap<Bitmap, ImageData> mImageMap; 94 95 protected final Context mContext; 96 protected final Resources mResources; 97 protected final Random mRNG; 98 protected final AlbumSettings mSettings; 99 protected final ContentResolver mResolver; 100 101 protected String mSourceName; 102 PhotoSource(Context context, SharedPreferences settings)103 public PhotoSource(Context context, SharedPreferences settings) { 104 this(context, settings, new StockSource(context, settings)); 105 } 106 PhotoSource(Context context, SharedPreferences settings, PhotoSource fallbackSource)107 public PhotoSource(Context context, SharedPreferences settings, PhotoSource fallbackSource) { 108 mSourceName = TAG; 109 mContext = context; 110 mSettings = AlbumSettings.getAlbumSettings(settings); 111 mResolver = mContext.getContentResolver(); 112 mResources = context.getResources(); 113 mImageQueue = new LinkedList<ImageData>(); 114 mMaxQueueSize = mResources.getInteger(R.integer.image_queue_size); 115 mMaxCropRatio = mResources.getInteger(R.integer.max_crop_ratio) / 1000000f; 116 mBadImageSkipLimit = mResources.getInteger(R.integer.bad_image_skip_limit); 117 mImageMap = new HashMap<Bitmap, ImageData>(); 118 mRNG = new Random(); 119 mFallbackSource = fallbackSource; 120 } 121 fillQueue()122 protected void fillQueue() { 123 log(TAG, "filling queue"); 124 mImageQueue.addAll(findImages(mMaxQueueSize - mImageQueue.size())); 125 Collections.shuffle(mImageQueue); 126 log(TAG, "queue contains: " + mImageQueue.size() + " items."); 127 } 128 next(BitmapFactory.Options options, int longSide, int shortSide)129 public Bitmap next(BitmapFactory.Options options, int longSide, int shortSide) { 130 log(TAG, "decoding a picasa resource to " + longSide + ", " + shortSide); 131 Bitmap image = null; 132 ImageData imageData = null; 133 int tries = 0; 134 135 while (image == null && tries < mBadImageSkipLimit) { 136 synchronized(mImageQueue) { 137 if (mImageQueue.isEmpty()) { 138 fillQueue(); 139 } 140 imageData = mImageQueue.poll(); 141 } 142 if (imageData != null) { 143 image = load(imageData, options, longSide, shortSide); 144 mImageMap.put(image, imageData); 145 imageData = null; 146 } 147 148 tries++; 149 } 150 151 if (image == null && mFallbackSource != null) { 152 image = load((ImageData) mFallbackSource.findImages(1).toArray()[0], 153 options, longSide, shortSide); 154 } 155 156 return image; 157 } 158 load(ImageData data, BitmapFactory.Options options, int longSide, int shortSide)159 public Bitmap load(ImageData data, BitmapFactory.Options options, int longSide, int shortSide) { 160 log(TAG, "decoding photo resource to " + longSide + ", " + shortSide); 161 InputStream is = data.getStream(longSide); 162 163 Bitmap image = null; 164 try { 165 BufferedInputStream bis = new BufferedInputStream(is); 166 bis.mark(BUFFER_SIZE); 167 168 options.inJustDecodeBounds = true; 169 options.inSampleSize = 1; 170 image = BitmapFactory.decodeStream(new BufferedInputStream(bis), null, options); 171 int rawLongSide = Math.max(options.outWidth, options.outHeight); 172 int rawShortSide = Math.min(options.outWidth, options.outHeight); 173 log(TAG, "I see bounds of " + rawLongSide + ", " + rawShortSide); 174 175 if (rawLongSide != -1 && rawShortSide != -1) { 176 float insideRatio = Math.max((float) longSide / (float) rawLongSide, 177 (float) shortSide / (float) rawShortSide); 178 float outsideRatio = Math.max((float) longSide / (float) rawLongSide, 179 (float) shortSide / (float) rawShortSide); 180 float ratio = (outsideRatio / insideRatio < mMaxCropRatio ? 181 outsideRatio : insideRatio); 182 183 while (ratio < 0.5) { 184 options.inSampleSize *= 2; 185 ratio *= 2; 186 } 187 188 log(TAG, "decoding with inSampleSize " + options.inSampleSize); 189 try { 190 bis.reset(); 191 } catch (IOException ioe) { 192 // start over, something went wrong and we read too far into the image. 193 bis.close(); 194 is = data.getStream(longSide); 195 bis = new BufferedInputStream(is); 196 log(TAG, "resetting the stream"); 197 } 198 options.inJustDecodeBounds = false; 199 image = BitmapFactory.decodeStream(bis, null, options); 200 rawLongSide = Math.max(options.outWidth, options.outHeight); 201 rawShortSide = Math.max(options.outWidth, options.outHeight); 202 if (image != null && rawLongSide != -1 && rawShortSide != -1) { 203 ratio = Math.max((float) longSide / (float) rawLongSide, 204 (float) shortSide / (float) rawShortSide); 205 206 if (Math.abs(ratio - 1.0f) > 0.001) { 207 log(TAG, "still too big, scaling down by " + ratio); 208 options.outWidth = (int) (ratio * options.outWidth); 209 options.outHeight = (int) (ratio * options.outHeight); 210 211 image = Bitmap.createScaledBitmap(image, 212 options.outWidth, options.outHeight, 213 true); 214 } 215 216 if (data.orientation != 0) { 217 log(TAG, "rotated by " + data.orientation + ": fixing"); 218 Matrix matrix = new Matrix(); 219 matrix.setRotate(data.orientation, 220 (float) Math.floor(image.getWidth() / 2f), 221 (float) Math.floor(image.getHeight() / 2f)); 222 image = Bitmap.createBitmap(image, 0, 0, 223 options.outWidth, options.outHeight, 224 matrix, true); 225 if (data.orientation == 90 || data.orientation == 270) { 226 int tmp = options.outWidth; 227 options.outWidth = options.outHeight; 228 options.outHeight = tmp; 229 } 230 } 231 232 log(TAG, "returning bitmap " + image.getWidth() + ", " + image.getHeight()); 233 } else { 234 image = null; 235 } 236 } else { 237 image = null; 238 } 239 if (image == null) { 240 log(TAG, "Stream decoding failed with no error" + 241 (options.mCancel ? " due to cancelation." : ".")); 242 } 243 } catch (OutOfMemoryError ome) { 244 log(TAG, "OUT OF MEMORY: " + ome); 245 image = null; 246 } catch (FileNotFoundException fnf) { 247 log(TAG, "file not found: " + fnf); 248 image = null; 249 } catch (IOException ioe) { 250 log(TAG, "i/o exception: " + ioe); 251 image = null; 252 } finally { 253 try { 254 if (is != null) { 255 is.close(); 256 } 257 } catch (Throwable t) { 258 log(TAG, "close fail: " + t.toString()); 259 } 260 } 261 262 return image; 263 } 264 setSeed(long seed)265 public void setSeed(long seed) { 266 mRNG.setSeed(seed); 267 } 268 log(String tag, String message)269 protected static void log(String tag, String message) { 270 if (DEBUG) { 271 Log.i(tag, message); 272 } 273 } 274 pickRandomStart(int total, int max)275 protected int pickRandomStart(int total, int max) { 276 if (max >= total) { 277 return -1; 278 } else { 279 return mRNG.nextInt(total - max) - 1; 280 } 281 } 282 naturalNext(Bitmap current, BitmapFactory.Options options, int longSide, int shortSide)283 public Bitmap naturalNext(Bitmap current, BitmapFactory.Options options, 284 int longSide, int shortSide) { 285 Bitmap image = null; 286 ImageData data = mImageMap.get(current); 287 if (data != null) { 288 ImageData next = data.naturalNext(); 289 if (next != null) { 290 image = load(next, options, longSide, shortSide); 291 mImageMap.put(image, next); 292 } 293 } 294 return image; 295 } 296 naturalPrevious(Bitmap current, BitmapFactory.Options options, int longSide, int shortSide)297 public Bitmap naturalPrevious(Bitmap current, BitmapFactory.Options options, 298 int longSide, int shortSide) { 299 Bitmap image = null; 300 ImageData data = mImageMap.get(current); 301 if (current != null) { 302 ImageData prev = data.naturalPrevious(); 303 if (prev != null) { 304 image = load(prev, options, longSide, shortSide); 305 mImageMap.put(image, prev); 306 } 307 } 308 return image; 309 } 310 donePaging(Bitmap current)311 public void donePaging(Bitmap current) { 312 ImageData data = mImageMap.get(current); 313 if (data != null) { 314 data.donePaging(); 315 } 316 } 317 recycle(Bitmap trash)318 public void recycle(Bitmap trash) { 319 if (trash != null) { 320 mImageMap.remove(trash); 321 trash.recycle(); 322 } 323 } 324 getStream(ImageData data, int longSide)325 protected abstract InputStream getStream(ImageData data, int longSide); findImages(int howMany)326 protected abstract Collection<ImageData> findImages(int howMany); naturalNext(ImageData current)327 protected abstract ImageData naturalNext(ImageData current); naturalPrevious(ImageData current)328 protected abstract ImageData naturalPrevious(ImageData current); donePaging(ImageData current)329 protected abstract void donePaging(ImageData current); 330 findAlbums()331 public abstract Collection<AlbumData> findAlbums(); 332 } 333