From 17ebb4c613401eaaa7f11a1307c21fc7a319216a Mon Sep 17 00:00:00 2001 From: GoneUp Date: Sat, 18 Feb 2017 00:22:23 +0100 Subject: [PATCH 1/2] Changed settings export to use the storage access framework --- .../java/com/fsck/k9/activity/Accounts.java | 75 +++++++++++++++---- .../fsck/k9/preferences/SettingsExporter.java | 26 ++++++- k9mail/src/main/res/values/strings.xml | 1 + 3 files changed, 87 insertions(+), 15 deletions(-) diff --git a/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java b/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java index 74765fd86..828df90ed 100644 --- a/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java +++ b/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java @@ -146,6 +146,7 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { private static final int ACTIVITY_REQUEST_PICK_SETTINGS_FILE = 1; + private static final int ACTIVITY_REQUEST_SAVE_SETTINGS_FILE = 2; class AccountsHandler extends Handler { private void setViewTitle() { @@ -1290,7 +1291,7 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { getString(R.string.app_revision_url) + "")) .append("


") - .append(String.format(getString(R.string.app_copyright_fmt), year, year)) + .append(String.format(getString(R.string.app_copyright_fmt), Integer.toString(year), Integer.toString(year))) .append("


") .append(getString(R.string.app_license)) .append("


"); @@ -1418,9 +1419,12 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { return; } switch (requestCode) { - case ACTIVITY_REQUEST_PICK_SETTINGS_FILE: - onImport(data.getData()); - break; + case ACTIVITY_REQUEST_PICK_SETTINGS_FILE: + onImport(data.getData()); + break; + case ACTIVITY_REQUEST_SAVE_SETTINGS_FILE: + onExport(data); + break; } } @@ -1883,16 +1887,47 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { } - public void onExport(final boolean includeGlobals, final Account account) { + public static final String EXTRA_INC_GLOBALS = "include_globals"; + public static final String EXTRA_ACCOUNTS = "accountUuids"; + public void onExport(final boolean includeGlobals, final Account account) { // TODO, prompt to allow a user to choose which accounts to export - Set accountUuids = null; + ArrayList accountUuids = null; if (account != null) { - accountUuids = new HashSet(); + accountUuids = new ArrayList<>(); accountUuids.add(account.getUuid()); } - ExportAsyncTask asyncTask = new ExportAsyncTask(this, includeGlobals, accountUuids); + if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT) { + Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT); + + intent.addCategory(Intent.CATEGORY_OPENABLE); + intent.setType("text/plain"); + intent.putExtra(Intent.EXTRA_TITLE, SettingsExporter.EXPORT_FILENAME); + intent.putStringArrayListExtra(EXTRA_ACCOUNTS, accountUuids); + intent.putExtra(EXTRA_INC_GLOBALS, includeGlobals); + + PackageManager packageManager = getPackageManager(); + List infos = packageManager.queryIntentActivities(intent, 0); + + if (infos.size() > 0) { + startActivityForResult(Intent.createChooser(intent, null), ACTIVITY_REQUEST_SAVE_SETTINGS_FILE); + } else { + showDialog(DIALOG_NO_FILE_MANAGER); + } + } else { + //Pre-Kitkat + ExportAsyncTask asyncTask = new ExportAsyncTask(this, includeGlobals, accountUuids, null); + setNonConfigurationInstance(asyncTask); + asyncTask.execute(); + } + } + + public void onExport(Intent intent) { + boolean includeGlobals = intent.getBooleanExtra(EXTRA_INC_GLOBALS, false); + ArrayList accountUuids = intent.getStringArrayListExtra(EXTRA_ACCOUNTS); + + ExportAsyncTask asyncTask = new ExportAsyncTask(this, includeGlobals, accountUuids, intent.getData()); setNonConfigurationInstance(asyncTask); asyncTask.execute(); } @@ -1904,13 +1939,17 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { private boolean mIncludeGlobals; private Set mAccountUuids; private String mFileName; + private Uri mUri; private ExportAsyncTask(Accounts activity, boolean includeGlobals, - Set accountUuids) { + List accountUuids, Uri uri) { super(activity); mIncludeGlobals = includeGlobals; - mAccountUuids = accountUuids; + mUri = uri; + if (accountUuids != null) { + mAccountUuids = new HashSet<>(accountUuids); + } } @Override @@ -1923,8 +1962,13 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { @Override protected Boolean doInBackground(Void... params) { try { - mFileName = SettingsExporter.exportToFile(mContext, mIncludeGlobals, + if (mUri == null) { + mFileName = SettingsExporter.exportToFile(mContext, mIncludeGlobals, mAccountUuids); + } else { + SettingsExporter.exportToUri(mContext, mIncludeGlobals, mAccountUuids, mUri); + } + } catch (SettingsImportExportException e) { Timber.w(e, "Exception during export"); return false; @@ -1942,8 +1986,13 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { removeProgressDialog(); if (success) { - activity.showSimpleDialog(R.string.settings_export_success_header, - R.string.settings_export_success, mFileName); + if (mFileName != null) { + activity.showSimpleDialog(R.string.settings_export_success_header, + R.string.settings_export_success, mFileName); + } else { + activity.showSimpleDialog(R.string.settings_export_success_header, + R.string.settings_export_success_generic); + } } else { //TODO: better error messages activity.showSimpleDialog(R.string.settings_export_failed_header, diff --git a/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java b/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java index 913173cde..1d6b112d2 100644 --- a/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java +++ b/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java @@ -15,6 +15,7 @@ import java.util.Set; import java.util.TreeMap; import android.content.Context; +import android.net.Uri; import android.os.Environment; import timber.log.Timber; import android.util.Xml; @@ -32,7 +33,7 @@ import org.xmlpull.v1.XmlSerializer; public class SettingsExporter { - private static final String EXPORT_FILENAME = "settings.k9s"; + public static final String EXPORT_FILENAME = "settings.k9s"; /** * File format version number. @@ -108,7 +109,28 @@ public class SettingsExporter { } } - static void exportPreferences(Context context, OutputStream os, boolean includeGlobals, Set accountUuids) + public static void exportToUri(Context context, boolean includeGlobals, Set accountUuids, Uri uri) + throws SettingsImportExportException { + + OutputStream os = null; + String filename = null; + try { + os = context.getContentResolver().openOutputStream(uri); + exportPreferences(context, os, includeGlobals, accountUuids); + } catch (Exception e) { + throw new SettingsImportExportException(e); + } finally { + if (os != null) { + try { + os.close(); + } catch (IOException ioe) { + Log.w(K9.LOG_TAG, "Couldn't close exported settings file: " + filename); + } + } + } + } + + static void exportPreferences(Context context, OutputStream os, boolean includeGlobals, Set accountUuids) throws SettingsImportExportException { try { diff --git a/k9mail/src/main/res/values/strings.xml b/k9mail/src/main/res/values/strings.xml index 052485cd6..6fac436c3 100644 --- a/k9mail/src/main/res/values/strings.xml +++ b/k9mail/src/main/res/values/strings.xml @@ -975,6 +975,7 @@ Please submit bug reports, contribute new features and ask questions at Importing settings… Scanning file… Saved exported settings to %s + Settings successfully exported Imported global settings from %s Imported %s from %s From 5fa77adfcbcad4acafdc4d315a7cf97624f3ee71 Mon Sep 17 00:00:00 2001 From: cketti Date: Sun, 12 Mar 2017 17:03:10 +0100 Subject: [PATCH 2/2] Fix settings export on API 19+ --- .../java/com/fsck/k9/activity/Accounts.java | 55 +++++++++++-------- .../fsck/k9/preferences/SettingsExporter.java | 36 ++++++------ 2 files changed, 48 insertions(+), 43 deletions(-) diff --git a/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java b/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java index 828df90ed..86022cfb0 100644 --- a/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java +++ b/k9mail/src/main/java/com/fsck/k9/activity/Accounts.java @@ -29,6 +29,7 @@ import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.net.Uri; +import android.os.Build; import android.os.Bundle; import android.os.Handler; import android.text.Editable; @@ -137,6 +138,9 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { private TextView mActionBarSubTitle; private TextView mActionBarUnread; + private boolean exportGlobalSettings; + private ArrayList exportAccountUuids; + /** * Contains information about objects that need to be retained on configuration changes. * @@ -329,6 +333,9 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { private static String ACCOUNT_STATS = "accountStats"; private static String STATE_UNREAD_COUNT = "unreadCount"; private static String SELECTED_CONTEXT_ACCOUNT = "selectedContextAccount"; + private static final String STATE_EXPORT_GLOBAL_SETTINGS = "exportGlobalSettings"; + private static final String STATE_EXPORT_ACCOUNTS = "exportAccountUuids"; + public static final String EXTRA_STARTUP = "startup"; @@ -477,6 +484,17 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { } outState.putSerializable(STATE_UNREAD_COUNT, mUnreadMessageCount); outState.putSerializable(ACCOUNT_STATS, accountStats); + + outState.putBoolean(STATE_EXPORT_GLOBAL_SETTINGS, exportGlobalSettings); + outState.putStringArrayList(STATE_EXPORT_ACCOUNTS, exportAccountUuids); + } + + @Override + protected void onRestoreInstanceState(Bundle state) { + super.onRestoreInstanceState(state); + + exportGlobalSettings = state.getBoolean(STATE_EXPORT_GLOBAL_SETTINGS, false); + exportAccountUuids = state.getStringArrayList(STATE_EXPORT_ACCOUNTS); } private StorageManager.StorageListener storageListener = new StorageManager.StorageListener() { @@ -1887,9 +1905,6 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { } - public static final String EXTRA_INC_GLOBALS = "include_globals"; - public static final String EXTRA_ACCOUNTS = "accountUuids"; - public void onExport(final boolean includeGlobals, final Account account) { // TODO, prompt to allow a user to choose which accounts to export ArrayList accountUuids = null; @@ -1898,36 +1913,28 @@ public class Accounts extends K9ListActivity implements OnItemClickListener { accountUuids.add(account.getUuid()); } - if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT) { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { + exportGlobalSettings = includeGlobals; + exportAccountUuids = accountUuids; + Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT); - - intent.addCategory(Intent.CATEGORY_OPENABLE); - intent.setType("text/plain"); + intent.setType("application/octet-stream"); intent.putExtra(Intent.EXTRA_TITLE, SettingsExporter.EXPORT_FILENAME); - intent.putStringArrayListExtra(EXTRA_ACCOUNTS, accountUuids); - intent.putExtra(EXTRA_INC_GLOBALS, includeGlobals); + intent.addCategory(Intent.CATEGORY_OPENABLE); - PackageManager packageManager = getPackageManager(); - List infos = packageManager.queryIntentActivities(intent, 0); - - if (infos.size() > 0) { - startActivityForResult(Intent.createChooser(intent, null), ACTIVITY_REQUEST_SAVE_SETTINGS_FILE); - } else { - showDialog(DIALOG_NO_FILE_MANAGER); - } + startActivityForResult(intent, ACTIVITY_REQUEST_SAVE_SETTINGS_FILE); } else { - //Pre-Kitkat - ExportAsyncTask asyncTask = new ExportAsyncTask(this, includeGlobals, accountUuids, null); - setNonConfigurationInstance(asyncTask); - asyncTask.execute(); + startExport(includeGlobals, accountUuids, null); } } public void onExport(Intent intent) { - boolean includeGlobals = intent.getBooleanExtra(EXTRA_INC_GLOBALS, false); - ArrayList accountUuids = intent.getStringArrayListExtra(EXTRA_ACCOUNTS); + Uri documentsUri = intent.getData(); + startExport(exportGlobalSettings, exportAccountUuids, documentsUri); + } - ExportAsyncTask asyncTask = new ExportAsyncTask(this, includeGlobals, accountUuids, intent.getData()); + private void startExport(boolean exportGlobalSettings, ArrayList exportAccountUuids, Uri documentsUri) { + ExportAsyncTask asyncTask = new ExportAsyncTask(this, exportGlobalSettings, exportAccountUuids, documentsUri); setNonConfigurationInstance(asyncTask); asyncTask.execute(); } diff --git a/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java b/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java index 1d6b112d2..37130cfa7 100644 --- a/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java +++ b/k9mail/src/main/java/com/fsck/k9/preferences/SettingsExporter.java @@ -17,11 +17,11 @@ import java.util.TreeMap; import android.content.Context; import android.net.Uri; import android.os.Environment; + import timber.log.Timber; import android.util.Xml; import com.fsck.k9.Account; -import com.fsck.k9.K9; import com.fsck.k9.Preferences; import com.fsck.k9.helper.FileHelper; import com.fsck.k9.mail.ServerSettings; @@ -81,7 +81,6 @@ public class SettingsExporter { throws SettingsImportExportException { OutputStream os = null; - String filename = null; try { File dir = new File(Environment.getExternalStorageDirectory() + File.separator + context.getPackageName()); if (!dir.mkdirs()) { @@ -89,7 +88,7 @@ public class SettingsExporter { } File file = FileHelper.createUniqueFile(dir, EXPORT_FILENAME); - filename = file.getAbsolutePath(); + String filename = file.getAbsolutePath(); os = new FileOutputStream(filename); exportPreferences(context, os, includeGlobals, accountUuids); @@ -99,13 +98,7 @@ public class SettingsExporter { } catch (Exception e) { throw new SettingsImportExportException(e); } finally { - if (os != null) { - try { - os.close(); - } catch (IOException ioe) { - Timber.w("Couldn't close exported settings file: %s", filename); - } - } + closeOrThrow(os); } } @@ -113,24 +106,29 @@ public class SettingsExporter { throws SettingsImportExportException { OutputStream os = null; - String filename = null; try { os = context.getContentResolver().openOutputStream(uri); exportPreferences(context, os, includeGlobals, accountUuids); } catch (Exception e) { throw new SettingsImportExportException(e); } finally { - if (os != null) { - try { - os.close(); - } catch (IOException ioe) { - Log.w(K9.LOG_TAG, "Couldn't close exported settings file: " + filename); - } - } + closeOrThrow(os); } } - static void exportPreferences(Context context, OutputStream os, boolean includeGlobals, Set accountUuids) + private static void closeOrThrow(OutputStream outputStream) throws SettingsImportExportException { + if (outputStream == null) { + return; + } + + try { + outputStream.close(); + } catch (IOException e) { + throw new SettingsImportExportException(e); + } + } + + static void exportPreferences(Context context, OutputStream os, boolean includeGlobals, Set accountUuids) throws SettingsImportExportException { try {