package org.thoughtcrime.securesms.mediasend; import android.content.Context; import android.database.Cursor; import android.net.Uri; import android.os.AsyncTask; import android.os.Build; import android.os.Environment; import android.provider.MediaStore.Images; import android.provider.MediaStore.Video; import android.support.annotation.NonNull; import android.support.annotation.Nullable; import android.support.annotation.WorkerThread; import com.annimon.stream.Stream; import org.thoughtcrime.securesms.R; import org.thoughtcrime.securesms.util.Util; import org.whispersystems.libsignal.util.guava.Optional; import java.io.File; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.LinkedList; import java.util.List; import java.util.Map; /** * Handles the retrieval of media present on the user's device. */ class MediaRepository { private static final String ALL_MEDIA_BUCKET_ID = "org.thoughtcrime.securesms.ALL_MEDIA"; /** * Retrieves a list of folders that contain media. */ void getFolders(@NonNull Context context, @NonNull Callback> callback) { AsyncTask.THREAD_POOL_EXECUTOR.execute(() -> callback.onComplete(getFolders(context))); } /** * Retrieves a list of media items (images and videos) that are present int he specified bucket. */ void getMediaInBucket(@NonNull Context context, @NonNull String bucketId, @NonNull Callback> callback) { AsyncTask.THREAD_POOL_EXECUTOR.execute(() -> callback.onComplete(getMediaInBucket(context, bucketId))); } @WorkerThread private @NonNull List getFolders(@NonNull Context context) { FolderResult imageFolders = getFolders(context, Images.Media.EXTERNAL_CONTENT_URI); FolderResult videoFolders = getFolders(context, Video.Media.EXTERNAL_CONTENT_URI); Map folders = new HashMap<>(imageFolders.getFolderData()); for (Map.Entry entry : videoFolders.getFolderData().entrySet()) { if (folders.containsKey(entry.getKey())) { folders.get(entry.getKey()).incrementCount(entry.getValue().getCount()); } else { folders.put(entry.getKey(), entry.getValue()); } } String cameraBucketId = imageFolders.getCameraBucketId() != null ? imageFolders.getCameraBucketId() : videoFolders.getCameraBucketId(); FolderData cameraFolder = cameraBucketId != null ? folders.remove(cameraBucketId) : null; List mediaFolders = Stream.of(folders.values()).map(folder -> new MediaFolder(folder.getThumbnail(), folder.getTitle(), folder.getCount(), folder.getBucketId(), MediaFolder.FolderType.NORMAL)) .sorted((o1, o2) -> o1.getTitle().toLowerCase().compareTo(o2.getTitle().toLowerCase())) .toList(); Uri allMediaThumbnail = imageFolders.getThumbnailTimestamp() > videoFolders.getThumbnailTimestamp() ? imageFolders.getThumbnail() : videoFolders.getThumbnail(); if (allMediaThumbnail != null) { int allMediaCount = Stream.of(mediaFolders).reduce(0, (count, folder) -> count + folder.getItemCount()); if (cameraFolder != null) { allMediaCount += cameraFolder.getCount(); } mediaFolders.add(0, new MediaFolder(allMediaThumbnail, context.getString(R.string.MediaRepository_all_media), allMediaCount, ALL_MEDIA_BUCKET_ID, MediaFolder.FolderType.NORMAL)); } if (cameraFolder != null) { mediaFolders.add(0, new MediaFolder(cameraFolder.getThumbnail(), cameraFolder.getTitle(), cameraFolder.getCount(), cameraFolder.getBucketId(), MediaFolder.FolderType.CAMERA)); } return mediaFolders; } @WorkerThread private @NonNull FolderResult getFolders(@NonNull Context context, @NonNull Uri contentUri) { String cameraPath = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM).getAbsolutePath() + File.separator + "Camera"; String cameraBucketId = null; Uri globalThumbnail = null; long thumbnailTimestamp = 0; Map folders = new HashMap<>(); String[] projection = new String[] { Images.Media.DATA, Images.Media.BUCKET_ID, Images.Media.BUCKET_DISPLAY_NAME, Images.Media.DATE_TAKEN }; String selection = Images.Media.DATA + " NOT NULL"; String sortBy = Images.Media.BUCKET_DISPLAY_NAME + " COLLATE NOCASE ASC, " + Images.Media.DATE_TAKEN + " DESC"; try (Cursor cursor = context.getContentResolver().query(contentUri, projection, selection, null, sortBy)) { while (cursor != null && cursor.moveToNext()) { String path = cursor.getString(cursor.getColumnIndexOrThrow(projection[0])); Uri thumbnail = Uri.fromFile(new File(path)); String bucketId = cursor.getString(cursor.getColumnIndexOrThrow(projection[1])); String title = cursor.getString(cursor.getColumnIndexOrThrow(projection[2])); long timestamp = cursor.getLong(cursor.getColumnIndexOrThrow(projection[3])); FolderData folder = Util.getOrDefault(folders, bucketId, new FolderData(thumbnail, title, bucketId)); folder.incrementCount(); folders.put(bucketId, folder); if (cameraBucketId == null && path.startsWith(cameraPath)) { cameraBucketId = bucketId; } if (timestamp > thumbnailTimestamp) { globalThumbnail = thumbnail; thumbnailTimestamp = timestamp; } } } return new FolderResult(cameraBucketId, globalThumbnail, thumbnailTimestamp, folders); } @WorkerThread private @NonNull List getMediaInBucket(@NonNull Context context, @NonNull String bucketId) { List images = getMediaInBucket(context, bucketId, Images.Media.EXTERNAL_CONTENT_URI); List videos = getMediaInBucket(context, bucketId, Video.Media.EXTERNAL_CONTENT_URI); List media = new ArrayList<>(images.size() + videos.size()); media.addAll(images); media.addAll(videos); Collections.sort(media, (o1, o2) -> Long.compare(o2.getDate(), o1.getDate())); return media; } @WorkerThread private @NonNull List getMediaInBucket(@NonNull Context context, @NonNull String bucketId, @NonNull Uri contentUri) { List media = new LinkedList<>(); String selection = Images.Media.BUCKET_ID + " = ? AND " + Images.Media.DATA + " NOT NULL"; String[] selectionArgs = new String[] { bucketId }; String sortBy = Images.Media.DATE_TAKEN + " DESC"; String[] projection = Build.VERSION.SDK_INT >= 16 ? new String[] { Images.Media._ID, Images.Media.MIME_TYPE, Images.Media.DATE_TAKEN, Images.Media.WIDTH, Images.Media.HEIGHT } : new String[] { Images.Media._ID, Images.Media.MIME_TYPE, Images.Media.DATE_TAKEN }; if (ALL_MEDIA_BUCKET_ID.equals(bucketId)) { selection = Images.Media.DATA + " NOT NULL"; selectionArgs = null; } try (Cursor cursor = context.getContentResolver().query(contentUri, projection, selection, selectionArgs, sortBy)) { while (cursor != null && cursor.moveToNext()) { Uri uri = Uri.withAppendedPath(contentUri, cursor.getString(cursor.getColumnIndexOrThrow(projection[0]))); String mimetype = cursor.getString(cursor.getColumnIndexOrThrow(projection[1])); long dateTaken = cursor.getLong(cursor.getColumnIndexOrThrow(projection[2])); int width = 0; int height = 0; if (Build.VERSION.SDK_INT >= 16) { width = cursor.getInt(cursor.getColumnIndexOrThrow(projection[3])); height = cursor.getInt(cursor.getColumnIndexOrThrow(projection[4])); } media.add(new Media(uri, mimetype, dateTaken, width, height, Optional.of(bucketId), Optional.absent())); } } return media; } private static class FolderResult { private final String cameraBucketId; private final Uri thumbnail; private final long thumbnailTimestamp; private final Map folderData; private FolderResult(@Nullable String cameraBucketId, @Nullable Uri thumbnail, long thumbnailTimestamp, @NonNull Map folderData) { this.cameraBucketId = cameraBucketId; this.thumbnail = thumbnail; this.thumbnailTimestamp = thumbnailTimestamp; this.folderData = folderData; } @Nullable String getCameraBucketId() { return cameraBucketId; } @Nullable Uri getThumbnail() { return thumbnail; } long getThumbnailTimestamp() { return thumbnailTimestamp; } @NonNull Map getFolderData() { return folderData; } } private static class FolderData { private final Uri thumbnail; private final String title; private final String bucketId; private int count; private FolderData(Uri thumbnail, String title, String bucketId) { this.thumbnail = thumbnail; this.title = title; this.bucketId = bucketId; } Uri getThumbnail() { return thumbnail; } String getTitle() { return title; } String getBucketId() { return bucketId; } int getCount() { return count; } void incrementCount() { incrementCount(1); } void incrementCount(int amount) { count += amount; } } interface Callback { void onComplete(@NonNull E result); } }