Returning to wizard, enable migration.

Bring primary storage migration back into the adoption flow, and
provide a path for long-lived notifications to re-launch into the
Settings app.  Also provide option to initiate migration if skipped
during wizard.  For now, estmiate migration size and time based on
a Class 10 card.

Follow other callback refactoring.

Bug: 19993667
Change-Id: Ia0c28eb114bc6c8066c17b3142ed74f962140c91
This commit is contained in:
Jeff Sharkey
2015-04-29 11:24:57 -07:00
parent a03f330af3
commit e77f0687dd
17 changed files with 295 additions and 80 deletions

View File

@@ -1489,8 +1489,17 @@
android:value="true" /> android:value="true" />
</activity> </activity>
<activity <activity android:name="Settings$PrivateVolumeSettingsActivity"
android:name="Settings$PublicVolumeSettingsActivity" android:label="@string/storage_settings_title"
android:taskAffinity="com.android.settings"
android:parentActivityName="Settings$StorageSettingsActivity">
<meta-data android:name="com.android.settings.FRAGMENT_CLASS"
android:value="com.android.settings.deviceinfo.PrivateVolumeSettings" />
<meta-data android:name="com.android.settings.TOP_LEVEL_HEADER_ID"
android:resource="@id/storage_settings" />
</activity>
<activity android:name="Settings$PublicVolumeSettingsActivity"
android:label="@string/storage_settings_title" android:label="@string/storage_settings_title"
android:taskAffinity="com.android.settings" android:taskAffinity="com.android.settings"
android:parentActivityName="Settings$StorageSettingsActivity"> android:parentActivityName="Settings$StorageSettingsActivity">
@@ -1508,8 +1517,7 @@
android:resource="@id/storage_settings" /> android:resource="@id/storage_settings" />
</activity> </activity>
<activity <activity android:name="Settings$PrivateVolumeForgetActivity"
android:name="Settings$PrivateVolumeForgetActivity"
android:label="@string/storage_settings_title" android:label="@string/storage_settings_title"
android:taskAffinity="com.android.settings" android:taskAffinity="com.android.settings"
android:parentActivityName="Settings$StorageSettingsActivity" android:parentActivityName="Settings$StorageSettingsActivity"
@@ -1541,17 +1549,20 @@
android:exported="false" /> android:exported="false" />
<activity android:name=".deviceinfo.StorageWizardMigrateProgress" <activity android:name=".deviceinfo.StorageWizardMigrateProgress"
android:taskAffinity="com.android.settings.storage_wizard" android:taskAffinity="com.android.settings.storage_wizard"
android:exported="false" /> android:exported="true"
android:permission="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<activity android:name=".deviceinfo.StorageWizardReady" <activity android:name=".deviceinfo.StorageWizardReady"
android:taskAffinity="com.android.settings.storage_wizard" android:taskAffinity="com.android.settings.storage_wizard"
android:exported="false" /> android:exported="true"
android:permission="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<activity android:name=".deviceinfo.StorageWizardMoveConfirm" <activity android:name=".deviceinfo.StorageWizardMoveConfirm"
android:taskAffinity="com.android.settings.storage_wizard" android:taskAffinity="com.android.settings.storage_wizard"
android:exported="false" /> android:exported="false" />
<activity android:name=".deviceinfo.StorageWizardMoveProgress" <activity android:name=".deviceinfo.StorageWizardMoveProgress"
android:taskAffinity="com.android.settings.storage_wizard" android:taskAffinity="com.android.settings.storage_wizard"
android:exported="false" /> android:exported="true"
android:permission="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<!-- Exported for SystemUI to trigger --> <!-- Exported for SystemUI to trigger -->
<receiver android:name=".deviceinfo.StorageUnmountReceiver" <receiver android:name=".deviceinfo.StorageUnmountReceiver"

View File

@@ -27,6 +27,9 @@
<item <item
android:id="@+id/storage_format" android:id="@+id/storage_format"
android:title="@string/storage_menu_format" /> android:title="@string/storage_menu_format" />
<item
android:id="@+id/storage_migrate"
android:title="@string/storage_menu_migrate" />
<item <item
android:id="@+id/storage_usb" android:id="@+id/storage_usb"
android:title="@string/storage_menu_usb" /> android:title="@string/storage_menu_usb" />

View File

@@ -2285,10 +2285,12 @@
<string name="storage_menu_format">Erase &amp; format</string> <string name="storage_menu_format">Erase &amp; format</string>
<!-- Storage setting. Menu option for erasing and formatting a storage device [CHAR LIMIT=30]--> <!-- Storage setting. Menu option for erasing and formatting a storage device [CHAR LIMIT=30]-->
<string name="storage_menu_format_internal">Erase &amp; format as internal storage</string> <string name="storage_menu_format_internal">Erase &amp; format as internal storage</string>
<!-- Storage setting. Menu option for USB transfer settings [CHAR LIMIT=30]--> <!-- Storage setting. Menu option for migrating data to a storage device [CHAR LIMIT=30]-->
<string name="storage_menu_usb">USB computer connection</string> <string name="storage_menu_migrate">Migrate data</string>
<!-- Storage setting. Menu option for forgetting a storage device [CHAR LIMIT=30]--> <!-- Storage setting. Menu option for forgetting a storage device [CHAR LIMIT=30]-->
<string name="storage_menu_forget">Forget</string> <string name="storage_menu_forget">Forget</string>
<!-- Storage setting. Menu option for USB transfer settings [CHAR LIMIT=30]-->
<string name="storage_menu_usb">USB computer connection</string>
<!-- Storage setting. Title for USB transfer settings [CHAR LIMIT=30]--> <!-- Storage setting. Title for USB transfer settings [CHAR LIMIT=30]-->
<string name="storage_title_usb">USB computer connection</string> <string name="storage_title_usb">USB computer connection</string>
@@ -2402,7 +2404,7 @@
<!-- Title of wizard step prompting user to start data migration [CHAR LIMIT=32] --> <!-- Title of wizard step prompting user to start data migration [CHAR LIMIT=32] -->
<string name="storage_wizard_migrate_confirm_title">Move data now</string> <string name="storage_wizard_migrate_confirm_title">Move data now</string>
<!-- Body of wizard step providing details about data migration [CHAR LIMIT=NONE] --> <!-- Body of wizard step providing details about data migration [CHAR LIMIT=NONE] -->
<string name="storage_wizard_migrate_confirm_body"><b>The move takes about <xliff:g id="time" example="1 hour">^1</xliff:g>. It will free <xliff:g id="size" example="1.2 GB">^2</xliff:g> of internal storage.</b></string> <string name="storage_wizard_migrate_confirm_body"><b>The move takes about <xliff:g id="time" example="1 hour">^1</xliff:g>. It will free <xliff:g id="size" example="1.2 GB">^2</xliff:g> on <xliff:g id="name" example="Internal storage">^3</xliff:g>.</b></string>
<!-- Title of wizard button prompting user to start data migration [CHAR LIMIT=32] --> <!-- Title of wizard button prompting user to start data migration [CHAR LIMIT=32] -->
<string name="storage_wizard_migrate_confirm_next">Move</string> <string name="storage_wizard_migrate_confirm_next">Move</string>

View File

