0
0
Fork 0
mirror of https://github.com/YTVanced/VancedManager synced 2025-01-07 07:51:00 +00:00

Added an option to switch between internal and external storage

This commit is contained in:
X1nto 2021-04-18 14:22:54 +04:00
parent 2d30eb60af
commit 98c4b88961
18 changed files with 210 additions and 44 deletions

View file

@ -24,6 +24,6 @@ object MicrogDownloader {
fun startMicrogInstall(context: Context) { fun startMicrogInstall(context: Context) {
installing.postValue(true) installing.postValue(true)
postReset() postReset()
install("$folderName/$fileName".managerFilepath, context) install(context.getFilePathInStorage("$folderName/$fileName"), context)
} }
} }

View file

@ -68,6 +68,6 @@ object MusicDownloader {
if (variant == "root") if (variant == "root")
installMusicRoot(context) installMusicRoot(context)
else else
install("music/nonroot/nonroot.apk".managerFilepath, context) install(context.getFilePathInStorage("music/nonroot/nonroot.apk"), context)
} }
} }

View file

@ -41,7 +41,7 @@ object VancedDownloader {
prefs = context.installPrefs prefs = context.installPrefs
variant = defPrefs.managerVariant variant = defPrefs.managerVariant
folderName = "vanced/$variant" folderName = "vanced/$variant"
downloadPath = context.getExternalFilesDir(folderName)?.path downloadPath = context.getFilePathInStorage(folderName!!)
File(downloadPath.toString()).deleteRecursively() File(downloadPath.toString()).deleteRecursively()
prefs.lang?.let { prefs.lang?.let {
lang = it.split(", ").toMutableList() lang = it.split(", ").toMutableList()

View file

@ -102,7 +102,7 @@ class MainActivity : AppCompatActivity() {
setFinalTheme() setFinalTheme()
super.onResume() super.onResume()
Crowdin.registerDataLoadingObserver(loadingObserver) Crowdin.registerDataLoadingObserver(loadingObserver)
if (!canAccessStorage(this)) { if (defPrefs.managerStorage == externalPath && !canAccessStorage(this)) {
DialogContainer.storageDialog(this) DialogContainer.storageDialog(this)
} }
} }

View file

@ -42,7 +42,7 @@ object DialogContainer {
setMessage(R.string.storage_access_required_summary) setMessage(R.string.storage_access_required_summary)
setPositiveButton(R.string.auth_dialog_ok) { dialog, _ -> setPositiveButton(R.string.auth_dialog_ok) { dialog, _ ->
dialog.cancel() dialog.cancel()
requestStoragePerms(activity) requestStoragePerms(activity, null)
} }
setCancelable(false) setCancelable(false)
create() create()

View file

@ -0,0 +1,67 @@
package com.vanced.manager.ui.dialogs
import android.os.Bundle
import android.view.LayoutInflater
import android.view.ViewGroup
import androidx.activity.result.contract.ActivityResultContracts
import androidx.preference.PreferenceManager.getDefaultSharedPreferences
import com.google.android.material.radiobutton.MaterialRadioButton
import com.vanced.manager.core.ui.base.BindingBottomSheetDialogFragment
import com.vanced.manager.databinding.DialogManagerStorageBinding
import com.vanced.manager.utils.*
class ManagerStorageDialog : BindingBottomSheetDialogFragment<DialogManagerStorageBinding>() {
private val permissionLauncher = registerForActivityResult(ActivityResultContracts.RequestMultiplePermissions()) { results ->
if (results.all { it.value == true }) {
prefs.managerStorage = externalPath
} else {
prefs.managerStorage = internalPath
}
dismiss()
}
companion object {
fun newInstance(): ManagerStorageDialog = ManagerStorageDialog().apply {
arguments = Bundle()
}
}
private val prefs by lazy { getDefaultSharedPreferences(requireActivity()) }
override fun binding(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
) = DialogManagerStorageBinding.inflate(inflater, container, false)
override fun otherSetups() {
bindData()
}
private fun bindData() {
with(binding) {
val storage = prefs.managerStorage
root.findViewWithTag<MaterialRadioButton>(storage).isChecked = true
storageSave.setOnClickListener {
val newPref = storageRadiogroup.checkedButtonTag
if (storage != newPref) {
if (newPref == externalPath) {
if (!canAccessStorage(requireActivity())) {
return@setOnClickListener requestStoragePerms(requireActivity(), permissionLauncher)
}
prefs.managerStorage = externalPath
} else {
prefs.managerStorage = internalPath
}
dismiss()
requireActivity().recreate()
} else {
dismiss()
}
}
}
}
}

View file

@ -10,7 +10,7 @@ import com.vanced.manager.core.ui.base.BindingFragment
import com.vanced.manager.databinding.FragmentLogBinding import com.vanced.manager.databinding.FragmentLogBinding
import com.vanced.manager.utils.AppUtils import com.vanced.manager.utils.AppUtils
import com.vanced.manager.utils.AppUtils.logs import com.vanced.manager.utils.AppUtils.logs
import com.vanced.manager.utils.managerFilepath import com.vanced.manager.utils.getFilePathInStorage
import com.vanced.manager.utils.performStorageAction import com.vanced.manager.utils.performStorageAction
import java.io.File import java.io.File
import java.io.FileWriter import java.io.FileWriter
@ -42,7 +42,7 @@ class LogFragment : BindingFragment<FragmentLogBinding>() {
val second = calendar.get(Calendar.SECOND) val second = calendar.get(Calendar.SECOND)
try { try {
performStorageAction(requireActivity()) { performStorageAction(requireActivity()) {
val logPath = File("logs".managerFilepath).apply { val logPath = File(requireActivity().getFilePathInStorage("logs")).apply {
if (!exists()) { if (!exists()) {
mkdirs() mkdirs()
} }

View file

@ -19,10 +19,7 @@ import com.vanced.manager.core.ui.base.BindingFragment
import com.vanced.manager.core.ui.ext.showDialog import com.vanced.manager.core.ui.ext.showDialog
import com.vanced.manager.databinding.FragmentSettingsBinding import com.vanced.manager.databinding.FragmentSettingsBinding
import com.vanced.manager.ui.dialogs.* import com.vanced.manager.ui.dialogs.*
import com.vanced.manager.utils.accentColor import com.vanced.manager.utils.*
import com.vanced.manager.utils.defAccentColor
import com.vanced.manager.utils.getLanguageFormat
import com.vanced.manager.utils.toHex
import java.io.File import java.io.File
class SettingsFragment : BindingFragment<FragmentSettingsBinding>() { class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
@ -60,6 +57,7 @@ class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
bindManagerTheme() bindManagerTheme()
bindManagerAccentColor() bindManagerAccentColor()
bindManagerLanguage() bindManagerLanguage()
bindManagerStorage()
selectApps.setOnClickListener { showDialog(SelectAppsDialog()) } selectApps.setOnClickListener { showDialog(SelectAppsDialog()) }
} }
} }
@ -86,6 +84,21 @@ class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
} }
} }
private fun FragmentSettingsBinding.bindManagerStorage() {
managerStorage.apply {
setSummary(
getString(
if (prefs.managerStorage == externalPath) {
R.string.storage_external
} else {
R.string.storage_internal
}
)
)
setOnClickListener { showDialog(ManagerStorageDialog()) }
}
}
private fun FragmentSettingsBinding.bindServiceDTimer() { private fun FragmentSettingsBinding.bindServiceDTimer() {
servicedTimer.apply { servicedTimer.apply {
if (variant == "root") this.isVisible = true if (variant == "root") this.isVisible = true
@ -96,10 +109,13 @@ class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
private fun FragmentSettingsBinding.bindClearFiles() { private fun FragmentSettingsBinding.bindClearFiles() {
clearFiles.setOnClickListener { clearFiles.setOnClickListener {
with(requireActivity()) { with(requireActivity()) {
listOf("vanced/nonroot", "vanced/root", "music/nonroot", "music/root", "microg").forEach { dir -> performStorageAction(this) { managerPath ->
File(getExternalFilesDir(dir)?.path.toString()).deleteRecursively() if (File(managerPath).deleteRecursively()) {
Toast.makeText(this, getString(R.string.cleared_files), Toast.LENGTH_SHORT).show()
} else {
Toast.makeText(this, getString(R.string.clear_files_failed), Toast.LENGTH_SHORT).show()
}
} }
Toast.makeText(this, getString(R.string.cleared_files), Toast.LENGTH_SHORT).show()
} }
} }
} }

View file

@ -110,7 +110,7 @@ class HomeViewModel(private val activity: FragmentActivity): ViewModel() {
activity.getString(R.string.vanced) -> { activity.getString(R.string.vanced) -> {
when (variant) { when (variant) {
"nonroot" -> { "nonroot" -> {
if (vancedInstallFilesExist()) { if (vancedInstallFilesExist(activity)) {
InstallationFilesDetectedDialog.newInstance(app).show(activity) InstallationFilesDetectedDialog.newInstance(app).show(activity)
} else { } else {
VancedPreferencesDialog().show(activity) VancedPreferencesDialog().show(activity)
@ -124,7 +124,7 @@ class HomeViewModel(private val activity: FragmentActivity): ViewModel() {
activity.getString(R.string.music) -> { activity.getString(R.string.music) -> {
when (variant) { when (variant) {
"nonroot" -> { "nonroot" -> {
if (musicApkExists()) { if (musicApkExists(activity)) {
InstallationFilesDetectedDialog.newInstance(app).show(activity) InstallationFilesDetectedDialog.newInstance(app).show(activity)
} else { } else {
MusicPreferencesDialog().show(activity) MusicPreferencesDialog().show(activity)
@ -136,7 +136,7 @@ class HomeViewModel(private val activity: FragmentActivity): ViewModel() {
} }
} }
activity.getString(R.string.microg) -> { activity.getString(R.string.microg) -> {
if (apkExist("microg.apk")) { if (apkExist("microg.apk", activity)) {
InstallationFilesDetectedDialog.newInstance(app).show(activity) InstallationFilesDetectedDialog.newInstance(app).show(activity)
} else { } else {
AppDownloadDialog.newInstance(app).show(activity) AppDownloadDialog.newInstance(app).show(activity)

View file

@ -49,7 +49,7 @@ object DownloadHelper : CoroutineScope by CoroutineScope(Dispatchers.IO) {
override fun onResponse(call: Call<ResponseBody>, response: Response<ResponseBody>) { override fun onResponse(call: Call<ResponseBody>, response: Response<ResponseBody>) {
if (response.isSuccessful) { if (response.isSuccessful) {
launch { launch {
if (response.body()?.let { writeFile(it, fileFolder, fileName) } == true) { if (response.body()?.let { writeFile(it, fileFolder, fileName, context) } == true) {
onDownloadComplete() onDownloadComplete()
} else { } else {
onError(response.errorBody().toString()) onError(response.errorBody().toString())
@ -78,12 +78,12 @@ object DownloadHelper : CoroutineScope by CoroutineScope(Dispatchers.IO) {
}) })
} }
fun writeFile(body: ResponseBody, folderName: String, fileName: String): Boolean = fun writeFile(body: ResponseBody, folderName: String, fileName: String, context: Context): Boolean =
try { try {
val totalBytes = body.contentLength() val totalBytes = body.contentLength()
val fileReader = ByteArray(8096) val fileReader = ByteArray(8096)
var downloadedBytes: Long = 0 var downloadedBytes: Long = 0
val dir = File(managerPath, folderName).apply { val dir = File(context.getFilePathInStorage(folderName)).apply {
if (!exists()) { if (!exists()) {
mkdirs() mkdirs()
} }
@ -114,7 +114,7 @@ object DownloadHelper : CoroutineScope by CoroutineScope(Dispatchers.IO) {
"manager.apk", "manager.apk",
context, context,
onDownloadComplete = { onDownloadComplete = {
val apk = File("manager/manager.apk".managerFilepath) val apk = File(context.getFilePathInStorage("manager/manager.apk"))
val uri = val uri =
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
FileProvider.getUriForFile(context, "${context.packageName}.provider", apk) FileProvider.getUriForFile(context, "${context.packageName}.provider", apk)

View file

@ -19,8 +19,6 @@ import java.util.*
val RadioGroup.checkedButtonTag: String? get() = findViewById<MaterialRadioButton>(checkedRadioButtonId)?.tag?.toString() val RadioGroup.checkedButtonTag: String? get() = findViewById<MaterialRadioButton>(checkedRadioButtonId)?.tag?.toString()
val String.managerFilepath get() = "$managerPath/$this"
fun DialogFragment.show(activity: FragmentActivity) { fun DialogFragment.show(activity: FragmentActivity) {
try { try {
show(activity.supportFragmentManager, "") show(activity.supportFragmentManager, "")

View file

@ -100,16 +100,16 @@ object PackageHelper {
} }
} }
fun apkExist(apk: String): Boolean { fun apkExist(apk: String, context: Context): Boolean {
val apkPath = File("${apk.removeSuffix(".apk")}/$apk".managerFilepath) val apkPath = File(context.getFilePathInStorage("${apk.removeSuffix(".apk")}/$apk"))
if (apkPath.exists()) if (apkPath.exists())
return true return true
return false return false
} }
fun musicApkExists(): Boolean { fun musicApkExists(context: Context): Boolean {
val apkPath = File("music/nonroot/nonroot.apk".managerFilepath) val apkPath = File(context.getFilePathInStorage("music/nonroot/nonroot.apk"))
if (apkPath.exists()) { if (apkPath.exists()) {
return true return true
} }
@ -117,8 +117,8 @@ object PackageHelper {
return false return false
} }
fun vancedInstallFilesExist(): Boolean { fun vancedInstallFilesExist(context: Context): Boolean {
val apksPath = File("vanced/nonroot".managerFilepath) val apksPath = File(context.getFilePathInStorage("vanced/nonroot"))
val splitFiles = mutableListOf<String>() val splitFiles = mutableListOf<String>()
if (apksPath.exists()) { if (apksPath.exists()) {
val files = apksPath.listFiles() val files = apksPath.listFiles()
@ -212,7 +212,7 @@ object PackageHelper {
private fun installRootApp(context: Context, app: String, appVerCode: Int?, pkg: String, modApkBool: (fileName: String) -> Boolean) = CoroutineScope(Dispatchers.IO).launch { private fun installRootApp(context: Context, app: String, appVerCode: Int?, pkg: String, modApkBool: (fileName: String) -> Boolean) = CoroutineScope(Dispatchers.IO).launch {
Shell.getShell { Shell.getShell {
val apkFilesPath = "$app/root".managerFilepath val apkFilesPath = context.getFilePathInStorage("$app/root")
val files = File(apkFilesPath).listFiles()?.toList() val files = File(apkFilesPath).listFiles()?.toList()
if (files != null) { if (files != null) {
val modApk: File? = files.lastOrNull { modApkBool(it.name) } val modApk: File? = files.lastOrNull { modApkBool(it.name) }
@ -268,7 +268,7 @@ object PackageHelper {
appName: String appName: String
) { ) {
val packageInstaller = context.packageManager.packageInstaller val packageInstaller = context.packageManager.packageInstaller
val folder = File("$appName/nonroot".managerFilepath) val folder = File(context.getFilePathInStorage("$appName/nonroot"))
var session: PackageInstaller.Session? = null var session: PackageInstaller.Session? = null
val sessionId: Int val sessionId: Int
val sessionParams = PackageInstaller.SessionParams(PackageInstaller.SessionParams.MODE_FULL_INSTALL) val sessionParams = PackageInstaller.SessionParams(PackageInstaller.SessionParams.MODE_FULL_INSTALL)

View file

@ -19,6 +19,10 @@ var SharedPreferences.managerVariant
get() = getString("vanced_variant", "nonroot") get() = getString("vanced_variant", "nonroot")
set(value) = edit { putString("vanced_variant", value) } set(value) = edit { putString("vanced_variant", value) }
var SharedPreferences.managerStorage
get() = getString("manager_storage", internalPath)
set(value) = edit { putString("manager_storage", value) }
var SharedPreferences.managerLang var SharedPreferences.managerLang
get() = getString("manager_lang", "System Default") get() = getString("manager_lang", "System Default")
set(value) = edit { putString("manager_lang", value) } set(value) = edit { putString("manager_lang", value) }

View file

@ -1,40 +1,63 @@
package com.vanced.manager.utils package com.vanced.manager.utils
import android.Manifest import android.Manifest
import android.content.Context
import android.content.Intent import android.content.Intent
import android.content.pm.PackageManager import android.content.pm.PackageManager
import android.os.Build import android.os.Build
import android.os.Environment import android.os.Environment
import android.provider.Settings import android.provider.Settings
import androidx.activity.result.ActivityResultLauncher
import androidx.fragment.app.FragmentActivity import androidx.fragment.app.FragmentActivity
import com.vanced.manager.ui.MainActivity
import com.vanced.manager.ui.dialogs.DialogContainer.storageDialog import com.vanced.manager.ui.dialogs.DialogContainer.storageDialog
@Suppress("DEPRECATION") const val storage = "/storage/emulated/0"
val managerPath get() = "${Environment.getExternalStorageDirectory().path}/Vanced Manager"
const val internalPath = "$storage/Android/data/com.vanced.manager/files"
const val externalPath = "$storage/Vanced Manager"
val Context.managerPath: String
get() {
var path = defPrefs.managerStorage ?: internalPath
if (path == externalPath && !canAccessStorage(this)) {
path = internalPath
}
return path
}
fun Context.getFilePathInStorage(child: String): String = "$managerPath/$child"
inline fun performStorageAction(activity: FragmentActivity, action: (managerPath: String) -> Unit) {
val managerPath = activity.managerPath
if (managerPath == internalPath) {
return action(managerPath)
}
inline fun performStorageAction(activity: FragmentActivity, action: () -> Unit) {
if (canAccessStorage(activity)) { if (canAccessStorage(activity)) {
action() action(managerPath)
} else { } else {
storageDialog(activity) storageDialog(activity)
} }
} }
fun canAccessStorage(activity: FragmentActivity): Boolean = when { fun canAccessStorage(context: Context): Boolean = when {
Build.VERSION.SDK_INT >= Build.VERSION_CODES.R -> Environment.isExternalStorageManager() Build.VERSION.SDK_INT >= Build.VERSION_CODES.R -> Environment.isExternalStorageManager()
Build.VERSION.SDK_INT >= Build.VERSION_CODES.M -> { Build.VERSION.SDK_INT >= Build.VERSION_CODES.M -> {
listOf( listOf(
Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE Manifest.permission.WRITE_EXTERNAL_STORAGE
).all { ).all {
activity.checkSelfPermission(it) == PackageManager.PERMISSION_GRANTED context.checkSelfPermission(it) == PackageManager.PERMISSION_GRANTED
} }
} }
else -> true else -> true
} }
fun requestStoragePerms(activity: FragmentActivity) { fun requestStoragePerms(activity: FragmentActivity, permissionLauncher: ActivityResultLauncher<Array<String>>?) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
activity.startActivity( activity.startActivity(
Intent( Intent(
@ -42,12 +65,11 @@ fun requestStoragePerms(activity: FragmentActivity) {
) )
) )
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
activity.requestPermissions( permissionLauncher?.launch(
arrayOf( arrayOf(
Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE Manifest.permission.READ_EXTERNAL_STORAGE
), )
MainActivity.REQUEST_CODE
) )
} }
} }

View file

@ -0,0 +1,46 @@
<?xml version="1.0" encoding="utf-8"?>
<com.google.android.material.card.MaterialCardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
style="@style/BottomDialogCard">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<TextView
android:text="@string/storage"
style="@style/BottomDialogCardTitle" />
<androidx.core.widget.NestedScrollView
android:layout_width="match_parent"
android:layout_height="wrap_content">
<RadioGroup
android:id="@+id/storage_radiogroup"
android:layout_width="match_parent"
android:layout_height="wrap_content"
tools:ignore="HardcodedText">
<com.vanced.manager.ui.core.ThemedMaterialRadioButton
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:tag="@string/internal_storage_path"
android:text="@string/storage_internal"
android:textSize="18sp" />
<com.vanced.manager.ui.core.ThemedMaterialRadioButton
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:tag="@string/external_storage_path"
android:text="@string/storage_external"
android:textSize="18sp" />
</RadioGroup>
</androidx.core.widget.NestedScrollView>
<com.vanced.manager.ui.core.ThemedMaterialButton
android:id="@+id/storage_save"
android:text="@string/save"
style="@style/BottomDialogButtonStyle" />
</LinearLayout>
</com.google.android.material.card.MaterialCardView>

View file

@ -49,6 +49,12 @@
android:layout_height="wrap_content" android:layout_height="wrap_content"
app:preference_title="@string/variant" /> app:preference_title="@string/variant" />
<com.vanced.manager.ui.core.EmptyPreference
android:id="@+id/manager_storage"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:preference_title="@string/storage" />
<com.vanced.manager.ui.core.EmptyPreference <com.vanced.manager.ui.core.EmptyPreference
android:id="@+id/serviced_timer" android:id="@+id/serviced_timer"
android:layout_width="match_parent" android:layout_width="match_parent"

View file

@ -16,4 +16,7 @@
<!-- prefs --> <!-- prefs -->
<string name="use_custom_tabs" translatable="false">use_custom_tabs</string> <string name="use_custom_tabs" translatable="false">use_custom_tabs</string>
<string name="internal_storage_path" translatable="false">/storage/emulated/0/Android/data/com.vanced.manager/files</string>
<string name="external_storage_path" translatable="false">/storage/emulated/0/Vanced Manager</string>
</resources> </resources>

View file

@ -50,6 +50,7 @@
<string name="category_appearance">Appearance</string> <string name="category_appearance">Appearance</string>
<string name="category_behaviour">Behavior</string> <string name="category_behaviour">Behavior</string>
<string name="clear_files">Clear downloaded files</string> <string name="clear_files">Clear downloaded files</string>
<string name="clear_files_failed">Failed to clear files</string>
<string name="cleared_files">Successfully cleared files</string> <string name="cleared_files">Successfully cleared files</string>
<string name="firebase_summary">This lets us collect information about app performance and crash logs</string> <string name="firebase_summary">This lets us collect information about app performance and crash logs</string>
<string name="firebase_title">Firebase Analytics</string> <string name="firebase_title">Firebase Analytics</string>
@ -95,8 +96,11 @@
<string name="please_be_patient">Please do NOT exit the app during this process!</string> <string name="please_be_patient">Please do NOT exit the app during this process!</string>
<string name="redownload">Redownload</string> <string name="redownload">Redownload</string>
<string name="security_context">Make sure that you downloaded the app from vancedapp.com, the Vanced Discord server, or the Vanced GitHub</string> <string name="security_context">Make sure that you downloaded the app from vancedapp.com, the Vanced Discord server, or the Vanced GitHub</string>
<string name="storage">Storage</string>
<string name="storage_access_required">Storage access required</string> <string name="storage_access_required">Storage access required</string>
<string name="storage_access_required_summary">In order for Vanced Manager to work, you must grant the storage permission</string> <string name="storage_access_required_summary">In order for Vanced Manager to work, you must grant the storage permission</string>
<string name="storage_external">Internal Storage</string>
<string name="storage_internal">App-Specific Storage</string>
<string name="version">Version</string> <string name="version">Version</string>
<string name="welcome">Welcome</string> <string name="welcome">Welcome</string>