/* * Copyright (C) 2011 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.camera; import android.content.ContentResolver; import android.content.ContentUris; import android.database.Cursor; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Matrix; import android.media.MediaMetadataRetriever; import android.net.Uri; import android.provider.MediaStore.Images; import android.provider.MediaStore.Images.ImageColumns; import android.provider.MediaStore.MediaColumns; import android.provider.MediaStore.Video; import android.provider.MediaStore.Video.VideoColumns; import android.util.Log; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.File; import java.io.FileDescriptor; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; public class Thumbnail { private static final String TAG = "Thumbnail"; public static final String LAST_THUMB_FILENAME = "last_thumb"; private static final int BUFSIZE = 4096; private Uri mUri; private Bitmap mBitmap; // whether this thumbnail is read from file private boolean mFromFile = false; // Camera, VideoCamera, and Panorama share the same thumbnail. Use sLock // to serialize the access. private static Object sLock = new Object(); public Thumbnail(Uri uri, Bitmap bitmap, int orientation) { mUri = uri; mBitmap = rotateImage(bitmap, orientation); if (mBitmap == null) throw new IllegalArgumentException("null bitmap"); } public Uri getUri() { return mUri; } public Bitmap getBitmap() { return mBitmap; } public void setFromFile(boolean fromFile) { mFromFile = fromFile; } public boolean fromFile() { return mFromFile; } private static Bitmap rotateImage(Bitmap bitmap, int orientation) { if (orientation != 0) { // We only rotate the thumbnail once even if we get OOM. Matrix m = new Matrix(); m.setRotate(orientation, bitmap.getWidth() * 0.5f, bitmap.getHeight() * 0.5f); try { Bitmap rotated = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), m, true); // If the rotated bitmap is the original bitmap, then it // should not be recycled. if (rotated != bitmap) bitmap.recycle(); return rotated; } catch (Throwable t) { Log.w(TAG, "Failed to rotate thumbnail", t); } } return bitmap; } // Stores the bitmap to the specified file. public void saveTo(File file) { FileOutputStream f = null; BufferedOutputStream b = null; DataOutputStream d = null; synchronized (sLock) { try { f = new FileOutputStream(file); b = new BufferedOutputStream(f, BUFSIZE); d = new DataOutputStream(b); d.writeUTF(mUri.toString()); mBitmap.compress(Bitmap.CompressFormat.JPEG, 90, d); d.close(); } catch (IOException e) { Log.e(TAG, "Fail to store bitmap. path=" + file.getPath(), e); } finally { Util.closeSilently(f); Util.closeSilently(b); Util.closeSilently(d); } } } // Loads the data from the specified file. // Returns null if failure. public static Thumbnail loadFrom(File file) { Uri uri = null; Bitmap bitmap = null; FileInputStream f = null; BufferedInputStream b = null; DataInputStream d = null; synchronized (sLock) { try { f = new FileInputStream(file); b = new BufferedInputStream(f, BUFSIZE); d = new DataInputStream(b); uri = Uri.parse(d.readUTF()); bitmap = BitmapFactory.decodeStream(d); d.close(); } catch (IOException e) { Log.i(TAG, "Fail to load bitmap. " + e); return null; } finally { Util.closeSilently(f); Util.closeSilently(b); Util.closeSilently(d); } } Thumbnail thumbnail = createThumbnail(uri, bitmap, 0); if (thumbnail != null) thumbnail.setFromFile(true); return thumbnail; } public static Thumbnail getLastThumbnail(ContentResolver resolver) { Media image = getLastImageThumbnail(resolver); Media video = getLastVideoThumbnail(resolver); if (image == null && video == null) return null; Bitmap bitmap = null; Media lastMedia; // If there is only image or video, get its thumbnail. If both exist, // get the thumbnail of the one that is newer. if (image != null && (video == null || image.dateTaken >= video.dateTaken)) { bitmap = Images.Thumbnails.getThumbnail(resolver, image.id, Images.Thumbnails.MINI_KIND, null); lastMedia = image; } else { bitmap = Video.Thumbnails.getThumbnail(resolver, video.id, Video.Thumbnails.MINI_KIND, null); lastMedia = video; } // Ensure database and storage are in sync. if (Util.isUriValid(lastMedia.uri, resolver)) { return createThumbnail(lastMedia.uri, bitmap, lastMedia.orientation); } return null; } private static class Media { public Media(long id, int orientation, long dateTaken, Uri uri) { this.id = id; this.orientation = orientation; this.dateTaken = dateTaken; this.uri = uri; } public final long id; public final int orientation; public final long dateTaken; public final Uri uri; } public static Media getLastImageThumbnail(ContentResolver resolver) { Uri baseUri = Images.Media.EXTERNAL_CONTENT_URI; Uri query = baseUri.buildUpon().appendQueryParameter("limit", "1").build(); String[] projection = new String[] {ImageColumns._ID, ImageColumns.ORIENTATION, ImageColumns.DATE_TAKEN}; String selection = ImageColumns.MIME_TYPE + "='image/jpeg' AND " + ImageColumns.BUCKET_ID + '=' + Storage.BUCKET_ID; String order = ImageColumns.DATE_TAKEN + " DESC," + ImageColumns._ID + " DESC"; Cursor cursor = null; try { cursor = resolver.query(query, projection, selection, null, order); if (cursor != null && cursor.moveToFirst()) { long id = cursor.getLong(0); return new Media(id, cursor.getInt(1), cursor.getLong(2), ContentUris.withAppendedId(baseUri, id)); } } finally { if (cursor != null) { cursor.close(); } } return null; } private static Media getLastVideoThumbnail(ContentResolver resolver) { Uri baseUri = Video.Media.EXTERNAL_CONTENT_URI; Uri query = baseUri.buildUpon().appendQueryParameter("limit", "1").build(); String[] projection = new String[] {VideoColumns._ID, MediaColumns.DATA, VideoColumns.DATE_TAKEN}; String selection = VideoColumns.BUCKET_ID + '=' + Storage.BUCKET_ID; String order = VideoColumns.DATE_TAKEN + " DESC," + VideoColumns._ID + " DESC"; Cursor cursor = null; try { cursor = resolver.query(query, projection, selection, null, order); if (cursor != null && cursor.moveToFirst()) { Log.d(TAG, "getLastVideoThumbnail: " + cursor.getString(1)); long id = cursor.getLong(0); return new Media(id, 0, cursor.getLong(2), ContentUris.withAppendedId(baseUri, id)); } } finally { if (cursor != null) { cursor.close(); } } return null; } public static Thumbnail createThumbnail(byte[] jpeg, int orientation, int inSampleSize, Uri uri) { // Create the thumbnail. BitmapFactory.Options options = new BitmapFactory.Options(); options.inSampleSize = inSampleSize; Bitmap bitmap = BitmapFactory.decodeByteArray(jpeg, 0, jpeg.length, options); return createThumbnail(uri, bitmap, orientation); } public static Bitmap createVideoThumbnail(FileDescriptor fd, int targetWidth) { return createVideoThumbnail(null, fd, targetWidth); } public static Bitmap createVideoThumbnail(String filePath, int targetWidth) { return createVideoThumbnail(filePath, null, targetWidth); } private static Bitmap createVideoThumbnail(String filePath, FileDescriptor fd, int targetWidth) { Bitmap bitmap = null; MediaMetadataRetriever retriever = new MediaMetadataRetriever(); try { if (filePath != null) { retriever.setDataSource(filePath); } else { retriever.setDataSource(fd); } bitmap = retriever.getFrameAtTime(-1); } catch (IllegalArgumentException ex) { // Assume this is a corrupt video file } catch (RuntimeException ex) { // Assume this is a corrupt video file. } finally { try { retriever.release(); } catch (RuntimeException ex) { // Ignore failures while cleaning up. } } if (bitmap == null) return null; // Scale down the bitmap if it is bigger than we need. int width = bitmap.getWidth(); int height = bitmap.getHeight(); if (width > targetWidth) { float scale = (float) targetWidth / width; int w = Math.round(scale * width); int h = Math.round(scale * height); bitmap = Bitmap.createScaledBitmap(bitmap, w, h, true); } return bitmap; } private static Thumbnail createThumbnail(Uri uri, Bitmap bitmap, int orientation) { if (bitmap == null) { Log.e(TAG, "Failed to create thumbnail from null bitmap"); return null; } try { return new Thumbnail(uri, bitmap, orientation); } catch (IllegalArgumentException e) { Log.e(TAG, "Failed to construct thumbnail", e); return null; } } }