@@ -33,8 +33,9 @@ public class Settings extends SettingsActivity {
public static class VpnSettingsActivity extends SettingsActivity { /* empty */ } public static class VpnSettingsActivity extends SettingsActivity { /* empty */ }
public static class DateTimeSettingsActivity extends SettingsActivity { /* empty */ } public static class DateTimeSettingsActivity extends SettingsActivity { /* empty */ }
public static class StorageSettingsActivity extends SettingsActivity { /* empty */ } public static class StorageSettingsActivity extends SettingsActivity { /* empty */ }
public static class PublicVolumeSettingsActivity extends SettingsActivity { /* empty */ }
public static class PrivateVolumeForgetActivity extends SettingsActivity { /* empty */ } public static class PrivateVolumeForgetActivity extends SettingsActivity { /* empty */ }
public static class PrivateVolumeSettingsActivity extends SettingsActivity { /* empty */ }
public static class PublicVolumeSettingsActivity extends SettingsActivity { /* empty */ }
public static class WifiSettingsActivity extends SettingsActivity { /* empty */ } public static class WifiSettingsActivity extends SettingsActivity { /* empty */ }
public static class WifiP2pSettingsActivity extends SettingsActivity { /* empty */ } public static class WifiP2pSettingsActivity extends SettingsActivity { /* empty */ }
public static class InputMethodAndLanguageSettingsActivity extends SettingsActivity { /* empty */ } public static class InputMethodAndLanguageSettingsActivity extends SettingsActivity { /* empty */ }

View File

@@ -84,6 +84,7 @@ import com.android.settings.dashboard.DashboardTile;
import com.android.settings.dashboard.NoHomeDialogFragment; import com.android.settings.dashboard.NoHomeDialogFragment;
import com.android.settings.dashboard.SearchResultsSummary; import com.android.settings.dashboard.SearchResultsSummary;
import com.android.settings.deviceinfo.PrivateVolumeForget; import com.android.settings.deviceinfo.PrivateVolumeForget;
import com.android.settings.deviceinfo.PrivateVolumeSettings;
import com.android.settings.deviceinfo.PublicVolumeSettings; import com.android.settings.deviceinfo.PublicVolumeSettings;
import com.android.settings.deviceinfo.StorageSettings; import com.android.settings.deviceinfo.StorageSettings;
import com.android.settings.deviceinfo.UsbSettings; import com.android.settings.deviceinfo.UsbSettings;
@@ -310,8 +311,9 @@ public class SettingsActivity extends Activity
com.android.settings.accessibility.ToggleDaltonizerPreferenceFragment.class.getName(), com.android.settings.accessibility.ToggleDaltonizerPreferenceFragment.class.getName(),
TextToSpeechSettings.class.getName(), TextToSpeechSettings.class.getName(),
StorageSettings.class.getName(), StorageSettings.class.getName(),
PublicVolumeSettings.class.getName(),
PrivateVolumeForget.class.getName(), PrivateVolumeForget.class.getName(),
PrivateVolumeSettings.class.getName(),
PublicVolumeSettings.class.getName(),
DevelopmentSettings.class.getName(), DevelopmentSettings.class.getName(),
UsbSettings.class.getName(), UsbSettings.class.getName(),
AndroidBeam.class.getName(), AndroidBeam.class.getName(),

View File

@@ -0,0 +1,126 @@
/*
* Copyright (C) 2015 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.settings.deviceinfo;
import static com.android.settings.deviceinfo.StorageSettings.TAG;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.net.TrafficStats;
import android.os.AsyncTask;
import android.os.IBinder;
import android.os.RemoteException;
import android.os.UserHandle;
import android.os.storage.StorageManager;
import android.os.storage.VolumeInfo;
import android.telecom.Log;
import android.text.format.DateUtils;
import android.text.format.Formatter;
import com.android.internal.app.IMediaContainerService;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
public abstract class MigrateEstimateTask extends AsyncTask<Void, Void, Long> implements
ServiceConnection {
private static final String EXTRA_SIZE_BYTES = "size_bytes";
private static final ComponentName DEFAULT_CONTAINER_COMPONENT = new ComponentName(
"com.android.defcontainer", "com.android.defcontainer.DefaultContainerService");
/**
* Assume roughly a Class 10 card.
*/
private static final long SPEED_ESTIMATE_BPS = 10 * TrafficStats.MB_IN_BYTES;
private final Context mContext;
private final StorageManager mStorage;
private final CountDownLatch mConnected = new CountDownLatch(1);
private IMediaContainerService mService;
private long mSizeBytes = -1;
private long mTimeMillis = -1;
public MigrateEstimateTask(Context context) {
mContext = context;
mStorage = context.getSystemService(StorageManager.class);
}
public void copyFrom(Intent intent) {
mSizeBytes = intent.getLongExtra(EXTRA_SIZE_BYTES, -1);
}
public void copyTo(Intent intent) {
intent.putExtra(EXTRA_SIZE_BYTES, mSizeBytes);
}
@Override
protected Long doInBackground(Void... params) {
if (mSizeBytes != -1) {
return mSizeBytes;
}
final VolumeInfo privateVol = mContext.getPackageManager().getPrimaryStorageCurrentVolume();
final VolumeInfo emulatedVol = mStorage.findEmulatedForPrivate(privateVol);
final String path = emulatedVol.getPath().getAbsolutePath();
Log.d(TAG, "Estimating for current path " + path);
final Intent intent = new Intent().setComponent(DEFAULT_CONTAINER_COMPONENT);
mContext.bindServiceAsUser(intent, this, Context.BIND_AUTO_CREATE, UserHandle.OWNER);
try {
if (mConnected.await(15, TimeUnit.SECONDS)) {
return mService.calculateDirectorySize(path);
}
} catch (InterruptedException | RemoteException e) {
Log.w(TAG, "Failed to measure " + path);
} finally {
mContext.unbindService(this);
}
return -1L;
}
@Override
protected void onPostExecute(Long result) {
mSizeBytes = result;
mTimeMillis = (mSizeBytes * DateUtils.SECOND_IN_MILLIS) / SPEED_ESTIMATE_BPS;
final String size = Formatter.formatFileSize(mContext, mSizeBytes);
final String time = DateUtils.formatDuration(mTimeMillis).toString();
onPostExecute(size, time);
}
public abstract void onPostExecute(String size, String time);
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
mService = IMediaContainerService.Stub.asInterface(service);
mConnected.countDown();
}
@Override
public void onServiceDisconnected(ComponentName name) {
// Ignored; we leave service in place for the background thread to
// run into DeadObjectException
}
}

