Android: Convert UserDataActivity to Kotlin
This commit is contained in:
parent
089eab96d7
commit
8d16aed581
|
@ -1,379 +0,0 @@
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
package org.dolphinemu.dolphinemu.activities;
|
|
||||||
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.content.ActivityNotFoundException;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.Intent;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.os.Bundle;
|
|
||||||
import android.view.View;
|
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
import androidx.appcompat.app.AppCompatActivity;
|
|
||||||
import androidx.core.graphics.Insets;
|
|
||||||
import androidx.core.view.ViewCompat;
|
|
||||||
import androidx.core.view.WindowCompat;
|
|
||||||
import androidx.core.view.WindowInsetsCompat;
|
|
||||||
|
|
||||||
import com.google.android.material.color.MaterialColors;
|
|
||||||
import com.google.android.material.dialog.MaterialAlertDialogBuilder;
|
|
||||||
|
|
||||||
import org.dolphinemu.dolphinemu.R;
|
|
||||||
import org.dolphinemu.dolphinemu.databinding.ActivityUserDataBinding;
|
|
||||||
import org.dolphinemu.dolphinemu.utils.DirectoryInitialization;
|
|
||||||
import org.dolphinemu.dolphinemu.utils.InsetsHelper;
|
|
||||||
import org.dolphinemu.dolphinemu.utils.Log;
|
|
||||||
import org.dolphinemu.dolphinemu.utils.ThemeHelper;
|
|
||||||
import org.dolphinemu.dolphinemu.utils.ThreadUtil;
|
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
import java.io.FileOutputStream;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.io.OutputStream;
|
|
||||||
import java.util.zip.ZipEntry;
|
|
||||||
import java.util.zip.ZipInputStream;
|
|
||||||
import java.util.zip.ZipOutputStream;
|
|
||||||
|
|
||||||
public class UserDataActivity extends AppCompatActivity
|
|
||||||
{
|
|
||||||
private static final int REQUEST_CODE_IMPORT = 0;
|
|
||||||
private static final int REQUEST_CODE_EXPORT = 1;
|
|
||||||
|
|
||||||
private static final int BUFFER_SIZE = 64 * 1024;
|
|
||||||
|
|
||||||
private boolean sMustRestartApp = false;
|
|
||||||
|
|
||||||
private ActivityUserDataBinding mBinding;
|
|
||||||
|
|
||||||
public static void launch(Context context)
|
|
||||||
{
|
|
||||||
Intent launcher = new Intent(context, UserDataActivity.class);
|
|
||||||
context.startActivity(launcher);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onCreate(Bundle savedInstanceState)
|
|
||||||
{
|
|
||||||
ThemeHelper.setTheme(this);
|
|
||||||
|
|
||||||
super.onCreate(savedInstanceState);
|
|
||||||
|
|
||||||
mBinding = ActivityUserDataBinding.inflate(getLayoutInflater());
|
|
||||||
setContentView(mBinding.getRoot());
|
|
||||||
|
|
||||||
WindowCompat.setDecorFitsSystemWindows(getWindow(), false);
|
|
||||||
|
|
||||||
boolean android_10 = Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q;
|
|
||||||
boolean android_11 = Build.VERSION.SDK_INT >= Build.VERSION_CODES.R;
|
|
||||||
boolean legacy = DirectoryInitialization.isUsingLegacyUserDirectory();
|
|
||||||
|
|
||||||
int user_data_new_location = android_10 ?
|
|
||||||
R.string.user_data_new_location_android_10 : R.string.user_data_new_location;
|
|
||||||
mBinding.textType.setText(legacy ? R.string.user_data_old_location : user_data_new_location);
|
|
||||||
|
|
||||||
mBinding.textPath.setText(DirectoryInitialization.getUserDirectory());
|
|
||||||
|
|
||||||
mBinding.textAndroid11.setVisibility(android_11 && !legacy ? View.VISIBLE : View.GONE);
|
|
||||||
|
|
||||||
mBinding.buttonOpenSystemFileManager.setVisibility(android_11 ? View.VISIBLE : View.GONE);
|
|
||||||
mBinding.buttonOpenSystemFileManager.setOnClickListener(view -> openFileManager());
|
|
||||||
|
|
||||||
mBinding.buttonImportUserData.setOnClickListener(view -> importUserData());
|
|
||||||
|
|
||||||
mBinding.buttonExportUserData.setOnClickListener(view -> exportUserData());
|
|
||||||
|
|
||||||
setSupportActionBar(mBinding.toolbarUserData);
|
|
||||||
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
|
||||||
|
|
||||||
setInsets();
|
|
||||||
ThemeHelper.enableScrollTint(this, mBinding.toolbarUserData, mBinding.appbarUserData);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onSupportNavigateUp()
|
|
||||||
{
|
|
||||||
onBackPressed();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onActivityResult(int requestCode, int resultCode, Intent data)
|
|
||||||
{
|
|
||||||
super.onActivityResult(requestCode, resultCode, data);
|
|
||||||
|
|
||||||
if (requestCode == REQUEST_CODE_IMPORT && resultCode == Activity.RESULT_OK)
|
|
||||||
{
|
|
||||||
Uri uri = data.getData();
|
|
||||||
|
|
||||||
new MaterialAlertDialogBuilder(this)
|
|
||||||
.setMessage(R.string.user_data_import_warning)
|
|
||||||
.setNegativeButton(R.string.no, (dialog, i) -> dialog.dismiss())
|
|
||||||
.setPositiveButton(R.string.yes, (dialog, i) ->
|
|
||||||
{
|
|
||||||
dialog.dismiss();
|
|
||||||
|
|
||||||
ThreadUtil.runOnThreadAndShowResult(this, R.string.import_in_progress,
|
|
||||||
R.string.do_not_close_app,
|
|
||||||
() -> getResources().getString(importUserData(uri)),
|
|
||||||
(dialogInterface) ->
|
|
||||||
{
|
|
||||||
if (sMustRestartApp)
|
|
||||||
{
|
|
||||||
System.exit(0);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
else if (requestCode == REQUEST_CODE_EXPORT && resultCode == Activity.RESULT_OK)
|
|
||||||
{
|
|
||||||
Uri uri = data.getData();
|
|
||||||
|
|
||||||
ThreadUtil.runOnThreadAndShowResult(this, R.string.export_in_progress, 0,
|
|
||||||
() -> getResources().getString(exportUserData(uri)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void openFileManager()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// First, try the package name used on "normal" phones
|
|
||||||
startActivity(getFileManagerIntent("com.google.android.documentsui"));
|
|
||||||
}
|
|
||||||
catch (ActivityNotFoundException e)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// Next, try the AOSP package name
|
|
||||||
startActivity(getFileManagerIntent("com.android.documentsui"));
|
|
||||||
}
|
|
||||||
catch (ActivityNotFoundException e2)
|
|
||||||
{
|
|
||||||
// Activity not found. Perhaps it was removed by the OEM, or by some new Android version
|
|
||||||
// that didn't exist at the time of writing. Not much we can do other than tell the user
|
|
||||||
new MaterialAlertDialogBuilder(this)
|
|
||||||
.setMessage(R.string.user_data_open_system_file_manager_failed)
|
|
||||||
.setPositiveButton(R.string.ok, null)
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private Intent getFileManagerIntent(String packageName)
|
|
||||||
{
|
|
||||||
// Fragile, but some phones don't expose the system file manager in any better way
|
|
||||||
Intent intent = new Intent(Intent.ACTION_MAIN);
|
|
||||||
intent.setClassName(packageName, "com.android.documentsui.files.FilesActivity");
|
|
||||||
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
|
||||||
return intent;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void importUserData()
|
|
||||||
{
|
|
||||||
Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
|
|
||||||
intent.setType("application/zip");
|
|
||||||
startActivityForResult(intent, REQUEST_CODE_IMPORT);
|
|
||||||
}
|
|
||||||
|
|
||||||
private int importUserData(Uri source)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (!isDolphinUserDataBackup(source))
|
|
||||||
{
|
|
||||||
return R.string.user_data_import_invalid_file;
|
|
||||||
}
|
|
||||||
|
|
||||||
try (InputStream is = getContentResolver().openInputStream(source))
|
|
||||||
{
|
|
||||||
try (ZipInputStream zis = new ZipInputStream(is))
|
|
||||||
{
|
|
||||||
File userDirectory = new File(DirectoryInitialization.getUserDirectory());
|
|
||||||
String userDirectoryCanonicalized = userDirectory.getCanonicalPath() + '/';
|
|
||||||
|
|
||||||
sMustRestartApp = true;
|
|
||||||
deleteChildrenRecursively(userDirectory);
|
|
||||||
|
|
||||||
DirectoryInitialization.getGameListCache(this).delete();
|
|
||||||
|
|
||||||
ZipEntry ze;
|
|
||||||
byte[] buffer = new byte[BUFFER_SIZE];
|
|
||||||
while ((ze = zis.getNextEntry()) != null)
|
|
||||||
{
|
|
||||||
File destFile = new File(userDirectory, ze.getName());
|
|
||||||
File destDirectory = ze.isDirectory() ? destFile : destFile.getParentFile();
|
|
||||||
|
|
||||||
if (!destFile.getCanonicalPath().startsWith(userDirectoryCanonicalized))
|
|
||||||
{
|
|
||||||
Log.error("Zip file attempted path traversal! " + ze.getName());
|
|
||||||
return R.string.user_data_import_failure;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!destDirectory.isDirectory() && !destDirectory.mkdirs())
|
|
||||||
{
|
|
||||||
throw new IOException("Failed to create directory " + destDirectory);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!ze.isDirectory())
|
|
||||||
{
|
|
||||||
try (FileOutputStream fos = new FileOutputStream(destFile))
|
|
||||||
{
|
|
||||||
int count;
|
|
||||||
while ((count = zis.read(buffer)) != -1)
|
|
||||||
{
|
|
||||||
fos.write(buffer, 0, count);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
long time = ze.getTime();
|
|
||||||
if (time > 0)
|
|
||||||
{
|
|
||||||
destFile.setLastModified(time);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (IOException | NullPointerException e)
|
|
||||||
{
|
|
||||||
e.printStackTrace();
|
|
||||||
return R.string.user_data_import_failure;
|
|
||||||
}
|
|
||||||
|
|
||||||
return R.string.user_data_import_success;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isDolphinUserDataBackup(Uri uri) throws IOException
|
|
||||||
{
|
|
||||||
try (InputStream is = getContentResolver().openInputStream(uri))
|
|
||||||
{
|
|
||||||
try (ZipInputStream zis = new ZipInputStream(is))
|
|
||||||
{
|
|
||||||
ZipEntry ze;
|
|
||||||
while ((ze = zis.getNextEntry()) != null)
|
|
||||||
{
|
|
||||||
String name = ze.getName();
|
|
||||||
if (name.equals("Config/Dolphin.ini"))
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void deleteChildrenRecursively(File directory) throws IOException
|
|
||||||
{
|
|
||||||
File[] children = directory.listFiles();
|
|
||||||
if (children == null)
|
|
||||||
{
|
|
||||||
throw new IOException("Could not find directory " + directory);
|
|
||||||
}
|
|
||||||
for (File child : children)
|
|
||||||
{
|
|
||||||
deleteRecursively(child);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void deleteRecursively(File file) throws IOException
|
|
||||||
{
|
|
||||||
if (file.isDirectory())
|
|
||||||
{
|
|
||||||
deleteChildrenRecursively(file);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!file.delete())
|
|
||||||
{
|
|
||||||
throw new IOException("Failed to delete " + file);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void exportUserData()
|
|
||||||
{
|
|
||||||
Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT);
|
|
||||||
intent.setType("application/zip");
|
|
||||||
intent.putExtra(Intent.EXTRA_TITLE, "dolphin-emu.zip");
|
|
||||||
startActivityForResult(intent, REQUEST_CODE_EXPORT);
|
|
||||||
}
|
|
||||||
|
|
||||||
private int exportUserData(Uri destination)
|
|
||||||
{
|
|
||||||
try (OutputStream os = getContentResolver().openOutputStream(destination))
|
|
||||||
{
|
|
||||||
try (ZipOutputStream zos = new ZipOutputStream(os))
|
|
||||||
{
|
|
||||||
exportUserData(zos, new File(DirectoryInitialization.getUserDirectory()), null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (IOException e)
|
|
||||||
{
|
|
||||||
e.printStackTrace();
|
|
||||||
return R.string.user_data_export_failure;
|
|
||||||
}
|
|
||||||
|
|
||||||
return R.string.user_data_export_success;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void exportUserData(ZipOutputStream zos, File input, @Nullable File pathRelativeToRoot)
|
|
||||||
throws IOException
|
|
||||||
{
|
|
||||||
if (input.isDirectory())
|
|
||||||
{
|
|
||||||
File[] children = input.listFiles();
|
|
||||||
if (children == null)
|
|
||||||
{
|
|
||||||
throw new IOException("Could not find directory " + input);
|
|
||||||
}
|
|
||||||
for (File child : children)
|
|
||||||
{
|
|
||||||
exportUserData(zos, child, new File(pathRelativeToRoot, child.getName()));
|
|
||||||
}
|
|
||||||
if (children.length == 0 && pathRelativeToRoot != null)
|
|
||||||
{
|
|
||||||
zos.putNextEntry(new ZipEntry(pathRelativeToRoot.getPath() + '/'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
try (FileInputStream fis = new FileInputStream(input))
|
|
||||||
{
|
|
||||||
byte[] buffer = new byte[BUFFER_SIZE];
|
|
||||||
ZipEntry entry = new ZipEntry(pathRelativeToRoot.getPath());
|
|
||||||
entry.setTime(input.lastModified());
|
|
||||||
zos.putNextEntry(entry);
|
|
||||||
int count;
|
|
||||||
while ((count = fis.read(buffer, 0, buffer.length)) != -1)
|
|
||||||
{
|
|
||||||
zos.write(buffer, 0, count);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setInsets()
|
|
||||||
{
|
|
||||||
ViewCompat.setOnApplyWindowInsetsListener(mBinding.appbarUserData, (v, windowInsets) ->
|
|
||||||
{
|
|
||||||
Insets insets = windowInsets.getInsets(WindowInsetsCompat.Type.systemBars());
|
|
||||||
|
|
||||||
InsetsHelper.insetAppBar(insets, mBinding.appbarUserData);
|
|
||||||
|
|
||||||
mBinding.scrollViewUserData.setPadding(insets.left, 0, insets.right, insets.bottom);
|
|
||||||
|
|
||||||
InsetsHelper.applyNavbarWorkaround(insets.bottom, mBinding.workaroundView);
|
|
||||||
ThemeHelper.setNavigationBarColor(this,
|
|
||||||
MaterialColors.getColor(mBinding.appbarUserData, R.attr.colorSurface));
|
|
||||||
|
|
||||||
return windowInsets;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,314 @@
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
package org.dolphinemu.dolphinemu.activities
|
||||||
|
|
||||||
|
import android.content.ActivityNotFoundException
|
||||||
|
import android.content.Context
|
||||||
|
import android.content.DialogInterface
|
||||||
|
import android.content.Intent
|
||||||
|
import android.net.Uri
|
||||||
|
import android.os.Build
|
||||||
|
import android.os.Bundle
|
||||||
|
import android.view.View
|
||||||
|
import androidx.appcompat.app.AppCompatActivity
|
||||||
|
import androidx.core.view.ViewCompat
|
||||||
|
import androidx.core.view.WindowCompat
|
||||||
|
import androidx.core.view.WindowInsetsCompat
|
||||||
|
import com.google.android.material.color.MaterialColors
|
||||||
|
import com.google.android.material.dialog.MaterialAlertDialogBuilder
|
||||||
|
import org.dolphinemu.dolphinemu.R
|
||||||
|
import org.dolphinemu.dolphinemu.databinding.ActivityUserDataBinding
|
||||||
|
import org.dolphinemu.dolphinemu.utils.*
|
||||||
|
import org.dolphinemu.dolphinemu.utils.ThemeHelper.enableScrollTint
|
||||||
|
import org.dolphinemu.dolphinemu.utils.ThemeHelper.setNavigationBarColor
|
||||||
|
import java.io.File
|
||||||
|
import java.io.FileInputStream
|
||||||
|
import java.io.FileOutputStream
|
||||||
|
import java.io.IOException
|
||||||
|
import java.util.zip.ZipEntry
|
||||||
|
import java.util.zip.ZipInputStream
|
||||||
|
import java.util.zip.ZipOutputStream
|
||||||
|
import kotlin.system.exitProcess
|
||||||
|
|
||||||
|
class UserDataActivity : AppCompatActivity() {
|
||||||
|
private var sMustRestartApp = false
|
||||||
|
|
||||||
|
private lateinit var mBinding: ActivityUserDataBinding
|
||||||
|
|
||||||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
ThemeHelper.setTheme(this)
|
||||||
|
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
|
||||||
|
mBinding = ActivityUserDataBinding.inflate(layoutInflater)
|
||||||
|
setContentView(mBinding.root)
|
||||||
|
|
||||||
|
WindowCompat.setDecorFitsSystemWindows(window, false)
|
||||||
|
|
||||||
|
val android10 = Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q
|
||||||
|
val android11 = Build.VERSION.SDK_INT >= Build.VERSION_CODES.R
|
||||||
|
val legacy = DirectoryInitialization.isUsingLegacyUserDirectory()
|
||||||
|
|
||||||
|
val userDataNewLocation =
|
||||||
|
if (android10) R.string.user_data_new_location_android_10 else R.string.user_data_new_location
|
||||||
|
mBinding.textType.setText(if (legacy) R.string.user_data_old_location else userDataNewLocation)
|
||||||
|
|
||||||
|
mBinding.textPath.text = DirectoryInitialization.getUserDirectory()
|
||||||
|
|
||||||
|
mBinding.textAndroid11.visibility = if (android11 && !legacy) View.VISIBLE else View.GONE
|
||||||
|
|
||||||
|
mBinding.buttonOpenSystemFileManager.visibility = if (android11) View.VISIBLE else View.GONE
|
||||||
|
mBinding.buttonOpenSystemFileManager.setOnClickListener { openFileManager() }
|
||||||
|
|
||||||
|
mBinding.buttonImportUserData.setOnClickListener { importUserData() }
|
||||||
|
|
||||||
|
mBinding.buttonExportUserData.setOnClickListener { exportUserData() }
|
||||||
|
|
||||||
|
setSupportActionBar(mBinding.toolbarUserData)
|
||||||
|
supportActionBar!!.setDisplayHomeAsUpEnabled(true)
|
||||||
|
|
||||||
|
setInsets()
|
||||||
|
enableScrollTint(this, mBinding.toolbarUserData, mBinding.appbarUserData)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onSupportNavigateUp(): Boolean {
|
||||||
|
onBackPressed()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
|
||||||
|
super.onActivityResult(requestCode, resultCode, data)
|
||||||
|
|
||||||
|
if (requestCode == REQUEST_CODE_IMPORT && resultCode == RESULT_OK) {
|
||||||
|
|
||||||
|
MaterialAlertDialogBuilder(this)
|
||||||
|
.setMessage(R.string.user_data_import_warning)
|
||||||
|
.setNegativeButton(R.string.no) { dialog: DialogInterface, _: Int -> dialog.dismiss() }
|
||||||
|
.setPositiveButton(R.string.yes) { dialog: DialogInterface, _: Int ->
|
||||||
|
dialog.dismiss()
|
||||||
|
|
||||||
|
ThreadUtil.runOnThreadAndShowResult(
|
||||||
|
this,
|
||||||
|
R.string.import_in_progress,
|
||||||
|
R.string.do_not_close_app,
|
||||||
|
{ resources.getString(importUserData(data!!.data!!)) }) {
|
||||||
|
if (sMustRestartApp) {
|
||||||
|
exitProcess(0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.show()
|
||||||
|
} else if (requestCode == REQUEST_CODE_EXPORT && resultCode == RESULT_OK) {
|
||||||
|
ThreadUtil.runOnThreadAndShowResult(
|
||||||
|
this,
|
||||||
|
R.string.export_in_progress,
|
||||||
|
0
|
||||||
|
) { resources.getString(exportUserData(data!!.data!!)) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun openFileManager() {
|
||||||
|
try {
|
||||||
|
// First, try the package name used on "normal" phones
|
||||||
|
startActivity(getFileManagerIntent("com.google.android.documentsui"))
|
||||||
|
} catch (e: ActivityNotFoundException) {
|
||||||
|
try {
|
||||||
|
// Next, try the AOSP package name
|
||||||
|
startActivity(getFileManagerIntent("com.android.documentsui"))
|
||||||
|
} catch (e2: ActivityNotFoundException) {
|
||||||
|
// Activity not found. Perhaps it was removed by the OEM, or by some new Android version
|
||||||
|
// that didn't exist at the time of writing. Not much we can do other than tell the user.
|
||||||
|
MaterialAlertDialogBuilder(this)
|
||||||
|
.setMessage(R.string.user_data_open_system_file_manager_failed)
|
||||||
|
.setPositiveButton(R.string.ok, null)
|
||||||
|
.show()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getFileManagerIntent(packageName: String): Intent {
|
||||||
|
// Fragile, but some phones don't expose the system file manager in any better way
|
||||||
|
val intent = Intent(Intent.ACTION_MAIN)
|
||||||
|
intent.setClassName(packageName, "com.android.documentsui.files.FilesActivity")
|
||||||
|
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
|
||||||
|
return intent
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun importUserData() {
|
||||||
|
val intent = Intent(Intent.ACTION_OPEN_DOCUMENT)
|
||||||
|
intent.type = "application/zip"
|
||||||
|
startActivityForResult(intent, REQUEST_CODE_IMPORT)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun importUserData(source: Uri): Int {
|
||||||
|
try {
|
||||||
|
if (!isDolphinUserDataBackup(source))
|
||||||
|
return R.string.user_data_import_invalid_file
|
||||||
|
|
||||||
|
contentResolver.openInputStream(source).use { `is` ->
|
||||||
|
ZipInputStream(`is`).use { zis ->
|
||||||
|
val userDirectory = File(DirectoryInitialization.getUserDirectory())
|
||||||
|
val userDirectoryCanonicalized = userDirectory.canonicalPath + '/'
|
||||||
|
|
||||||
|
sMustRestartApp = true
|
||||||
|
deleteChildrenRecursively(userDirectory)
|
||||||
|
|
||||||
|
DirectoryInitialization.getGameListCache(this).delete()
|
||||||
|
|
||||||
|
var ze: ZipEntry? = zis.nextEntry
|
||||||
|
val buffer = ByteArray(BUFFER_SIZE)
|
||||||
|
while (ze != null) {
|
||||||
|
val destFile = File(userDirectory, ze.name)
|
||||||
|
val destDirectory = if (ze.isDirectory) destFile else destFile.parentFile
|
||||||
|
|
||||||
|
if (!destFile.canonicalPath.startsWith(userDirectoryCanonicalized)) {
|
||||||
|
Log.error("Zip file attempted path traversal! " + ze.name)
|
||||||
|
return R.string.user_data_import_failure
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!destDirectory.isDirectory && !destDirectory.mkdirs()) {
|
||||||
|
throw IOException("Failed to create directory $destDirectory")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ze.isDirectory) {
|
||||||
|
FileOutputStream(destFile).use { fos ->
|
||||||
|
var count: Int
|
||||||
|
while (zis.read(buffer).also { count = it } != -1) {
|
||||||
|
fos.write(buffer, 0, count)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val time = ze.time
|
||||||
|
if (time > 0) {
|
||||||
|
destFile.setLastModified(time)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ze = zis.nextEntry
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e: IOException) {
|
||||||
|
e.printStackTrace()
|
||||||
|
return R.string.user_data_import_failure
|
||||||
|
} catch (e: NullPointerException) {
|
||||||
|
e.printStackTrace()
|
||||||
|
return R.string.user_data_import_failure
|
||||||
|
}
|
||||||
|
return R.string.user_data_import_success
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun isDolphinUserDataBackup(uri: Uri): Boolean {
|
||||||
|
contentResolver.openInputStream(uri).use { `is` ->
|
||||||
|
ZipInputStream(`is`).use { zis ->
|
||||||
|
var ze: ZipEntry
|
||||||
|
while (zis.nextEntry.also { ze = it } != null) {
|
||||||
|
val name = ze.name
|
||||||
|
if (name == "Config/Dolphin.ini") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun deleteChildrenRecursively(directory: File) {
|
||||||
|
val children =
|
||||||
|
directory.listFiles() ?: throw IOException("Could not find directory $directory")
|
||||||
|
for (child in children) {
|
||||||
|
deleteRecursively(child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun deleteRecursively(file: File) {
|
||||||
|
if (file.isDirectory) {
|
||||||
|
deleteChildrenRecursively(file)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!file.delete()) {
|
||||||
|
throw IOException("Failed to delete $file")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun exportUserData() {
|
||||||
|
val intent = Intent(Intent.ACTION_CREATE_DOCUMENT)
|
||||||
|
intent.type = "application/zip"
|
||||||
|
intent.putExtra(Intent.EXTRA_TITLE, "dolphin-emu.zip")
|
||||||
|
startActivityForResult(intent, REQUEST_CODE_EXPORT)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun exportUserData(destination: Uri): Int {
|
||||||
|
try {
|
||||||
|
contentResolver.openOutputStream(destination).use { os ->
|
||||||
|
ZipOutputStream(os).use { zos ->
|
||||||
|
exportUserData(
|
||||||
|
zos,
|
||||||
|
File(DirectoryInitialization.getUserDirectory()),
|
||||||
|
null
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e: IOException) {
|
||||||
|
e.printStackTrace()
|
||||||
|
return R.string.user_data_export_failure
|
||||||
|
}
|
||||||
|
return R.string.user_data_export_success
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun exportUserData(zos: ZipOutputStream, input: File, pathRelativeToRoot: File?) {
|
||||||
|
if (input.isDirectory) {
|
||||||
|
val children = input.listFiles() ?: throw IOException("Could not find directory $input")
|
||||||
|
for (child in children) {
|
||||||
|
exportUserData(zos, child, File(pathRelativeToRoot, child.name))
|
||||||
|
}
|
||||||
|
if (children.isEmpty() && pathRelativeToRoot != null) {
|
||||||
|
zos.putNextEntry(ZipEntry(pathRelativeToRoot.path + '/'))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
FileInputStream(input).use { fis ->
|
||||||
|
val buffer = ByteArray(BUFFER_SIZE)
|
||||||
|
val entry = ZipEntry(pathRelativeToRoot!!.path)
|
||||||
|
entry.time = input.lastModified()
|
||||||
|
zos.putNextEntry(entry)
|
||||||
|
var count: Int
|
||||||
|
while (fis.read(buffer, 0, buffer.size).also { count = it } != -1) {
|
||||||
|
zos.write(buffer, 0, count)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setInsets() {
|
||||||
|
ViewCompat.setOnApplyWindowInsetsListener(mBinding.appbarUserData) { _: View?, windowInsets: WindowInsetsCompat ->
|
||||||
|
val insets = windowInsets.getInsets(WindowInsetsCompat.Type.systemBars())
|
||||||
|
|
||||||
|
InsetsHelper.insetAppBar(insets, mBinding.appbarUserData)
|
||||||
|
|
||||||
|
mBinding.scrollViewUserData.setPadding(insets.left, 0, insets.right, insets.bottom)
|
||||||
|
|
||||||
|
InsetsHelper.applyNavbarWorkaround(insets.bottom, mBinding.workaroundView)
|
||||||
|
setNavigationBarColor(
|
||||||
|
this,
|
||||||
|
MaterialColors.getColor(mBinding.appbarUserData, R.attr.colorSurface)
|
||||||
|
)
|
||||||
|
windowInsets
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private const val REQUEST_CODE_IMPORT = 0
|
||||||
|
private const val REQUEST_CODE_EXPORT = 1
|
||||||
|
|
||||||
|
private const val BUFFER_SIZE = 64 * 1024
|
||||||
|
|
||||||
|
@JvmStatic
|
||||||
|
fun launch(context: Context) {
|
||||||
|
val launcher = Intent(context, UserDataActivity::class.java)
|
||||||
|
context.startActivity(launcher)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue