/*
* 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.fastergallery.onetimeinitializer;
import android.content.Context;
import android.content.SharedPreferences;
import android.os.Build;
import android.os.Environment;
import android.preference.PreferenceManager;
import android.util.Log;
import com.android.fastergallery.app.GalleryApp;
import com.android.fastergallery.common.ApiHelper;
import com.android.fastergallery.data.DataManager;
import com.android.fastergallery.data.LocalAlbum;
import com.android.fastergallery.data.MediaSet;
import com.android.fastergallery.data.Path;
import com.android.fastergallery.gadget.WidgetDatabaseHelper;
import com.android.fastergallery.gadget.WidgetDatabaseHelper.Entry;
import com.android.fastergallery.util.GalleryUtils;
import java.io.File;
import java.util.HashMap;
import java.util.List;
/**
* This one-timer migrates local-album gallery app widgets from old paths from
* prior releases to updated paths in the current build version. This migration
* is needed because of bucket ID (i.e., directory hash) change in JB and JB MR1
* (The external storage path has changed from /mnt/sdcard in pre-JB releases,
* to /storage/sdcard0 in JB, then again to /external/storage/sdcard/0 in JB
* MR1).
*/
public class GalleryWidgetMigrator {
private static final String TAG = "GalleryWidgetMigrator";
private static final String PRE_JB_EXT_PATH = "/mnt/sdcard";
private static final String JB_EXT_PATH = "/storage/sdcard0";
private static final String NEW_EXT_PATH = Environment
.getExternalStorageDirectory().getAbsolutePath();
private static final int RELATIVE_PATH_START = NEW_EXT_PATH.length();
private static final String KEY_EXT_PATH = "external_storage_path";
/**
* Migrates local-album gallery widgets from prior releases to current
* release due to bucket ID (i.e., directory hash) change.
*/
public static void migrateGalleryWidgets(Context context) {
SharedPreferences prefs = PreferenceManager
.getDefaultSharedPreferences(context);
// Migration is only needed when external storage path has changed
String extPath = prefs.getString(KEY_EXT_PATH, null);
boolean isDone = NEW_EXT_PATH.equals(extPath);
if (isDone)
return;
try {
migrateGalleryWidgetsInternal(context);
prefs.edit().putString(KEY_EXT_PATH, NEW_EXT_PATH).commit();
} catch (Throwable t) {
// exception may be thrown if external storage is not available(?)
Log.w(TAG, "migrateGalleryWidgets", t);
}
}
private static void migrateGalleryWidgetsInternal(Context context) {
GalleryApp galleryApp = (GalleryApp) context.getApplicationContext();
DataManager manager = galleryApp.getDataManager();
WidgetDatabaseHelper dbHelper = new WidgetDatabaseHelper(context);
// only need to migrate local-album entries of type TYPE_ALBUM
List<Entry> entries = dbHelper
.getEntries(WidgetDatabaseHelper.TYPE_ALBUM);
if (entries == null)
return;
// Check each entry's relativePath. If exists, update bucket id using
// relative
// path combined with external storage path. Otherwise, iterate through
// old external
// storage paths to find the relative path that matches the old bucket
// id, and then update
// bucket id and relative path
HashMap<Integer, Entry> localEntries = new HashMap<Integer, Entry>(
entries.size());
for (Entry entry : entries) {
Path path = Path.fromString(entry.albumPath);
MediaSet mediaSet = (MediaSet) manager.getMediaObject(path);
if (mediaSet instanceof LocalAlbum) {
if (entry.relativePath != null
&& entry.relativePath.length() > 0) {
// update entry using relative path + external storage path
updateEntryUsingRelativePath(entry, dbHelper);
} else {
int bucketId = Integer.parseInt(path.getSuffix());
localEntries.put(bucketId, entry);
}
}
}
if (!localEntries.isEmpty())
migrateLocalEntries(context, localEntries, dbHelper);
}
private static void migrateLocalEntries(Context context,
HashMap<Integer, Entry> entries, WidgetDatabaseHelper dbHelper) {
SharedPreferences prefs = PreferenceManager
.getDefaultSharedPreferences(context);
String oldExtPath = prefs.getString(KEY_EXT_PATH, null);
if (oldExtPath != null) {
migrateLocalEntries(entries, dbHelper, oldExtPath);
return;
}
// If old external storage path is unknown, it could be either Pre-JB or
// JB version
// we need to try both.
migrateLocalEntries(entries, dbHelper, PRE_JB_EXT_PATH);
if (!entries.isEmpty()
&& Build.VERSION.SDK_INT > ApiHelper.VERSION_CODES.JELLY_BEAN) {
migrateLocalEntries(entries, dbHelper, JB_EXT_PATH);
}
}
private static void migrateLocalEntries(HashMap<Integer, Entry> entries,
WidgetDatabaseHelper dbHelper, String oldExtPath) {
File root = Environment.getExternalStorageDirectory();
// check the DCIM directory first; this should take care of 99% use
// cases
updatePath(new File(root, "DCIM"), entries, dbHelper, oldExtPath);
// check other directories if DCIM doesn't cut it
if (!entries.isEmpty())
updatePath(root, entries, dbHelper, oldExtPath);
}
private static void updatePath(File root, HashMap<Integer, Entry> entries,
WidgetDatabaseHelper dbHelper, String oldExtStorage) {
File[] files = root.listFiles();
if (files != null) {
for (File file : files) {
if (file.isDirectory() && !entries.isEmpty()) {
String path = file.getAbsolutePath();
String oldPath = oldExtStorage
+ path.substring(RELATIVE_PATH_START);
int oldBucketId = GalleryUtils.getBucketId(oldPath);
Entry entry = entries.remove(oldBucketId);
if (entry != null) {
int newBucketId = GalleryUtils.getBucketId(path);
String newAlbumPath = Path.fromString(entry.albumPath)
.getParent().getChild(newBucketId).toString();
Log.d(TAG, "migrate from " + entry.albumPath + " to "
+ newAlbumPath);
entry.albumPath = newAlbumPath;
// update entry's relative path
entry.relativePath = path
.substring(RELATIVE_PATH_START);
dbHelper.updateEntry(entry);
}
updatePath(file, entries, dbHelper, oldExtStorage); // recursion
}
}
}
}
private static void updateEntryUsingRelativePath(Entry entry,
WidgetDatabaseHelper dbHelper) {
String newPath = NEW_EXT_PATH + entry.relativePath;
int newBucketId = GalleryUtils.getBucketId(newPath);
String newAlbumPath = Path.fromString(entry.albumPath).getParent()
.getChild(newBucketId).toString();
entry.albumPath = newAlbumPath;
dbHelper.updateEntry(entry);
}
}