Home | History | Annotate | Download | only in data
      1 /*
      2  * Copyright (C) 2010 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.gallery3d.data;
     18 
     19 import android.net.Uri;
     20 import android.provider.MediaStore;
     21 
     22 import com.android.gallery3d.common.ApiHelper;
     23 
     24 import java.lang.ref.SoftReference;
     25 import java.util.ArrayList;
     26 import java.util.Comparator;
     27 import java.util.NoSuchElementException;
     28 import java.util.SortedMap;
     29 import java.util.TreeMap;
     30 
     31 // MergeAlbum merges items from two or more MediaSets. It uses a Comparator to
     32 // determine the order of items. The items are assumed to be sorted in the input
     33 // media sets (with the same order that the Comparator uses).
     34 //
     35 // This only handles MediaItems, not SubMediaSets.
     36 public class LocalMergeAlbum extends MediaSet implements ContentListener {
     37     @SuppressWarnings("unused")
     38     private static final String TAG = "LocalMergeAlbum";
     39     private static final int PAGE_SIZE = 64;
     40 
     41     private final Comparator<MediaItem> mComparator;
     42     private final MediaSet[] mSources;
     43 
     44     private FetchCache[] mFetcher;
     45     private int mSupportedOperation;
     46     private int mBucketId;
     47 
     48     // mIndex maps global position to the position of each underlying media sets.
     49     private TreeMap<Integer, int[]> mIndex = new TreeMap<Integer, int[]>();
     50 
     51     public LocalMergeAlbum(
     52             Path path, Comparator<MediaItem> comparator, MediaSet[] sources, int bucketId) {
     53         super(path, INVALID_DATA_VERSION);
     54         mComparator = comparator;
     55         mSources = sources;
     56         mBucketId = bucketId;
     57         for (MediaSet set : mSources) {
     58             set.addContentListener(this);
     59         }
     60         reload();
     61     }
     62 
     63     @Override
     64     public boolean isCameraRoll() {
     65         if (mSources.length == 0) return false;
     66         for(MediaSet set : mSources) {
     67             if (!set.isCameraRoll()) return false;
     68         }
     69         return true;
     70     }
     71 
     72     private void updateData() {
     73         ArrayList<MediaSet> matches = new ArrayList<MediaSet>();
     74         int supported = mSources.length == 0 ? 0 : MediaItem.SUPPORT_ALL;
     75         mFetcher = new FetchCache[mSources.length];
     76         for (int i = 0, n = mSources.length; i < n; ++i) {
     77             mFetcher[i] = new FetchCache(mSources[i]);
     78             supported &= mSources[i].getSupportedOperations();
     79         }
     80         mSupportedOperation = supported;
     81         mIndex.clear();
     82         mIndex.put(0, new int[mSources.length]);
     83     }
     84 
     85     private void invalidateCache() {
     86         for (int i = 0, n = mSources.length; i < n; i++) {
     87             mFetcher[i].invalidate();
     88         }
     89         mIndex.clear();
     90         mIndex.put(0, new int[mSources.length]);
     91     }
     92 
     93     @Override
     94     public Uri getContentUri() {
     95         String bucketId = String.valueOf(mBucketId);
     96         if (ApiHelper.HAS_MEDIA_PROVIDER_FILES_TABLE) {
     97             return MediaStore.Files.getContentUri("external").buildUpon()
     98                     .appendQueryParameter(LocalSource.KEY_BUCKET_ID, bucketId)
     99                     .build();
    100         } else {
    101             // We don't have a single URL for a merged image before ICS
    102             // So we used the image's URL as a substitute.
    103             return MediaStore.Images.Media.EXTERNAL_CONTENT_URI.buildUpon()
    104                     .appendQueryParameter(LocalSource.KEY_BUCKET_ID, bucketId)
    105                     .build();
    106         }
    107     }
    108 
    109     @Override
    110     public String getName() {
    111         return mSources.length == 0 ? "" : mSources[0].getName();
    112     }
    113 
    114     @Override
    115     public int getMediaItemCount() {
    116         return getTotalMediaItemCount();
    117     }
    118 
    119     @Override
    120     public ArrayList<MediaItem> getMediaItem(int start, int count) {
    121 
    122         // First find the nearest mark position <= start.
    123         SortedMap<Integer, int[]> head = mIndex.headMap(start + 1);
    124         int markPos = head.lastKey();
    125         int[] subPos = head.get(markPos).clone();
    126         MediaItem[] slot = new MediaItem[mSources.length];
    127 
    128         int size = mSources.length;
    129 
    130         // fill all slots
    131         for (int i = 0; i < size; i++) {
    132             slot[i] = mFetcher[i].getItem(subPos[i]);
    133         }
    134 
    135         ArrayList<MediaItem> result = new ArrayList<MediaItem>();
    136 
    137         for (int i = markPos; i < start + count; i++) {
    138             int k = -1;  // k points to the best slot up to now.
    139             for (int j = 0; j < size; j++) {
    140                 if (slot[j] != null) {
    141                     if (k == -1 || mComparator.compare(slot[j], slot[k]) < 0) {
    142                         k = j;
    143                     }
    144                 }
    145             }
    146 
    147             // If we don't have anything, all streams are exhausted.
    148             if (k == -1) break;
    149 
    150             // Pick the best slot and refill it.
    151             subPos[k]++;
    152             if (i >= start) {
    153                 result.add(slot[k]);
    154             }
    155             slot[k] = mFetcher[k].getItem(subPos[k]);
    156 
    157             // Periodically leave a mark in the index, so we can come back later.
    158             if ((i + 1) % PAGE_SIZE == 0) {
    159                 mIndex.put(i + 1, subPos.clone());
    160             }
    161         }
    162 
    163         return result;
    164     }
    165 
    166     @Override
    167     public int getTotalMediaItemCount() {
    168         int count = 0;
    169         for (MediaSet set : mSources) {
    170             count += set.getTotalMediaItemCount();
    171         }
    172         return count;
    173     }
    174 
    175     @Override
    176     public long reload() {
    177         boolean changed = false;
    178         for (int i = 0, n = mSources.length; i < n; ++i) {
    179             if (mSources[i].reload() > mDataVersion) changed = true;
    180         }
    181         if (changed) {
    182             mDataVersion = nextVersionNumber();
    183             updateData();
    184             invalidateCache();
    185         }
    186         return mDataVersion;
    187     }
    188 
    189     @Override
    190     public void onContentDirty() {
    191         notifyContentChanged();
    192     }
    193 
    194     @Override
    195     public int getSupportedOperations() {
    196         return mSupportedOperation;
    197     }
    198 
    199     @Override
    200     public void delete() {
    201         for (MediaSet set : mSources) {
    202             set.delete();
    203         }
    204     }
    205 
    206     @Override
    207     public void rotate(int degrees) {
    208         for (MediaSet set : mSources) {
    209             set.rotate(degrees);
    210         }
    211     }
    212 
    213     private static class FetchCache {
    214         private MediaSet mBaseSet;
    215         private SoftReference<ArrayList<MediaItem>> mCacheRef;
    216         private int mStartPos;
    217 
    218         public FetchCache(MediaSet baseSet) {
    219             mBaseSet = baseSet;
    220         }
    221 
    222         public void invalidate() {
    223             mCacheRef = null;
    224         }
    225 
    226         public MediaItem getItem(int index) {
    227             boolean needLoading = false;
    228             ArrayList<MediaItem> cache = null;
    229             if (mCacheRef == null
    230                     || index < mStartPos || index >= mStartPos + PAGE_SIZE) {
    231                 needLoading = true;
    232             } else {
    233                 cache = mCacheRef.get();
    234                 if (cache == null) {
    235                     needLoading = true;
    236                 }
    237             }
    238 
    239             if (needLoading) {
    240                 cache = mBaseSet.getMediaItem(index, PAGE_SIZE);
    241                 mCacheRef = new SoftReference<ArrayList<MediaItem>>(cache);
    242                 mStartPos = index;
    243             }
    244 
    245             if (index < mStartPos || index >= mStartPos + cache.size()) {
    246                 return null;
    247             }
    248 
    249             return cache.get(index - mStartPos);
    250         }
    251     }
    252 
    253     @Override
    254     public boolean isLeafAlbum() {
    255         return true;
    256     }
    257 }
    258