View File

@@ -67,6 +67,7 @@ public class PrivateVolumeFormat extends InstrumentedFragment {
final Intent intent = new Intent(getActivity(), StorageWizardFormatProgress.class); final Intent intent = new Intent(getActivity(), StorageWizardFormatProgress.class);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId());
intent.putExtra(StorageWizardFormatConfirm.EXTRA_FORMAT_PRIVATE, false); intent.putExtra(StorageWizardFormatConfirm.EXTRA_FORMAT_PRIVATE, false);
intent.putExtra(StorageWizardFormatConfirm.EXTRA_FORGET_UUID, mVolume.getFsUuid());
startActivity(intent); startActivity(intent);
getActivity().finish(); getActivity().finish();
} }

View File

@@ -34,6 +34,7 @@ import android.os.Bundle;
import android.os.Environment; import android.os.Environment;
import android.os.UserHandle; import android.os.UserHandle;
import android.os.UserManager; import android.os.UserManager;
import android.os.storage.DiskInfo;
import android.os.storage.StorageEventListener; import android.os.storage.StorageEventListener;
import android.os.storage.StorageManager; import android.os.storage.StorageManager;
import android.os.storage.VolumeInfo; import android.os.storage.VolumeInfo;
@@ -271,6 +272,7 @@ public class PrivateVolumeSettings extends SettingsPreferenceFragment {
final MenuItem mount = menu.findItem(R.id.storage_mount); final MenuItem mount = menu.findItem(R.id.storage_mount);
final MenuItem unmount = menu.findItem(R.id.storage_unmount); final MenuItem unmount = menu.findItem(R.id.storage_unmount);
final MenuItem format = menu.findItem(R.id.storage_format); final MenuItem format = menu.findItem(R.id.storage_format);
final MenuItem migrate = menu.findItem(R.id.storage_migrate);
final MenuItem usb = menu.findItem(R.id.storage_usb); final MenuItem usb = menu.findItem(R.id.storage_usb);
// Actions live in menu for non-internal private volumes; they're shown // Actions live in menu for non-internal private volumes; they're shown
@@ -287,6 +289,11 @@ public class PrivateVolumeSettings extends SettingsPreferenceFragment {
format.setVisible(true); format.setVisible(true);
} }
// Only offer to migrate when not current storage
final VolumeInfo privateVol = getActivity().getPackageManager()
.getPrimaryStorageCurrentVolume();
migrate.setVisible(!Objects.equals(mVolume, privateVol));
// TODO: show usb if we jumped past first screen // TODO: show usb if we jumped past first screen
usb.setVisible(false); usb.setVisible(false);
} }
@@ -312,6 +319,11 @@ public class PrivateVolumeSettings extends SettingsPreferenceFragment {
startFragment(this, PrivateVolumeFormat.class.getCanonicalName(), startFragment(this, PrivateVolumeFormat.class.getCanonicalName(),
R.string.storage_menu_format, 0, args); R.string.storage_menu_format, 0, args);
return true; return true;
case R.id.storage_migrate:
final Intent intent = new Intent(context, StorageWizardMigrateConfirm.class);
intent.putExtra(VolumeInfo.EXTRA_VOLUME_ID, mVolume.getId());
startActivity(intent);
return true;
case R.id.storage_usb: case R.id.storage_usb:
startFragment(this, UsbSettings.class.getCanonicalName(), startFragment(this, UsbSettings.class.getCanonicalName(),
R.string.storage_title_usb, 0, null); R.string.storage_title_usb, 0, null);
@@ -442,8 +454,8 @@ public class PrivateVolumeSettings extends SettingsPreferenceFragment {
} }
@Override @Override
public void onVolumeMetadataChanged(String fsUuid) { public void onVolumeRecordChanged(VolumeRecord rec) {
if (Objects.equals(mVolume.getFsUuid(), fsUuid)) { if (Objects.equals(mVolume.getFsUuid(), rec.getFsUuid())) {
mVolume = mStorageManager.findVolumeById(mVolumeId); mVolume = mStorageManager.findVolumeById(mVolumeId);
update(); update();
} }

View File

@@ -24,6 +24,7 @@ import android.os.storage.DiskInfo;
import android.os.storage.StorageEventListener; import android.os.storage.StorageEventListener;
import android.os.storage.StorageManager; import android.os.storage.StorageManager;
import android.os.storage.VolumeInfo; import android.os.storage.VolumeInfo;
import android.os.storage.VolumeRecord;
import android.preference.Preference; import android.preference.Preference;
import android.preference.PreferenceScreen; import android.preference.PreferenceScreen;
import android.provider.DocumentsContract; import android.provider.DocumentsContract;
@@ -220,8 +221,8 @@ public class PublicVolumeSettings extends SettingsPreferenceFragment {
} }
@Override @Override
public void onVolumeMetadataChanged(String fsUuid) { public void onVolumeRecordChanged(VolumeRecord rec) {
if (Objects.equals(mVolume.getFsUuid(), fsUuid)) { if (Objects.equals(mVolume.getFsUuid(), rec.getFsUuid())) {
mVolume = mStorageManager.findVolumeById(mVolumeId); mVolume = mStorageManager.findVolumeById(mVolumeId);
update(); update();
} }

View File

@@ -35,6 +35,8 @@ import com.android.setupwizardlib.view.NavigationBar;
import com.android.setupwizardlib.view.NavigationBar.NavigationBarListener; import com.android.setupwizardlib.view.NavigationBar.NavigationBarListener;
import java.text.NumberFormat; import java.text.NumberFormat;
import java.util.List;
import java.util.Objects;
public abstract class StorageWizardBase extends Activity implements NavigationBarListener { public abstract class StorageWizardBase extends Activity implements NavigationBarListener {
protected StorageManager mStorage; protected StorageManager mStorage;
@@ -56,7 +58,7 @@ public abstract class StorageWizardBase extends Activity implements NavigationBa
final String diskId = getIntent().getStringExtra(DiskInfo.EXTRA_DISK_ID); final String diskId = getIntent().getStringExtra(DiskInfo.EXTRA_DISK_ID);
if (!TextUtils.isEmpty(diskId)) { if (!TextUtils.isEmpty(diskId)) {
mDisk = mStorage.findDiskById(diskId); mDisk = mStorage.findDiskById(diskId);
} else { } else if (mVolume != null) {
mDisk = mVolume.getDisk(); mDisk = mVolume.getDisk();
} }
@@ -130,4 +132,14 @@ public abstract class StorageWizardBase extends Activity implements NavigationBa
public void onNavigateNext() { public void onNavigateNext() {
throw new UnsupportedOperationException(); throw new UnsupportedOperationException();
} }
protected VolumeInfo findFirstVolume(int type) {
final List<VolumeInfo> vols = mStorage.getVolumes();
for (VolumeInfo vol : vols) {
if (Objects.equals(mDisk.getId(), vol.getDiskId()) && (vol.getType() == type)) {
return vol;
}
}
return null;
}
} }

View File

@@ -25,6 +25,7 @@ import com.android.settings.R;
public class StorageWizardFormatConfirm extends StorageWizardBase { public class StorageWizardFormatConfirm extends StorageWizardBase {
public static final String EXTRA_FORMAT_PRIVATE = "format_private"; public static final String EXTRA_FORMAT_PRIVATE = "format_private";
public static final String EXTRA_FORGET_UUID = "forget_uuid";
private boolean mFormatPrivate; private boolean mFormatPrivate;
@@ -56,6 +57,7 @@ public class StorageWizardFormatConfirm extends StorageWizardBase {
final Intent intent = new Intent(this, StorageWizardFormatProgress.class); final Intent intent = new Intent(this, StorageWizardFormatProgress.class);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId());
intent.putExtra(EXTRA_FORMAT_PRIVATE, mFormatPrivate); intent.putExtra(EXTRA_FORMAT_PRIVATE, mFormatPrivate);
intent.putExtra(EXTRA_FORGET_UUID, getIntent().getStringExtra(EXTRA_FORGET_UUID));
startActivity(intent); startActivity(intent);
finishAffinity(); finishAffinity();
} }

View File

@@ -23,6 +23,8 @@ import android.content.Intent;
import android.os.AsyncTask; import android.os.AsyncTask;
import android.os.Bundle; import android.os.Bundle;
import android.os.storage.DiskInfo; import android.os.storage.DiskInfo;
import android.os.storage.VolumeInfo;
import android.text.TextUtils;
import android.util.Log; import android.util.Log;
import android.view.View; import android.view.View;
import android.widget.Toast; import android.widget.Toast;
@@ -72,12 +74,25 @@ public class StorageWizardFormatProgress extends StorageWizardBase {
protected void onPostExecute(Exception e) { protected void onPostExecute(Exception e) {
final Context context = StorageWizardFormatProgress.this; final Context context = StorageWizardFormatProgress.this;
if (e == null) { if (e == null) {
final String forgetUuid = getIntent().getStringExtra(
StorageWizardFormatConfirm.EXTRA_FORGET_UUID);
if (!TextUtils.isEmpty(forgetUuid)) {
mStorage.forgetVolume(forgetUuid);
}
final boolean offerMigrate;
if (mFormatPrivate) { if (mFormatPrivate) {
// TODO: bring back migration once implemented // Offer to migrate only if storage is currently internal
// final Intent intent = new Intent(context, StorageWizardMigrate.class); final VolumeInfo privateVol = getPackageManager()
// intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); .getPrimaryStorageCurrentVolume();
// startActivity(intent); offerMigrate = (privateVol != null
final Intent intent = new Intent(context, StorageWizardReady.class); && VolumeInfo.ID_PRIVATE_INTERNAL.equals(privateVol.getId()));
} else {
offerMigrate = false;
}
if (offerMigrate) {
final Intent intent = new Intent(context, StorageWizardMigrate.class);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId());
startActivity(intent); startActivity(intent);
} else { } else {

View File

@@ -19,8 +19,6 @@ package com.android.settings.deviceinfo;
import android.content.Intent; import android.content.Intent;
import android.os.Bundle; import android.os.Bundle;
import android.os.storage.DiskInfo; import android.os.storage.DiskInfo;
import android.text.format.DateUtils;
import android.text.format.Formatter;
import android.widget.CompoundButton; import android.widget.CompoundButton;
import android.widget.CompoundButton.OnCheckedChangeListener; import android.widget.CompoundButton.OnCheckedChangeListener;
import android.widget.RadioButton; import android.widget.RadioButton;
@@ -29,6 +27,8 @@ import com.android.internal.util.Preconditions;
import com.android.settings.R; import com.android.settings.R;
public class StorageWizardMigrate extends StorageWizardBase { public class StorageWizardMigrate extends StorageWizardBase {
private MigrateEstimateTask mEstimate;
private RadioButton mRadioNow; private RadioButton mRadioNow;
private RadioButton mRadioLater; private RadioButton mRadioLater;
@@ -39,11 +39,8 @@ public class StorageWizardMigrate extends StorageWizardBase {
Preconditions.checkNotNull(mDisk); Preconditions.checkNotNull(mDisk);
final String time = DateUtils.formatDuration(0).toString();
final String size = Formatter.formatFileSize(this, 0);
setHeaderText(R.string.storage_wizard_migrate_title, mDisk.getDescription()); setHeaderText(R.string.storage_wizard_migrate_title, mDisk.getDescription());
setBodyText(R.string.storage_wizard_migrate_body, mDisk.getDescription(), time, size); setBodyText(R.string.memory_calculating_size);
mRadioNow = (RadioButton) findViewById(R.id.storage_wizard_migrate_now); mRadioNow = (RadioButton) findViewById(R.id.storage_wizard_migrate_now);
mRadioLater = (RadioButton) findViewById(R.id.storage_wizard_migrate_later); mRadioLater = (RadioButton) findViewById(R.id.storage_wizard_migrate_later);
@@ -52,6 +49,17 @@ public class StorageWizardMigrate extends StorageWizardBase {
mRadioLater.setOnCheckedChangeListener(mRadioListener); mRadioLater.setOnCheckedChangeListener(mRadioListener);
mRadioNow.setChecked(true); mRadioNow.setChecked(true);
mEstimate = new MigrateEstimateTask(this) {
@Override
public void onPostExecute(String size, String time) {
setBodyText(R.string.storage_wizard_migrate_body,
mDisk.getDescription(), time, size);
}
};
mEstimate.copyFrom(getIntent());
mEstimate.execute();
} }
private final OnCheckedChangeListener mRadioListener = new OnCheckedChangeListener() { private final OnCheckedChangeListener mRadioListener = new OnCheckedChangeListener() {
@@ -72,6 +80,7 @@ public class StorageWizardMigrate extends StorageWizardBase {
if (mRadioNow.isChecked()) { if (mRadioNow.isChecked()) {
final Intent intent = new Intent(this, StorageWizardMigrateConfirm.class); final Intent intent = new Intent(this, StorageWizardMigrateConfirm.class);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId());
mEstimate.copyTo(intent);
startActivity(intent); startActivity(intent);
} else if (mRadioLater.isChecked()) { } else if (mRadioLater.isChecked()) {
final Intent intent = new Intent(this, StorageWizardReady.class); final Intent intent = new Intent(this, StorageWizardReady.class);

View File

@@ -17,36 +17,54 @@
package com.android.settings.deviceinfo; package com.android.settings.deviceinfo;
import android.content.Intent; import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Bundle; import android.os.Bundle;
import android.os.storage.DiskInfo; import android.os.storage.VolumeInfo;
import android.text.format.DateUtils;
import android.text.format.Formatter;
import com.android.internal.util.Preconditions;
import com.android.settings.R; import com.android.settings.R;
public class StorageWizardMigrateConfirm extends StorageWizardBase { public class StorageWizardMigrateConfirm extends StorageWizardBase {
private MigrateEstimateTask mEstimate;
@Override @Override
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
setContentView(R.layout.storage_wizard_generic); setContentView(R.layout.storage_wizard_generic);
Preconditions.checkNotNull(mDisk); // When called with just disk, find the first private volume
if (mVolume == null) {
mVolume = findFirstVolume(VolumeInfo.TYPE_PRIVATE);
}
final String time = DateUtils.formatDuration(0).toString(); final VolumeInfo sourceVol = getPackageManager().getPrimaryStorageCurrentVolume();
final String size = Formatter.formatFileSize(this, 0); final String sourceDescrip = mStorage.getBestVolumeDescription(sourceVol);
final String targetDescrip = mStorage.getBestVolumeDescription(mVolume);
setHeaderText(R.string.storage_wizard_migrate_confirm_title, mDisk.getDescription()); setHeaderText(R.string.storage_wizard_migrate_confirm_title, targetDescrip);
setBodyText(R.string.storage_wizard_migrate_confirm_body, time, size); setBodyText(R.string.memory_calculating_size);
setSecondaryBodyText(R.string.storage_wizard_migrate_details, mDisk.getDescription()); setSecondaryBodyText(R.string.storage_wizard_migrate_details, targetDescrip);
mEstimate = new MigrateEstimateTask(this) {
@Override
public void onPostExecute(String size, String time) {
setBodyText(R.string.storage_wizard_migrate_confirm_body, time, size,
sourceDescrip);
}
};
mEstimate.copyFrom(getIntent());
mEstimate.execute();
getNextButton().setText(R.string.storage_wizard_migrate_confirm_next); getNextButton().setText(R.string.storage_wizard_migrate_confirm_next);
} }
@Override @Override
public void onNavigateNext() { public void onNavigateNext() {
final int moveId = getPackageManager().movePrimaryStorage(mVolume);
final Intent intent = new Intent(this, StorageWizardMigrateProgress.class); final Intent intent = new Intent(this, StorageWizardMigrateProgress.class);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); intent.putExtra(VolumeInfo.EXTRA_VOLUME_ID, mVolume.getId());
intent.putExtra(PackageManager.EXTRA_MOVE_ID, moveId);
startActivity(intent); startActivity(intent);
finishAffinity(); finishAffinity();
} }

View File

@@ -16,13 +16,15 @@
package com.android.settings.deviceinfo; package com.android.settings.deviceinfo;
import static android.content.pm.PackageManager.EXTRA_MOVE_ID;
import static com.android.settings.deviceinfo.StorageSettings.TAG; import static com.android.settings.deviceinfo.StorageSettings.TAG;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.os.AsyncTask; import android.content.pm.PackageManager;
import android.content.pm.PackageManager.MoveCallback;
import android.os.Bundle; import android.os.Bundle;
import android.os.SystemClock; import android.os.Handler;
import android.os.storage.DiskInfo; import android.os.storage.DiskInfo;
import android.util.Log; import android.util.Log;
import android.view.View; import android.view.View;
@@ -32,45 +34,51 @@ import com.android.internal.util.Preconditions;
import com.android.settings.R; import com.android.settings.R;
public class StorageWizardMigrateProgress extends StorageWizardBase { public class StorageWizardMigrateProgress extends StorageWizardBase {
private int mMoveId;
@Override @Override
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
setContentView(R.layout.storage_wizard_progress); setContentView(R.layout.storage_wizard_progress);
Preconditions.checkNotNull(mDisk); Preconditions.checkNotNull(mVolume);
setHeaderText(R.string.storage_wizard_migrate_progress_title, mDisk.getDescription()); mMoveId = getIntent().getIntExtra(EXTRA_MOVE_ID, -1);
setBodyText(R.string.storage_wizard_migrate_details, mDisk.getDescription());
setCurrentProgress(20); final String descrip = mStorage.getBestVolumeDescription(mVolume);
setHeaderText(R.string.storage_wizard_migrate_progress_title, descrip);
setBodyText(R.string.storage_wizard_migrate_details, descrip);
getNextButton().setVisibility(View.GONE); getNextButton().setVisibility(View.GONE);
new MigrateTask().execute(); // Register for updates and push through current status
getPackageManager().registerMoveCallback(mCallback, new Handler());
mCallback.onStatusChanged(mMoveId, getPackageManager().getMoveStatus(mMoveId), -1);
} }
public class MigrateTask extends AsyncTask<Void, Void, Exception> { private final MoveCallback mCallback = new MoveCallback() {
@Override @Override
protected Exception doInBackground(Void... params) { public void onStatusChanged(int moveId, int status, long estMillis) {
// TODO: wire up migration if (mMoveId != moveId) return;
SystemClock.sleep(2000);
return null;
}
@Override
protected void onPostExecute(Exception e) {
final Context context = StorageWizardMigrateProgress.this; final Context context = StorageWizardMigrateProgress.this;
if (e == null) { if (PackageManager.isMoveStatusFinished(status)) {
final Intent intent = new Intent(context, StorageWizardReady.class); Log.d(TAG, "Finished with status " + status);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId()); if (status == PackageManager.MOVE_SUCCEEDED) {
startActivity(intent); if (mDisk != null) {
final Intent intent = new Intent(context, StorageWizardReady.class);
intent.putExtra(DiskInfo.EXTRA_DISK_ID, mDisk.getId());
startActivity(intent);
}
} else {
Toast.makeText(context, getString(R.string.insufficient_storage),
Toast.LENGTH_LONG).show();
}
finishAffinity(); finishAffinity();
} else { } else {
Log.e(TAG, "Failed to migrate", e); setCurrentProgress(status);
Toast.makeText(context, e.getMessage(), Toast.LENGTH_LONG).show();
finishAffinity();
} }
} }
} };
} }

View File

@@ -49,7 +49,7 @@ public class StorageWizardMoveProgress extends StorageWizardBase {
// Register for updates and push through current status // Register for updates and push through current status
getPackageManager().registerMoveCallback(mCallback, new Handler()); getPackageManager().registerMoveCallback(mCallback, new Handler());
mCallback.onStatusChanged(mMoveId, null, getPackageManager().getMoveStatus(mMoveId), -1); mCallback.onStatusChanged(mMoveId, getPackageManager().getMoveStatus(mMoveId), -1);
} }
@Override @Override
@@ -60,7 +60,7 @@ public class StorageWizardMoveProgress extends StorageWizardBase {
private final MoveCallback mCallback = new MoveCallback() { private final MoveCallback mCallback = new MoveCallback() {
@Override @Override
public void onStatusChanged(int moveId, String moveTitle, int status, long estMillis) { public void onStatusChanged(int moveId, int status, long estMillis) {
if (mMoveId != moveId) return; if (mMoveId != moveId) return;
if (PackageManager.isMoveStatusFinished(status)) { if (PackageManager.isMoveStatusFinished(status)) {

View File

@@ -22,9 +22,6 @@ import android.os.storage.VolumeInfo;
import com.android.internal.util.Preconditions; import com.android.internal.util.Preconditions;
import com.android.settings.R; import com.android.settings.R;
import java.util.List;
import java.util.Objects;
public class StorageWizardReady extends StorageWizardBase { public class StorageWizardReady extends StorageWizardBase {
@Override @Override
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState) {
@@ -37,19 +34,14 @@ public class StorageWizardReady extends StorageWizardBase {
// TODO: handle mixed partition cases instead of just guessing based on // TODO: handle mixed partition cases instead of just guessing based on
// first volume type we encounter // first volume type we encounter
final List<VolumeInfo> vols = mStorage.getVolumes(); final VolumeInfo publicVol = findFirstVolume(VolumeInfo.TYPE_PUBLIC);
for (VolumeInfo vol : vols) { final VolumeInfo privateVol = findFirstVolume(VolumeInfo.TYPE_PRIVATE);
if (!Objects.equals(mDisk.getId(), vol.getDiskId())) continue; if (publicVol != null) {
setBodyText(R.string.storage_wizard_ready_external_body,
if (vol.getType() == VolumeInfo.TYPE_PUBLIC) { mDisk.getDescription());
setBodyText(R.string.storage_wizard_ready_external_body, } else if (privateVol != null) {
mDisk.getDescription()); setBodyText(R.string.storage_wizard_ready_internal_body,
break; mDisk.getDescription());
} else if (vol.getType() == VolumeInfo.TYPE_PRIVATE) {
setBodyText(R.string.storage_wizard_ready_internal_body,
mDisk.getDescription());
break;
}
} }
getNextButton().setText(R.string.done); getNextButton().setText(R.string.done);