blob: 7775c372067681bb2539e5f5acab77cf468df5ab [file] [log] [blame]
/*
* 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;
}
}
}