You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
312 lines
11 KiB
312 lines
11 KiB
/*
|
|
* 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 | IOException 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;
|
|
}
|
|
}
|
|
}
|