blob: cf2e0ecd4b39e81b2396a24146110622da160297 [file] [log] [blame]
/*
* Copyright (C) 2012 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.dreams.phototable;
import android.content.Context;
import android.content.SharedPreferences;
import android.database.Cursor;
import android.provider.MediaStore;
import java.io.FileInputStream;
import java.io.InputStream;
import java.util.Collection;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.Set;
/**
* Loads images from the local store.
*/
public class LocalSource extends CursorPhotoSource {
private static final String TAG = "PhotoTable.LocalSource";
private final String mUnknownAlbumName;
private final String mLocalSourceName;
private Set<String> mFoundAlbumIds;
private int mLastPosition;
public LocalSource(Context context, SharedPreferences settings) {
super(context, settings);
mLocalSourceName = mResources.getString(R.string.local_source_name, "Photos on Device");
mUnknownAlbumName = mResources.getString(R.string.unknown_album_name, "Unknown");
mSourceName = TAG;
mLastPosition = INVALID;
fillQueue();
}
private Set<String> getFoundAlbums() {
if (mFoundAlbumIds == null) {
findAlbums();
}
return mFoundAlbumIds;
}
@Override
public Collection<AlbumData> findAlbums() {
log(TAG, "finding albums");
HashMap<String, AlbumData> foundAlbums = new HashMap<String, AlbumData>();
String[] projection = {MediaStore.Images.Media.DATA, MediaStore.Images.Media.BUCKET_ID,
MediaStore.Images.Media.BUCKET_DISPLAY_NAME, MediaStore.Images.Media.DATE_TAKEN};
// This is a horrible hack that closes the where clause and injects a grouping clause.
Cursor cursor = mResolver.query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
projection, null, null, null);
if (cursor != null) {
cursor.moveToPosition(-1);
int dataIndex = cursor.getColumnIndex(MediaStore.Images.Media.DATA);
int bucketIndex = cursor.getColumnIndex(MediaStore.Images.Media.BUCKET_ID);
int nameIndex = cursor.getColumnIndex(MediaStore.Images.Media.BUCKET_DISPLAY_NAME);
int updatedIndex = cursor.getColumnIndex(MediaStore.Images.Media.DATE_TAKEN);
if (bucketIndex < 0) {
log(TAG, "can't find the ID column!");
} else {
while (cursor.moveToNext()) {
String id = TAG + ":" + cursor.getString(bucketIndex);
AlbumData data = foundAlbums.get(id);
if (foundAlbums.get(id) == null) {
data = new AlbumData();
data.id = id;
data.account = mLocalSourceName;
if (dataIndex >= 0) {
data.thumbnailUrl = cursor.getString(dataIndex);
}
if (nameIndex >= 0) {
data.title = cursor.getString(nameIndex);
} else {
data.title = mUnknownAlbumName;
}
log(TAG, data.title + " found");
foundAlbums.put(id, data);
}
if (updatedIndex >= 0) {
long updated = cursor.getLong(updatedIndex);
data.updated = (data.updated == 0 ?
updated :
Math.min(data.updated, updated));
}
}
}
cursor.close();
}
log(TAG, "found " + foundAlbums.size() + " items.");
mFoundAlbumIds = foundAlbums.keySet();
return foundAlbums.values();
}
@Override
protected void openCursor(ImageData data) {
log(TAG, "opening single album");
String[] projection = {MediaStore.Images.Media.DATA, MediaStore.Images.Media.ORIENTATION,
MediaStore.Images.Media.BUCKET_ID, MediaStore.Images.Media.BUCKET_DISPLAY_NAME};
String selection = MediaStore.Images.Media.BUCKET_ID + " = '" + data.albumId + "'";
data.cursor = mResolver.query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
projection, selection, null, null);
}
@Override
protected void findPosition(ImageData data) {
if (data.position == -1) {
if (data.cursor == null) {
openCursor(data);
}
if (data.cursor != null) {
int dataIndex = data.cursor.getColumnIndex(MediaStore.Images.Media.DATA);
data.cursor.moveToPosition(-1);
while (data.position == -1 && data.cursor.moveToNext()) {
String url = data.cursor.getString(dataIndex);
if (url != null && url.equals(data.url)) {
data.position = data.cursor.getPosition();
}
}
if (data.position == -1) {
// oops! The image isn't in this album. How did we get here?
data.position = INVALID;
}
}
}
}
@Override
protected ImageData unpackImageData(Cursor cursor, ImageData data) {
if (data == null) {
data = new ImageData();
}
int dataIndex = cursor.getColumnIndex(MediaStore.Images.Media.DATA);
int orientationIndex = cursor.getColumnIndex(MediaStore.Images.Media.ORIENTATION);
int bucketIndex = cursor.getColumnIndex(MediaStore.Images.Media.BUCKET_ID);
data.url = cursor.getString(dataIndex);
data.albumId = cursor.getString(bucketIndex);
data.position = UNINITIALIZED;
data.cursor = null;
data.orientation = cursor.getInt(orientationIndex);
return data;
}
@Override
protected Collection<ImageData> findImages(int howMany) {
log(TAG, "finding images");
LinkedList<ImageData> foundImages = new LinkedList<ImageData>();
String[] projection = {MediaStore.Images.Media.DATA, MediaStore.Images.Media.ORIENTATION,
MediaStore.Images.Media.BUCKET_ID, MediaStore.Images.Media.BUCKET_DISPLAY_NAME};
String selection = "";
for (String id : getFoundAlbums()) {
if (mSettings.isAlbumEnabled(id)) {
String[] parts = id.split(":");
if (parts.length > 1) {
if (selection.length() > 0) {
selection += " OR ";
}
selection += MediaStore.Images.Media.BUCKET_ID + " = '" + parts[1] + "'";
}
}
}
if (selection.isEmpty()) {
return foundImages;
}
Cursor cursor = mResolver.query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
projection, selection, null, null);
if (cursor != null) {
int dataIndex = cursor.getColumnIndex(MediaStore.Images.Media.DATA);
if (cursor.getCount() > howMany && mLastPosition == INVALID) {
mLastPosition = pickRandomStart(cursor.getCount(), howMany);
}
cursor.moveToPosition(mLastPosition);
if (dataIndex < 0) {
log(TAG, "can't find the DATA column!");
} else {
while (foundImages.size() < howMany && cursor.moveToNext()) {
ImageData data = unpackImageData(cursor, null);
foundImages.offer(data);
mLastPosition = cursor.getPosition();
}
if (cursor.isAfterLast()) {
mLastPosition = -1;
}
if (cursor.isBeforeFirst()) {
mLastPosition = INVALID;
}
}
cursor.close();
}
log(TAG, "found " + foundImages.size() + " items.");
return foundImages;
}
@Override
protected InputStream getStream(ImageData data, int longSide) {
FileInputStream fis = null;
try {
log(TAG, "opening:" + data.url);
fis = new FileInputStream(data.url);
} catch (Exception ex) {
log(TAG, ex.toString());
fis = null;
}
return (InputStream) fis;
}
}