mirror of
https://github.com/YTVanced/VancedManager
synced 2025-01-03 14:10:59 +00:00
Added an option to switch between internal and external storage
This commit is contained in:
parent
2d30eb60af
commit
98c4b88961
18 changed files with 210 additions and 44 deletions
|
@ -24,6 +24,6 @@ object MicrogDownloader {
|
|||
fun startMicrogInstall(context: Context) {
|
||||
installing.postValue(true)
|
||||
postReset()
|
||||
install("$folderName/$fileName".managerFilepath, context)
|
||||
install(context.getFilePathInStorage("$folderName/$fileName"), context)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -68,6 +68,6 @@ object MusicDownloader {
|
|||
if (variant == "root")
|
||||
installMusicRoot(context)
|
||||
else
|
||||
install("music/nonroot/nonroot.apk".managerFilepath, context)
|
||||
install(context.getFilePathInStorage("music/nonroot/nonroot.apk"), context)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -41,7 +41,7 @@ object VancedDownloader {
|
|||
prefs = context.installPrefs
|
||||
variant = defPrefs.managerVariant
|
||||
folderName = "vanced/$variant"
|
||||
downloadPath = context.getExternalFilesDir(folderName)?.path
|
||||
downloadPath = context.getFilePathInStorage(folderName!!)
|
||||
File(downloadPath.toString()).deleteRecursively()
|
||||
prefs.lang?.let {
|
||||
lang = it.split(", ").toMutableList()
|
||||
|
|
|
@ -102,7 +102,7 @@ class MainActivity : AppCompatActivity() {
|
|||
setFinalTheme()
|
||||
super.onResume()
|
||||
Crowdin.registerDataLoadingObserver(loadingObserver)
|
||||
if (!canAccessStorage(this)) {
|
||||
if (defPrefs.managerStorage == externalPath && !canAccessStorage(this)) {
|
||||
DialogContainer.storageDialog(this)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -42,7 +42,7 @@ object DialogContainer {
|
|||
setMessage(R.string.storage_access_required_summary)
|
||||
setPositiveButton(R.string.auth_dialog_ok) { dialog, _ ->
|
||||
dialog.cancel()
|
||||
requestStoragePerms(activity)
|
||||
requestStoragePerms(activity, null)
|
||||
}
|
||||
setCancelable(false)
|
||||
create()
|
||||
|
|
|
@ -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()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -10,7 +10,7 @@ import com.vanced.manager.core.ui.base.BindingFragment
|
|||
import com.vanced.manager.databinding.FragmentLogBinding
|
||||
import com.vanced.manager.utils.AppUtils
|
||||
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 java.io.File
|
||||
import java.io.FileWriter
|
||||
|
@ -42,7 +42,7 @@ class LogFragment : BindingFragment<FragmentLogBinding>() {
|
|||
val second = calendar.get(Calendar.SECOND)
|
||||
try {
|
||||
performStorageAction(requireActivity()) {
|
||||
val logPath = File("logs".managerFilepath).apply {
|
||||
val logPath = File(requireActivity().getFilePathInStorage("logs")).apply {
|
||||
if (!exists()) {
|
||||
mkdirs()
|
||||
}
|
||||
|
|
|
@ -19,10 +19,7 @@ import com.vanced.manager.core.ui.base.BindingFragment
|
|||
import com.vanced.manager.core.ui.ext.showDialog
|
||||
import com.vanced.manager.databinding.FragmentSettingsBinding
|
||||
import com.vanced.manager.ui.dialogs.*
|
||||
import com.vanced.manager.utils.accentColor
|
||||
import com.vanced.manager.utils.defAccentColor
|
||||
import com.vanced.manager.utils.getLanguageFormat
|
||||
import com.vanced.manager.utils.toHex
|
||||
import com.vanced.manager.utils.*
|
||||
import java.io.File
|
||||
|
||||
class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
|
||||
|
@ -60,6 +57,7 @@ class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
|
|||
bindManagerTheme()
|
||||
bindManagerAccentColor()
|
||||
bindManagerLanguage()
|
||||
bindManagerStorage()
|
||||
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() {
|
||||
servicedTimer.apply {
|
||||
if (variant == "root") this.isVisible = true
|
||||
|
@ -96,10 +109,13 @@ class SettingsFragment : BindingFragment<FragmentSettingsBinding>() {
|
|||
private fun FragmentSettingsBinding.bindClearFiles() {
|
||||
clearFiles.setOnClickListener {
|
||||
with(requireActivity()) {
|
||||
listOf("vanced/nonroot", "vanced/root", "music/nonroot", "music/root", "microg").forEach { dir ->
|
||||
File(getExternalFilesDir(dir)?.path.toString()).deleteRecursively()
|
||||
performStorageAction(this) { managerPath ->
|
||||
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()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -110,7 +110,7 @@ class HomeViewModel(private val activity: FragmentActivity): ViewModel() {
|
|||
activity.getString(R.string.vanced) -> {
|
||||
when (variant) {
|
||||
"nonroot" -> {
|
||||
if (vancedInstallFilesExist()) {
|
||||
if (vancedInstallFilesExist(activity)) {
|
||||
InstallationFilesDetectedDialog.newInstance(app).show(activity)
|
||||
} else {
|
||||
VancedPreferencesDialog().show(activity)
|
||||
|
@ -124,7 +124,7 @@ class HomeViewModel(private val activity: FragmentActivity): ViewModel() {
|
|||
activity.getString(R.string.music) -> {
|
||||
when (variant) {
|
||||
"nonroot" -> {
|
||||
if (musicApkExists()) {
|
||||
if (musicApkExists(activity)) {
|
||||
InstallationFilesDetectedDialog.newInstance(app).show(activity)
|
||||
} else {
|
||||
MusicPreferencesDialog().show(activity)
|
||||
|
@ -136,7 +136,7 @@ class HomeViewModel(private val activity: FragmentActivity): ViewModel() {
|
|||
}
|
||||
}
|
||||
activity.getString(R.string.microg) -> {
|
||||
if (apkExist("microg.apk")) {
|
||||
if (apkExist("microg.apk", activity)) {
|
||||
InstallationFilesDetectedDialog.newInstance(app).show(activity)
|
||||
} else {
|
||||
AppDownloadDialog.newInstance(app).show(activity)
|
||||
|
|
|
@ -49,7 +49,7 @@ object DownloadHelper : CoroutineScope by CoroutineScope(Dispatchers.IO) {
|
|||
override fun onResponse(call: Call<ResponseBody>, response: Response<ResponseBody>) {
|
||||
if (response.isSuccessful) {
|
||||
launch {
|
||||
if (response.body()?.let { writeFile(it, fileFolder, fileName) } == true) {
|
||||
if (response.body()?.let { writeFile(it, fileFolder, fileName, context) } == true) {
|
||||
onDownloadComplete()
|
||||
} else {
|
||||
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 {
|
||||
val totalBytes = body.contentLength()
|
||||
val fileReader = ByteArray(8096)
|
||||
var downloadedBytes: Long = 0
|
||||
val dir = File(managerPath, folderName).apply {
|
||||
val dir = File(context.getFilePathInStorage(folderName)).apply {
|
||||
if (!exists()) {
|
||||
mkdirs()
|
||||
}
|
||||
|
@ -114,7 +114,7 @@ object DownloadHelper : CoroutineScope by CoroutineScope(Dispatchers.IO) {
|
|||
"manager.apk",
|
||||
context,
|
||||
onDownloadComplete = {
|
||||
val apk = File("manager/manager.apk".managerFilepath)
|
||||
val apk = File(context.getFilePathInStorage("manager/manager.apk"))
|
||||
val uri =
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
|
||||
FileProvider.getUriForFile(context, "${context.packageName}.provider", apk)
|
||||
|
|
|
@ -19,8 +19,6 @@ import java.util.*
|
|||
|
||||
val RadioGroup.checkedButtonTag: String? get() = findViewById<MaterialRadioButton>(checkedRadioButtonId)?.tag?.toString()
|
||||
|
||||
val String.managerFilepath get() = "$managerPath/$this"
|
||||
|
||||
fun DialogFragment.show(activity: FragmentActivity) {
|
||||
try {
|
||||
show(activity.supportFragmentManager, "")
|
||||
|
|
|
@ -100,16 +100,16 @@ object PackageHelper {
|
|||
}
|
||||
}
|
||||
|
||||
fun apkExist(apk: String): Boolean {
|
||||
val apkPath = File("${apk.removeSuffix(".apk")}/$apk".managerFilepath)
|
||||
fun apkExist(apk: String, context: Context): Boolean {
|
||||
val apkPath = File(context.getFilePathInStorage("${apk.removeSuffix(".apk")}/$apk"))
|
||||
if (apkPath.exists())
|
||||
return true
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
fun musicApkExists(): Boolean {
|
||||
val apkPath = File("music/nonroot/nonroot.apk".managerFilepath)
|
||||
fun musicApkExists(context: Context): Boolean {
|
||||
val apkPath = File(context.getFilePathInStorage("music/nonroot/nonroot.apk"))
|
||||
if (apkPath.exists()) {
|
||||
return true
|
||||
}
|
||||
|
@ -117,8 +117,8 @@ object PackageHelper {
|
|||
return false
|
||||
}
|
||||
|
||||
fun vancedInstallFilesExist(): Boolean {
|
||||
val apksPath = File("vanced/nonroot".managerFilepath)
|
||||
fun vancedInstallFilesExist(context: Context): Boolean {
|
||||
val apksPath = File(context.getFilePathInStorage("vanced/nonroot"))
|
||||
val splitFiles = mutableListOf<String>()
|
||||
if (apksPath.exists()) {
|
||||
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 {
|
||||
Shell.getShell {
|
||||
val apkFilesPath = "$app/root".managerFilepath
|
||||
val apkFilesPath = context.getFilePathInStorage("$app/root")
|
||||
val files = File(apkFilesPath).listFiles()?.toList()
|
||||
if (files != null) {
|
||||
val modApk: File? = files.lastOrNull { modApkBool(it.name) }
|
||||
|
@ -268,7 +268,7 @@ object PackageHelper {
|
|||
appName: String
|
||||
) {
|
||||
val packageInstaller = context.packageManager.packageInstaller
|
||||
val folder = File("$appName/nonroot".managerFilepath)
|
||||
val folder = File(context.getFilePathInStorage("$appName/nonroot"))
|
||||
var session: PackageInstaller.Session? = null
|
||||
val sessionId: Int
|
||||
val sessionParams = PackageInstaller.SessionParams(PackageInstaller.SessionParams.MODE_FULL_INSTALL)
|
||||
|
|
|
@ -19,6 +19,10 @@ var SharedPreferences.managerVariant
|
|||
get() = getString("vanced_variant", "nonroot")
|
||||
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
|
||||
get() = getString("manager_lang", "System Default")
|
||||
set(value) = edit { putString("manager_lang", value) }
|
||||
|
|
|
@ -1,40 +1,63 @@
|
|||
package com.vanced.manager.utils
|
||||
|
||||
import android.Manifest
|
||||
import android.content.Context
|
||||
import android.content.Intent
|
||||
import android.content.pm.PackageManager
|
||||
import android.os.Build
|
||||
import android.os.Environment
|
||||
import android.provider.Settings
|
||||
import androidx.activity.result.ActivityResultLauncher
|
||||
import androidx.fragment.app.FragmentActivity
|
||||
import com.vanced.manager.ui.MainActivity
|
||||
import com.vanced.manager.ui.dialogs.DialogContainer.storageDialog
|
||||
|
||||
@Suppress("DEPRECATION")
|
||||
val managerPath get() = "${Environment.getExternalStorageDirectory().path}/Vanced Manager"
|
||||
const val storage = "/storage/emulated/0"
|
||||
|
||||
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)) {
|
||||
action()
|
||||
action(managerPath)
|
||||
} else {
|
||||
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.M -> {
|
||||
listOf(
|
||||
Manifest.permission.READ_EXTERNAL_STORAGE,
|
||||
Manifest.permission.WRITE_EXTERNAL_STORAGE
|
||||
).all {
|
||||
activity.checkSelfPermission(it) == PackageManager.PERMISSION_GRANTED
|
||||
context.checkSelfPermission(it) == PackageManager.PERMISSION_GRANTED
|
||||
}
|
||||
}
|
||||
else -> true
|
||||
}
|
||||
|
||||
fun requestStoragePerms(activity: FragmentActivity) {
|
||||
fun requestStoragePerms(activity: FragmentActivity, permissionLauncher: ActivityResultLauncher<Array<String>>?) {
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
|
||||
activity.startActivity(
|
||||
Intent(
|
||||
|
@ -42,12 +65,11 @@ fun requestStoragePerms(activity: FragmentActivity) {
|
|||
)
|
||||
)
|
||||
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
||||
activity.requestPermissions(
|
||||
permissionLauncher?.launch(
|
||||
arrayOf(
|
||||
Manifest.permission.READ_EXTERNAL_STORAGE,
|
||||
Manifest.permission.WRITE_EXTERNAL_STORAGE
|
||||
),
|
||||
MainActivity.REQUEST_CODE
|
||||
Manifest.permission.WRITE_EXTERNAL_STORAGE,
|
||||
Manifest.permission.READ_EXTERNAL_STORAGE
|
||||
)
|
||||
)
|
||||
}
|
||||
}
|
46
app/src/main/res/layout/dialog_manager_storage.xml
Normal file
46
app/src/main/res/layout/dialog_manager_storage.xml
Normal 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>
|
|
@ -49,6 +49,12 @@
|
|||
android:layout_height="wrap_content"
|
||||
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
|
||||
android:id="@+id/serviced_timer"
|
||||
android:layout_width="match_parent"
|
||||
|
|
|
@ -16,4 +16,7 @@
|
|||
<!-- prefs -->
|
||||
<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>
|
||||
|
|
|
@ -50,6 +50,7 @@
|
|||
<string name="category_appearance">Appearance</string>
|
||||
<string name="category_behaviour">Behavior</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="firebase_summary">This lets us collect information about app performance and crash logs</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="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="storage">Storage</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_external">Internal Storage</string>
|
||||
<string name="storage_internal">App-Specific Storage</string>
|
||||
<string name="version">Version</string>
|
||||
<string name="welcome">Welcome</string>
|
||||
|
||||
|
|
Loading…
Reference in a new issue