mirror of
				https://github.com/topjohnwu/Magisk
				synced 2025-11-03 15:52:30 +01:00 
			
		
		
		
	Compare commits
	
		
			67 Commits
		
	
	
		
			canary-270
			...
			canary-281
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					47cc532d96 | ||
| 
						 | 
					218327f92b | ||
| 
						 | 
					4eae66a1a7 | ||
| 
						 | 
					b09ceeb43c | ||
| 
						 | 
					4fb539c110 | ||
| 
						 | 
					849b284da5 | ||
| 
						 | 
					895b5f6cbf | ||
| 
						 | 
					cb3d4ea514 | ||
| 
						 | 
					0d89a2a97d | ||
| 
						 | 
					3ca5913055 | ||
| 
						 | 
					df6b808f49 | ||
| 
						 | 
					09c7ac754b | ||
| 
						 | 
					805da67c23 | ||
| 
						 | 
					3c6889505b | ||
| 
						 | 
					c8e9ce7627 | ||
| 
						 | 
					837c679a31 | ||
| 
						 | 
					06616659b8 | ||
| 
						 | 
					a34c04f999 | ||
| 
						 | 
					da43ac89a0 | ||
| 
						 | 
					830fc758b9 | ||
| 
						 | 
					0f3cfef278 | ||
| 
						 | 
					b32d7bfafd | ||
| 
						 | 
					c0899f2939 | ||
| 
						 | 
					082330808f | ||
| 
						 | 
					024da05888 | ||
| 
						 | 
					377b6d0cc2 | ||
| 
						 | 
					c661009b31 | ||
| 
						 | 
					613f2d31c5 | ||
| 
						 | 
					7dbb973db5 | ||
| 
						 | 
					f4502f8be8 | ||
| 
						 | 
					455b13b83c | ||
| 
						 | 
					8b98709743 | ||
| 
						 | 
					1b12f45f39 | ||
| 
						 | 
					a5cad532ff | ||
| 
						 | 
					070719db50 | ||
| 
						 | 
					28cccdf7aa | ||
| 
						 | 
					b7e0986a5c | ||
| 
						 | 
					da709745dd | ||
| 
						 | 
					8b6771d487 | ||
| 
						 | 
					e1b847fbc5 | ||
| 
						 | 
					7188de1205 | ||
| 
						 | 
					44fb7dbcbe | ||
| 
						 | 
					8086b5933c | ||
| 
						 | 
					7f675f4bf7 | ||
| 
						 | 
					5e6b53e0da | ||
| 
						 | 
					5b29fefc65 | ||
| 
						 | 
					16a168535d | ||
| 
						 | 
					33f70f8f6d | ||
| 
						 | 
					4f18a66d73 | ||
| 
						 | 
					250dc16007 | ||
| 
						 | 
					7af273e047 | ||
| 
						 | 
					e381aebaa0 | ||
| 
						 | 
					45d91c9658 | ||
| 
						 | 
					4a9158f667 | ||
| 
						 | 
					0d9ee89e7f | ||
| 
						 | 
					abaff72304 | ||
| 
						 | 
					b828e2d0b2 | ||
| 
						 | 
					53d7cbc11b | ||
| 
						 | 
					310be7ab47 | ||
| 
						 | 
					60894e458f | ||
| 
						 | 
					fbebb6ac10 | ||
| 
						 | 
					a9f8c20703 | ||
| 
						 | 
					ae0b15d197 | ||
| 
						 | 
					869aa62328 | ||
| 
						 | 
					dcd3bc58a3 | ||
| 
						 | 
					a82f17c594 | ||
| 
						 | 
					b38fd1ca5f | 
							
								
								
									
										4
									
								
								.github/actions/setup/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/actions/setup/action.yml
									
									
									
									
										vendored
									
									
								
							@@ -6,11 +6,11 @@ inputs:
 | 
			
		||||
runs:
 | 
			
		||||
  using: "composite"
 | 
			
		||||
  steps:
 | 
			
		||||
    - name: Set up JDK 17
 | 
			
		||||
    - name: Set up JDK 21
 | 
			
		||||
      uses: actions/setup-java@v4
 | 
			
		||||
      with:
 | 
			
		||||
        distribution: "temurin"
 | 
			
		||||
        java-version: "17"
 | 
			
		||||
        java-version: "21"
 | 
			
		||||
 | 
			
		||||
    - name: Set up Python 3
 | 
			
		||||
      uses: actions/setup-python@v5
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							@@ -83,10 +83,10 @@ jobs:
 | 
			
		||||
    strategy:
 | 
			
		||||
      fail-fast: false
 | 
			
		||||
      matrix:
 | 
			
		||||
        version: [23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34]
 | 
			
		||||
        version: [23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35]
 | 
			
		||||
        type: [""]
 | 
			
		||||
        include:
 | 
			
		||||
          - version: 35
 | 
			
		||||
          - version: "Baklava"
 | 
			
		||||
            type: "google_apis"
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -14,6 +14,7 @@ native/out
 | 
			
		||||
*.iml
 | 
			
		||||
.gradle
 | 
			
		||||
.idea
 | 
			
		||||
.kotlin
 | 
			
		||||
/local.properties
 | 
			
		||||
/build
 | 
			
		||||
/captures
 | 
			
		||||
 
 | 
			
		||||
@@ -20,9 +20,9 @@ Some highlight features:
 | 
			
		||||
 | 
			
		||||
Click the icon below to download Magisk apk.
 | 
			
		||||
 | 
			
		||||
[](https://github.com/topjohnwu/Magisk/releases/tag/v27.0)
 | 
			
		||||
[](https://github.com/topjohnwu/Magisk/releases/tag/v27.0)
 | 
			
		||||
[](https://github.com/topjohnwu/Magisk/releases/tag/canary-27007)
 | 
			
		||||
[](https://github.com/topjohnwu/Magisk/releases/tag/v28.1)
 | 
			
		||||
[](https://github.com/topjohnwu/Magisk/releases/tag/v28.1)
 | 
			
		||||
[](https://github.com/topjohnwu/Magisk/releases/tag/canary-28101)
 | 
			
		||||
 | 
			
		||||
## Useful Links
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -6,46 +6,37 @@ plugins {
 | 
			
		||||
    id("androidx.navigation.safeargs.kotlin")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
setupAppCommon()
 | 
			
		||||
setupMainApk()
 | 
			
		||||
 | 
			
		||||
kapt {
 | 
			
		||||
    correctErrorTypes = true
 | 
			
		||||
    useBuildCache = true
 | 
			
		||||
    mapDiagnosticLocations = true
 | 
			
		||||
    javacOptions {
 | 
			
		||||
        option("-Xmaxerrs", 1000)
 | 
			
		||||
        option("-Xmaxerrs", "1000")
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
android {
 | 
			
		||||
    namespace = "com.topjohnwu.magisk"
 | 
			
		||||
    buildFeatures {
 | 
			
		||||
        dataBinding = true
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    defaultConfig {
 | 
			
		||||
        applicationId = "com.topjohnwu.magisk"
 | 
			
		||||
        vectorDrawables.useSupportLibrary = true
 | 
			
		||||
        versionName = Config.version
 | 
			
		||||
        versionCode = Config.versionCode
 | 
			
		||||
        ndk {
 | 
			
		||||
            abiFilters += listOf("armeabi-v7a", "arm64-v8a", "x86", "x86_64", "riscv64")
 | 
			
		||||
            debugSymbolLevel = "FULL"
 | 
			
		||||
        }
 | 
			
		||||
    compileOptions {
 | 
			
		||||
        isCoreLibraryDesugaringEnabled = true
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    buildTypes {
 | 
			
		||||
        release {
 | 
			
		||||
            isMinifyEnabled = true
 | 
			
		||||
            isShrinkResources = true
 | 
			
		||||
            proguardFiles("proguard-rules.pro")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    buildFeatures {
 | 
			
		||||
        dataBinding = true
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    implementation(project(":app:core"))
 | 
			
		||||
    coreLibraryDesugaring(libs.jdk.libs)
 | 
			
		||||
 | 
			
		||||
    implementation(libs.indeterminate.checkbox)
 | 
			
		||||
    implementation(libs.rikka.layoutinflater)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,13 +1,18 @@
 | 
			
		||||
package com.topjohnwu.magisk.dialog
 | 
			
		||||
 | 
			
		||||
import android.widget.Toast
 | 
			
		||||
import androidx.core.os.postDelayed
 | 
			
		||||
import androidx.lifecycle.lifecycleScope
 | 
			
		||||
import com.topjohnwu.magisk.core.BuildConfig
 | 
			
		||||
import com.topjohnwu.magisk.core.Info
 | 
			
		||||
import com.topjohnwu.magisk.core.R
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.reboot
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.toast
 | 
			
		||||
import com.topjohnwu.magisk.core.tasks.MagiskInstaller
 | 
			
		||||
import com.topjohnwu.magisk.events.DialogBuilder
 | 
			
		||||
import com.topjohnwu.magisk.ui.home.HomeViewModel
 | 
			
		||||
import com.topjohnwu.magisk.view.MagiskDialog
 | 
			
		||||
import com.topjohnwu.superuser.internal.UiThreadHandler
 | 
			
		||||
import kotlinx.coroutines.launch
 | 
			
		||||
 | 
			
		||||
class EnvFixDialog(private val vm: HomeViewModel, private val code: Int) : DialogBuilder {
 | 
			
		||||
@@ -27,9 +32,15 @@ class EnvFixDialog(private val vm: HomeViewModel, private val code: Int) : Dialo
 | 
			
		||||
                        setCancelable(false)
 | 
			
		||||
                    }
 | 
			
		||||
                    dialog.activity.lifecycleScope.launch {
 | 
			
		||||
                        MagiskInstaller.FixEnv {
 | 
			
		||||
                        MagiskInstaller.FixEnv().exec { success ->
 | 
			
		||||
                            dialog.dismiss()
 | 
			
		||||
                        }.exec()
 | 
			
		||||
                            context.toast(
 | 
			
		||||
                                if (success) R.string.reboot_delay_toast else R.string.setup_fail,
 | 
			
		||||
                                Toast.LENGTH_LONG
 | 
			
		||||
                            )
 | 
			
		||||
                            if (success)
 | 
			
		||||
                                UiThreadHandler.handler.postDelayed(5000) { reboot() }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,17 @@
 | 
			
		||||
package com.topjohnwu.magisk.dialog
 | 
			
		||||
 | 
			
		||||
import android.app.ProgressDialog
 | 
			
		||||
import android.content.Context
 | 
			
		||||
import android.widget.Toast
 | 
			
		||||
import androidx.lifecycle.lifecycleScope
 | 
			
		||||
import com.topjohnwu.magisk.arch.NavigationActivity
 | 
			
		||||
import com.topjohnwu.magisk.arch.UIActivity
 | 
			
		||||
import com.topjohnwu.magisk.core.R
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.toast
 | 
			
		||||
import com.topjohnwu.magisk.core.tasks.MagiskInstaller
 | 
			
		||||
import com.topjohnwu.magisk.events.DialogBuilder
 | 
			
		||||
import com.topjohnwu.magisk.ui.flash.FlashFragment
 | 
			
		||||
import com.topjohnwu.magisk.view.MagiskDialog
 | 
			
		||||
import com.topjohnwu.superuser.Shell
 | 
			
		||||
import kotlinx.coroutines.launch
 | 
			
		||||
 | 
			
		||||
class UninstallDialog : DialogBuilder {
 | 
			
		||||
 | 
			
		||||
@@ -19,7 +21,7 @@ class UninstallDialog : DialogBuilder {
 | 
			
		||||
            setMessage(R.string.uninstall_magisk_msg)
 | 
			
		||||
            setButton(MagiskDialog.ButtonType.POSITIVE) {
 | 
			
		||||
                text = R.string.restore_img
 | 
			
		||||
                onClick { restore(dialog.context) }
 | 
			
		||||
                onClick { restore(dialog.activity) }
 | 
			
		||||
            }
 | 
			
		||||
            setButton(MagiskDialog.ButtonType.NEGATIVE) {
 | 
			
		||||
                text = R.string.complete_uninstall
 | 
			
		||||
@@ -29,18 +31,20 @@ class UninstallDialog : DialogBuilder {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Suppress("DEPRECATION")
 | 
			
		||||
    private fun restore(context: Context) {
 | 
			
		||||
        val dialog = ProgressDialog(context).apply {
 | 
			
		||||
            setMessage(context.getString(R.string.restore_img_msg))
 | 
			
		||||
    private fun restore(activity: UIActivity<*>) {
 | 
			
		||||
        val dialog = ProgressDialog(activity).apply {
 | 
			
		||||
            setMessage(activity.getString(R.string.restore_img_msg))
 | 
			
		||||
            show()
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        Shell.cmd("restore_imgs").submit { result ->
 | 
			
		||||
            dialog.dismiss()
 | 
			
		||||
            if (result.isSuccess) {
 | 
			
		||||
                context.toast(R.string.restore_done, Toast.LENGTH_SHORT)
 | 
			
		||||
            } else {
 | 
			
		||||
                context.toast(R.string.restore_fail, Toast.LENGTH_LONG)
 | 
			
		||||
        activity.lifecycleScope.launch {
 | 
			
		||||
            MagiskInstaller.Restore().exec { success ->
 | 
			
		||||
                dialog.dismiss()
 | 
			
		||||
                if (success) {
 | 
			
		||||
                    activity.toast(R.string.restore_done, Toast.LENGTH_SHORT)
 | 
			
		||||
                } else {
 | 
			
		||||
                    activity.toast(R.string.restore_fail, Toast.LENGTH_LONG)
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -71,7 +71,7 @@ class FlashFragment : BaseFragment<FragmentFlashMd2Binding>(), MenuProvider {
 | 
			
		||||
        super.onViewCreated(view, savedInstanceState)
 | 
			
		||||
 | 
			
		||||
        defaultOrientation = activity?.requestedOrientation ?: -1
 | 
			
		||||
        activity?.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_NOSENSOR
 | 
			
		||||
        activity?.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_LOCKED
 | 
			
		||||
        if (savedInstanceState == null) {
 | 
			
		||||
            viewModel.startFlashing()
 | 
			
		||||
        }
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,108 @@
 | 
			
		||||
package com.topjohnwu.magisk.ui.module
 | 
			
		||||
 | 
			
		||||
import android.annotation.SuppressLint
 | 
			
		||||
import android.content.pm.ActivityInfo
 | 
			
		||||
import android.os.Bundle
 | 
			
		||||
import android.view.KeyEvent
 | 
			
		||||
import android.view.Menu
 | 
			
		||||
import android.view.MenuInflater
 | 
			
		||||
import android.view.MenuItem
 | 
			
		||||
import android.view.View
 | 
			
		||||
import android.view.ViewTreeObserver
 | 
			
		||||
import android.widget.Toast
 | 
			
		||||
import androidx.core.view.MenuProvider
 | 
			
		||||
import androidx.core.view.isVisible
 | 
			
		||||
import com.topjohnwu.magisk.R
 | 
			
		||||
import com.topjohnwu.magisk.arch.BaseFragment
 | 
			
		||||
import com.topjohnwu.magisk.arch.viewModel
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.toast
 | 
			
		||||
import com.topjohnwu.magisk.databinding.FragmentActionMd2Binding
 | 
			
		||||
import com.topjohnwu.magisk.core.R as CoreR
 | 
			
		||||
 | 
			
		||||
class ActionFragment : BaseFragment<FragmentActionMd2Binding>(), MenuProvider {
 | 
			
		||||
 | 
			
		||||
    override val layoutRes = R.layout.fragment_action_md2
 | 
			
		||||
    override val viewModel by viewModel<ActionViewModel>()
 | 
			
		||||
    override val snackbarView: View get() = binding.snackbarContainer
 | 
			
		||||
 | 
			
		||||
    private var defaultOrientation = -1
 | 
			
		||||
 | 
			
		||||
    override fun onCreate(savedInstanceState: Bundle?) {
 | 
			
		||||
        super.onCreate(savedInstanceState)
 | 
			
		||||
        viewModel.args = ActionFragmentArgs.fromBundle(requireArguments())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onStart() {
 | 
			
		||||
        super.onStart()
 | 
			
		||||
        activity?.setTitle(viewModel.args.name)
 | 
			
		||||
        binding.closeBtn.setOnClickListener {
 | 
			
		||||
            activity?.onBackPressed()
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        viewModel.state.observe(this) {
 | 
			
		||||
            if (it != ActionViewModel.State.RUNNING) {
 | 
			
		||||
                binding.closeBtn.apply {
 | 
			
		||||
                    if (!this.isVisible) this.show()
 | 
			
		||||
                    if (!this.isFocused) this.requestFocus()
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            if (it != ActionViewModel.State.SUCCESS) return@observe
 | 
			
		||||
            view?.viewTreeObserver?.addOnWindowFocusChangeListener(
 | 
			
		||||
                object : ViewTreeObserver.OnWindowFocusChangeListener {
 | 
			
		||||
                    override fun onWindowFocusChanged(hasFocus: Boolean) {
 | 
			
		||||
                        if (hasFocus) return
 | 
			
		||||
                        view?.viewTreeObserver?.removeOnWindowFocusChangeListener(this)
 | 
			
		||||
                        view?.context?.apply {
 | 
			
		||||
                            toast(
 | 
			
		||||
                                getString(CoreR.string.done_action, viewModel.args.name),
 | 
			
		||||
                                Toast.LENGTH_SHORT
 | 
			
		||||
                            )
 | 
			
		||||
                        }
 | 
			
		||||
                        viewModel.back()
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onCreateMenu(menu: Menu, inflater: MenuInflater) {
 | 
			
		||||
        inflater.inflate(R.menu.menu_flash, menu)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onMenuItemSelected(item: MenuItem): Boolean {
 | 
			
		||||
        return viewModel.onMenuItemClicked(item)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
 | 
			
		||||
        super.onViewCreated(view, savedInstanceState)
 | 
			
		||||
 | 
			
		||||
        defaultOrientation = activity?.requestedOrientation ?: -1
 | 
			
		||||
        activity?.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_LOCKED
 | 
			
		||||
        if (savedInstanceState == null) {
 | 
			
		||||
            viewModel.startRunAction()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @SuppressLint("WrongConstant")
 | 
			
		||||
    override fun onDestroyView() {
 | 
			
		||||
        if (defaultOrientation != -1) {
 | 
			
		||||
            activity?.requestedOrientation = defaultOrientation
 | 
			
		||||
        }
 | 
			
		||||
        super.onDestroyView()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onKeyEvent(event: KeyEvent): Boolean {
 | 
			
		||||
        return when (event.keyCode) {
 | 
			
		||||
            KeyEvent.KEYCODE_VOLUME_UP, KeyEvent.KEYCODE_VOLUME_DOWN -> true
 | 
			
		||||
 | 
			
		||||
            else -> false
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onBackPressed(): Boolean {
 | 
			
		||||
        if (viewModel.state.value == ActionViewModel.State.RUNNING) return true
 | 
			
		||||
        return super.onBackPressed()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun onPreBind(binding: FragmentActionMd2Binding) = Unit
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,88 @@
 | 
			
		||||
package com.topjohnwu.magisk.ui.module
 | 
			
		||||
 | 
			
		||||
import android.view.MenuItem
 | 
			
		||||
import androidx.databinding.ObservableArrayList
 | 
			
		||||
import androidx.lifecycle.LiveData
 | 
			
		||||
import androidx.lifecycle.MutableLiveData
 | 
			
		||||
import androidx.lifecycle.viewModelScope
 | 
			
		||||
import com.topjohnwu.magisk.R
 | 
			
		||||
import com.topjohnwu.magisk.arch.BaseViewModel
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.synchronized
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.timeFormatStandard
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.toTime
 | 
			
		||||
import com.topjohnwu.magisk.core.utils.MediaStoreUtils
 | 
			
		||||
import com.topjohnwu.magisk.core.utils.MediaStoreUtils.outputStream
 | 
			
		||||
import com.topjohnwu.magisk.events.SnackbarEvent
 | 
			
		||||
import com.topjohnwu.magisk.ui.flash.ConsoleItem
 | 
			
		||||
import com.topjohnwu.superuser.CallbackList
 | 
			
		||||
import com.topjohnwu.superuser.Shell
 | 
			
		||||
import kotlinx.coroutines.Dispatchers
 | 
			
		||||
import kotlinx.coroutines.launch
 | 
			
		||||
import kotlinx.coroutines.withContext
 | 
			
		||||
import timber.log.Timber
 | 
			
		||||
import java.io.IOException
 | 
			
		||||
 | 
			
		||||
class ActionViewModel : BaseViewModel() {
 | 
			
		||||
 | 
			
		||||
    enum class State {
 | 
			
		||||
        RUNNING, SUCCESS, FAILED
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private val _state = MutableLiveData(State.RUNNING)
 | 
			
		||||
    val state: LiveData<State> get() = _state
 | 
			
		||||
 | 
			
		||||
    val items = ObservableArrayList<ConsoleItem>()
 | 
			
		||||
    lateinit var args: ActionFragmentArgs
 | 
			
		||||
 | 
			
		||||
    private val logItems = mutableListOf<String>().synchronized()
 | 
			
		||||
    private val outItems = object : CallbackList<String>() {
 | 
			
		||||
        override fun onAddElement(e: String?) {
 | 
			
		||||
            e ?: return
 | 
			
		||||
            items.add(ConsoleItem(e))
 | 
			
		||||
            logItems.add(e)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun startRunAction() = viewModelScope.launch {
 | 
			
		||||
        onResult(withContext(Dispatchers.IO) {
 | 
			
		||||
            try {
 | 
			
		||||
                Shell.cmd("run_action \'${args.id}\'")
 | 
			
		||||
                    .to(outItems, logItems)
 | 
			
		||||
                    .exec().isSuccess
 | 
			
		||||
            } catch (e: IOException) {
 | 
			
		||||
                Timber.e(e)
 | 
			
		||||
                false
 | 
			
		||||
            }
 | 
			
		||||
        })
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun onResult(success: Boolean) {
 | 
			
		||||
        _state.value = if (success) State.SUCCESS else State.FAILED
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun onMenuItemClicked(item: MenuItem): Boolean {
 | 
			
		||||
        when (item.itemId) {
 | 
			
		||||
            R.id.action_save -> savePressed()
 | 
			
		||||
        }
 | 
			
		||||
        return true
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun savePressed() = withExternalRW {
 | 
			
		||||
        viewModelScope.launch(Dispatchers.IO) {
 | 
			
		||||
            val name = "%s_action_log_%s.log".format(
 | 
			
		||||
                args.name,
 | 
			
		||||
                System.currentTimeMillis().toTime(timeFormatStandard)
 | 
			
		||||
            )
 | 
			
		||||
            val file = MediaStoreUtils.getFile(name)
 | 
			
		||||
            file.uri.outputStream().bufferedWriter().use { writer ->
 | 
			
		||||
                synchronized(logItems) {
 | 
			
		||||
                    logItems.forEach {
 | 
			
		||||
                        writer.write(it)
 | 
			
		||||
                        writer.newLine()
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            SnackbarEvent(file.toString()).publish()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -25,6 +25,7 @@ class LocalModuleRvItem(
 | 
			
		||||
    override val layoutRes = R.layout.item_module_md2
 | 
			
		||||
 | 
			
		||||
    val showNotice: Boolean
 | 
			
		||||
    val showAction: Boolean
 | 
			
		||||
    val noticeText: TextHolder
 | 
			
		||||
 | 
			
		||||
    init {
 | 
			
		||||
@@ -35,6 +36,7 @@ class LocalModuleRvItem(
 | 
			
		||||
        showNotice = zygiskUnloaded ||
 | 
			
		||||
            (Info.isZygiskEnabled && isRiru) ||
 | 
			
		||||
            (!Info.isZygiskEnabled && isZygisk)
 | 
			
		||||
        showAction = item.hasAction && !showNotice
 | 
			
		||||
        noticeText =
 | 
			
		||||
            when {
 | 
			
		||||
                zygiskUnloaded -> CoreR.string.zygisk_module_unloaded.asText()
 | 
			
		||||
 
 | 
			
		||||
@@ -4,8 +4,10 @@ import android.net.Uri
 | 
			
		||||
import androidx.databinding.Bindable
 | 
			
		||||
import androidx.lifecycle.MutableLiveData
 | 
			
		||||
import com.topjohnwu.magisk.BR
 | 
			
		||||
import com.topjohnwu.magisk.MainDirections
 | 
			
		||||
import com.topjohnwu.magisk.R
 | 
			
		||||
import com.topjohnwu.magisk.arch.AsyncLoadViewModel
 | 
			
		||||
import com.topjohnwu.magisk.core.Const
 | 
			
		||||
import com.topjohnwu.magisk.core.Info
 | 
			
		||||
import com.topjohnwu.magisk.core.base.ContentResultCallback
 | 
			
		||||
import com.topjohnwu.magisk.core.model.module.LocalModule
 | 
			
		||||
@@ -96,6 +98,10 @@ class ModuleViewModel : AsyncLoadViewModel() {
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun runAction(id: String, name: String) {
 | 
			
		||||
        MainDirections.actionActionFragment(id, name).navigate()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        private val uri = MutableLiveData<Uri?>()
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -78,8 +78,8 @@ class SuperuserViewModel(
 | 
			
		||||
                            this@SuperuserViewModel, policy,
 | 
			
		||||
                            info.packageName,
 | 
			
		||||
                            info.sharedUserId != null,
 | 
			
		||||
                            info.applicationInfo.loadIcon(pm),
 | 
			
		||||
                            info.applicationInfo.getLabel(pm)
 | 
			
		||||
                            info.applicationInfo?.loadIcon(pm) ?: pm.defaultActivityIcon,
 | 
			
		||||
                            info.applicationInfo?.getLabel(pm) ?: info.packageName
 | 
			
		||||
                        )
 | 
			
		||||
                    } catch (e: PackageManager.NameNotFoundException) {
 | 
			
		||||
                        null
 | 
			
		||||
 
 | 
			
		||||
@@ -111,7 +111,7 @@ class SuRequestViewModel(
 | 
			
		||||
            // shared UID. We have no way to know where this request comes from.
 | 
			
		||||
            icon = pm.defaultActivityIcon
 | 
			
		||||
            title = "[SharedUID] ${info.sharedUserId}"
 | 
			
		||||
            packageName = info.sharedUserId
 | 
			
		||||
            packageName = info.sharedUserId.toString()
 | 
			
		||||
        } else {
 | 
			
		||||
            val prefix = if (info.sharedUserId == null) "" else "[SharedUID] "
 | 
			
		||||
            icon = app.loadIcon(pm)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										5
									
								
								app/apk/src/main/res/drawable/ic_action_md2.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								app/apk/src/main/res/drawable/ic_action_md2.xml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#000000" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
 | 
			
		||||
 | 
			
		||||
    <path android:fillColor="@android:color/white" android:pathData="M8,5v14l11,-7z"/>
 | 
			
		||||
 | 
			
		||||
</vector>
 | 
			
		||||
							
								
								
									
										68
									
								
								app/apk/src/main/res/layout/fragment_action_md2.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								app/apk/src/main/res/layout/fragment_action_md2.xml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,68 @@
 | 
			
		||||
<?xml version="1.0" encoding="utf-8"?>
 | 
			
		||||
<layout xmlns:android="http://schemas.android.com/apk/res/android"
 | 
			
		||||
    xmlns:app="http://schemas.android.com/apk/res-auto"
 | 
			
		||||
    xmlns:tools="http://schemas.android.com/tools">
 | 
			
		||||
 | 
			
		||||
    <data>
 | 
			
		||||
 | 
			
		||||
        <variable
 | 
			
		||||
            name="viewModel"
 | 
			
		||||
            type="com.topjohnwu.magisk.ui.module.ActionViewModel" />
 | 
			
		||||
 | 
			
		||||
    </data>
 | 
			
		||||
 | 
			
		||||
    <androidx.coordinatorlayout.widget.CoordinatorLayout
 | 
			
		||||
        android:layout_width="match_parent"
 | 
			
		||||
        android:layout_height="match_parent">
 | 
			
		||||
 | 
			
		||||
        <HorizontalScrollView
 | 
			
		||||
            android:layout_width="match_parent"
 | 
			
		||||
            android:layout_height="match_parent"
 | 
			
		||||
            android:layout_marginTop="@dimen/internal_action_bar_size"
 | 
			
		||||
            app:layout_fitsSystemWindowsInsets="top"
 | 
			
		||||
            tools:layout_marginTop="@dimen/internal_action_bar_size">
 | 
			
		||||
 | 
			
		||||
            <androidx.recyclerview.widget.RecyclerView
 | 
			
		||||
                android:id="@+id/flash_content"
 | 
			
		||||
                scrollToLast="@{true}"
 | 
			
		||||
                android:layout_width="wrap_content"
 | 
			
		||||
                android:layout_height="match_parent"
 | 
			
		||||
                android:clipToPadding="false"
 | 
			
		||||
                android:orientation="vertical"
 | 
			
		||||
                app:fitsSystemWindowsInsets="start|end|bottom"
 | 
			
		||||
                app:items="@{viewModel.items}"
 | 
			
		||||
                app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
 | 
			
		||||
                app:layout_behavior="@string/appbar_scrolling_view_behavior"
 | 
			
		||||
                tools:listitem="@layout/item_console_md2" />
 | 
			
		||||
 | 
			
		||||
        </HorizontalScrollView>
 | 
			
		||||
 | 
			
		||||
        <com.google.android.material.floatingactionbutton.ExtendedFloatingActionButton
 | 
			
		||||
            android:id="@+id/close_btn"
 | 
			
		||||
            android:visibility="gone"
 | 
			
		||||
            android:layout_width="wrap_content"
 | 
			
		||||
            android:layout_height="wrap_content"
 | 
			
		||||
            android:layout_gravity="bottom|end"
 | 
			
		||||
            android:layout_margin="@dimen/l1"
 | 
			
		||||
            android:layout_marginBottom="@dimen/l1"
 | 
			
		||||
            android:clickable="true"
 | 
			
		||||
            android:enabled="true"
 | 
			
		||||
            android:focusable="true"
 | 
			
		||||
            android:text="@string/close"
 | 
			
		||||
            android:textAllCaps="false"
 | 
			
		||||
            android:textColor="?colorOnPrimary"
 | 
			
		||||
            android:textStyle="bold"
 | 
			
		||||
            app:backgroundTint="?colorPrimary"
 | 
			
		||||
            app:icon="@drawable/ic_close_md2"
 | 
			
		||||
            app:iconTint="?colorOnPrimary"
 | 
			
		||||
            app:layout_fitsSystemWindowsInsets="bottom" />
 | 
			
		||||
 | 
			
		||||
        <androidx.coordinatorlayout.widget.CoordinatorLayout
 | 
			
		||||
            android:id="@+id/snackbar_container"
 | 
			
		||||
            android:layout_width="match_parent"
 | 
			
		||||
            android:layout_height="match_parent"
 | 
			
		||||
            app:fitsSystemWindowsInsets="top|bottom" />
 | 
			
		||||
 | 
			
		||||
    </androidx.coordinatorlayout.widget.CoordinatorLayout>
 | 
			
		||||
 | 
			
		||||
</layout>
 | 
			
		||||
@@ -189,12 +189,32 @@
 | 
			
		||||
                    android:textColor="?colorError"
 | 
			
		||||
                    app:layout_constraintBottom_toBottomOf="@+id/module_remove"
 | 
			
		||||
                    app:layout_constraintEnd_toStartOf="@+id/bottom_bar_barrier"
 | 
			
		||||
                    app:layout_constraintStart_toStartOf="parent"
 | 
			
		||||
                    app:layout_constraintStart_toEndOf="@id/module_config"
 | 
			
		||||
                    app:layout_constraintTop_toTopOf="@+id/module_remove"
 | 
			
		||||
                    tools:lines="2"
 | 
			
		||||
                    tools:text="@tools:sample/lorem/random"
 | 
			
		||||
                    tools:visibility="visible" />
 | 
			
		||||
 | 
			
		||||
                <Button
 | 
			
		||||
                    android:id="@+id/module_config"
 | 
			
		||||
                    style="@style/WidgetFoundation.Button.Text"
 | 
			
		||||
                    goneUnless="@{item.showAction}"
 | 
			
		||||
                    android:layout_width="wrap_content"
 | 
			
		||||
                    android:layout_height="wrap_content"
 | 
			
		||||
                    android:clickable="true"
 | 
			
		||||
                    android:enabled="@{item.enabled}"
 | 
			
		||||
                    android:focusable="true"
 | 
			
		||||
                    android:onClick="@{() -> viewModel.runAction(item.item.id, item.item.name)}"
 | 
			
		||||
                    android:text="@string/module_action"
 | 
			
		||||
                    android:textAllCaps="false"
 | 
			
		||||
                    android:visibility="gone"
 | 
			
		||||
                    app:icon="@drawable/ic_action_md2"
 | 
			
		||||
                    app:iconGravity="textStart"
 | 
			
		||||
                    app:layout_constraintBottom_toBottomOf="@+id/module_remove"
 | 
			
		||||
                    app:layout_constraintTop_toTopOf="@+id/module_remove"
 | 
			
		||||
                    app:layout_constraintStart_toStartOf="parent"
 | 
			
		||||
                    app:srcCompat="@drawable/ic_download_md2" />
 | 
			
		||||
 | 
			
		||||
            </androidx.constraintlayout.widget.ConstraintLayout>
 | 
			
		||||
 | 
			
		||||
        </com.google.android.material.card.MaterialCardView>
 | 
			
		||||
 
 | 
			
		||||
@@ -53,6 +53,21 @@
 | 
			
		||||
 | 
			
		||||
    </fragment>
 | 
			
		||||
 | 
			
		||||
    <fragment
 | 
			
		||||
        android:id="@+id/actionFragment"
 | 
			
		||||
        android:name="com.topjohnwu.magisk.ui.module.ActionFragment"
 | 
			
		||||
        android:label="ActionFragment"
 | 
			
		||||
        tools:layout="@layout/fragment_action_md2" >
 | 
			
		||||
 | 
			
		||||
        <argument
 | 
			
		||||
            android:name="id"
 | 
			
		||||
            app:argType="string" />
 | 
			
		||||
 | 
			
		||||
        <argument
 | 
			
		||||
            android:name="name"
 | 
			
		||||
            app:argType="string" />
 | 
			
		||||
    </fragment>
 | 
			
		||||
 | 
			
		||||
    <fragment
 | 
			
		||||
        android:id="@+id/installFragment"
 | 
			
		||||
        android:name="com.topjohnwu.magisk.ui.install.InstallFragment"
 | 
			
		||||
@@ -152,4 +167,12 @@
 | 
			
		||||
        app:popEnterAnim="@anim/fragment_enter_pop"
 | 
			
		||||
        app:popExitAnim="@anim/fragment_exit_pop" />
 | 
			
		||||
 | 
			
		||||
    <action
 | 
			
		||||
        android:id="@+id/action_actionFragment"
 | 
			
		||||
        app:destination="@id/actionFragment"
 | 
			
		||||
        app:enterAnim="@anim/fragment_enter"
 | 
			
		||||
        app:exitAnim="@anim/fragment_exit"
 | 
			
		||||
        app:popEnterAnim="@anim/fragment_enter_pop"
 | 
			
		||||
        app:popExitAnim="@anim/fragment_exit_pop" />
 | 
			
		||||
 | 
			
		||||
</navigation>
 | 
			
		||||
 
 | 
			
		||||
@@ -19,6 +19,7 @@ android {
 | 
			
		||||
        buildConfigField("int", "APP_VERSION_CODE", "${Config.versionCode}")
 | 
			
		||||
        buildConfigField("String", "APP_VERSION_NAME", "\"${Config.version}\"")
 | 
			
		||||
        buildConfigField("int", "STUB_VERSION", Config.stubVersion)
 | 
			
		||||
        consumerProguardFile("proguard-rules.pro")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    buildFeatures {
 | 
			
		||||
 
 | 
			
		||||
@@ -22,36 +22,21 @@
 | 
			
		||||
  int mActivityHandlesConfigFlags;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
# main
 | 
			
		||||
-keep,allowoptimization public class com.topjohnwu.magisk.signing.SignBoot {
 | 
			
		||||
    public static void main(java.lang.String[]);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
# Strip Timber verbose and debug logging
 | 
			
		||||
-assumenosideeffects class timber.log.Timber$Tree {
 | 
			
		||||
  public void v(**);
 | 
			
		||||
  public void d(**);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
# https://github.com/square/retrofit/issues/3751#issuecomment-1192043644
 | 
			
		||||
# Keep generic signature of Call, Response (R8 full mode strips signatures from non-kept items).
 | 
			
		||||
-keep,allowobfuscation,allowshrinking interface retrofit2.Call
 | 
			
		||||
-keep,allowobfuscation,allowshrinking class retrofit2.Response
 | 
			
		||||
 | 
			
		||||
# With R8 full mode generic signatures are stripped for classes that are not
 | 
			
		||||
# kept. Suspend functions are wrapped in continuations where the type argument
 | 
			
		||||
# is used.
 | 
			
		||||
-keep,allowobfuscation,allowshrinking class kotlin.coroutines.Continuation
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Excessive obfuscation
 | 
			
		||||
-repackageclasses 'a'
 | 
			
		||||
-flattenpackagehierarchy
 | 
			
		||||
-allowaccessmodification
 | 
			
		||||
 | 
			
		||||
-obfuscationdictionary ../dict.txt
 | 
			
		||||
-classobfuscationdictionary ../dict.txt
 | 
			
		||||
-packageobfuscationdictionary ../dict.txt
 | 
			
		||||
 | 
			
		||||
-dontwarn org.bouncycastle.jsse.BCSSLParameters
 | 
			
		||||
-dontwarn org.bouncycastle.jsse.BCSSLSocket
 | 
			
		||||
-dontwarn org.bouncycastle.jsse.provider.BouncyCastleJsseProvider
 | 
			
		||||
@@ -37,6 +37,7 @@ data class LocalModule(
 | 
			
		||||
    val isRiru: Boolean get() = (id == "riru-core") || riruFolder.exists()
 | 
			
		||||
    val isZygisk: Boolean get() = zygiskFolder.exists()
 | 
			
		||||
    val zygiskUnloaded: Boolean get() = unloaded.exists()
 | 
			
		||||
    val hasAction: Boolean;
 | 
			
		||||
 | 
			
		||||
    var enable: Boolean
 | 
			
		||||
        get() = !disableFile.exists()
 | 
			
		||||
@@ -100,6 +101,8 @@ data class LocalModule(
 | 
			
		||||
        if (name.isEmpty()) {
 | 
			
		||||
            name = id
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        hasAction = RootUtils.fs.getFile(path, "action.sh").exists()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    suspend fun fetch(): Boolean {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
package com.topjohnwu.magisk.core.model.su
 | 
			
		||||
 | 
			
		||||
import android.content.pm.PackageInfo
 | 
			
		||||
import android.content.pm.ApplicationInfo
 | 
			
		||||
import android.content.pm.PackageManager
 | 
			
		||||
import androidx.room.Entity
 | 
			
		||||
import androidx.room.PrimaryKey
 | 
			
		||||
@@ -24,7 +24,7 @@ class SuLog(
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun PackageManager.createSuLog(
 | 
			
		||||
    info: PackageInfo,
 | 
			
		||||
    info: ApplicationInfo,
 | 
			
		||||
    toUid: Int,
 | 
			
		||||
    fromPid: Int,
 | 
			
		||||
    command: String,
 | 
			
		||||
@@ -33,13 +33,12 @@ fun PackageManager.createSuLog(
 | 
			
		||||
    context: String,
 | 
			
		||||
    gids: String,
 | 
			
		||||
): SuLog {
 | 
			
		||||
    val appInfo = info.applicationInfo
 | 
			
		||||
    return SuLog(
 | 
			
		||||
        fromUid = appInfo.uid,
 | 
			
		||||
        fromUid = info.uid,
 | 
			
		||||
        toUid = toUid,
 | 
			
		||||
        fromPid = fromPid,
 | 
			
		||||
        packageName = getNameForUid(appInfo.uid)!!,
 | 
			
		||||
        appName = appInfo.getLabel(this),
 | 
			
		||||
        packageName = getNameForUid(info.uid)!!,
 | 
			
		||||
        appName = info.getLabel(this),
 | 
			
		||||
        command = command,
 | 
			
		||||
        action = policy,
 | 
			
		||||
        target = target,
 | 
			
		||||
 
 | 
			
		||||
@@ -64,7 +64,7 @@ object SuCallbackHandler {
 | 
			
		||||
        val pm = context.packageManager
 | 
			
		||||
 | 
			
		||||
        val log = runCatching {
 | 
			
		||||
            pm.getPackageInfo(fromUid, pid)?.let {
 | 
			
		||||
            pm.getPackageInfo(fromUid, pid)?.applicationInfo?.let {
 | 
			
		||||
                pm.createSuLog(it, toUid, pid, command, policy, target, seContext, gids)
 | 
			
		||||
            }
 | 
			
		||||
        }.getOrNull() ?: createSuLog(fromUid, toUid, pid, command, policy, target, seContext, gids)
 | 
			
		||||
 
 | 
			
		||||
@@ -125,8 +125,9 @@ object AppMigration {
 | 
			
		||||
        apk: File, out: OutputStream,
 | 
			
		||||
        pkg: String, label: CharSequence
 | 
			
		||||
    ): Boolean {
 | 
			
		||||
        val info = context.packageManager.getPackageArchiveInfo(apk.path, 0) ?: return false
 | 
			
		||||
        val origLabel = info.applicationInfo.nonLocalizedLabel.toString()
 | 
			
		||||
        val pm = context.packageManager
 | 
			
		||||
        val info = pm.getPackageArchiveInfo(apk.path, 0)?.applicationInfo ?: return false
 | 
			
		||||
        val origLabel = info.nonLocalizedLabel.toString()
 | 
			
		||||
        try {
 | 
			
		||||
            JarMap.open(apk, true).use { jar ->
 | 
			
		||||
                val je = jar.getJarEntry(ANDROID_MANIFEST)
 | 
			
		||||
@@ -190,11 +191,12 @@ object AppMigration {
 | 
			
		||||
 | 
			
		||||
        // Install and auto launch app
 | 
			
		||||
        val session = APKInstall.startSession(activity, pkg, onFailure) {
 | 
			
		||||
            Config.suManager = pkg
 | 
			
		||||
            Shell.cmd("touch $AppApkPath").exec()
 | 
			
		||||
            launchApp(activity, pkg)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        Config.suManager = pkg
 | 
			
		||||
        val cmd = "touch $AppApkPath; adb_pm_install $repack $pkg"
 | 
			
		||||
        val cmd = "adb_pm_install $repack $pkg"
 | 
			
		||||
        if (Shell.cmd(cmd).exec().isSuccess) return true
 | 
			
		||||
 | 
			
		||||
        try {
 | 
			
		||||
@@ -239,11 +241,12 @@ object AppMigration {
 | 
			
		||||
        }
 | 
			
		||||
        val apk = StubApk.current(activity)
 | 
			
		||||
        val session = APKInstall.startSession(activity, APP_PACKAGE_NAME, onFailure) {
 | 
			
		||||
            Config.suManager = ""
 | 
			
		||||
            Shell.cmd("touch $AppApkPath").exec()
 | 
			
		||||
            launchApp(activity, APP_PACKAGE_NAME)
 | 
			
		||||
            dialog.dismiss()
 | 
			
		||||
        }
 | 
			
		||||
        Config.suManager = ""
 | 
			
		||||
        val cmd = "touch $AppApkPath; adb_pm_install $apk $APP_PACKAGE_NAME"
 | 
			
		||||
        val cmd = "adb_pm_install $apk $APP_PACKAGE_NAME"
 | 
			
		||||
        if (Shell.cmd(cmd).await().isSuccess) return
 | 
			
		||||
        val success = withContext(Dispatchers.IO) {
 | 
			
		||||
            try {
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,6 @@ import android.system.ErrnoException
 | 
			
		||||
import android.system.Os
 | 
			
		||||
import android.system.OsConstants
 | 
			
		||||
import android.system.OsConstants.O_WRONLY
 | 
			
		||||
import android.widget.Toast
 | 
			
		||||
import androidx.annotation.WorkerThread
 | 
			
		||||
import androidx.core.os.postDelayed
 | 
			
		||||
import com.topjohnwu.magisk.StubApk
 | 
			
		||||
@@ -15,13 +14,10 @@ import com.topjohnwu.magisk.core.BuildConfig
 | 
			
		||||
import com.topjohnwu.magisk.core.Config
 | 
			
		||||
import com.topjohnwu.magisk.core.Const
 | 
			
		||||
import com.topjohnwu.magisk.core.Info
 | 
			
		||||
import com.topjohnwu.magisk.core.R
 | 
			
		||||
import com.topjohnwu.magisk.core.di.ServiceLocator
 | 
			
		||||
import com.topjohnwu.magisk.core.isRunningAsStub
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.copyAll
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.copyAndClose
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.reboot
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.toast
 | 
			
		||||
import com.topjohnwu.magisk.core.ktx.writeTo
 | 
			
		||||
import com.topjohnwu.magisk.core.utils.DummyList
 | 
			
		||||
import com.topjohnwu.magisk.core.utils.MediaStoreUtils
 | 
			
		||||
@@ -585,6 +581,8 @@ abstract class MagiskInstallImpl protected constructor(
 | 
			
		||||
 | 
			
		||||
    protected suspend fun fixEnv() = extractFiles() && "fix_env $installDir".sh().isSuccess
 | 
			
		||||
 | 
			
		||||
    protected fun restore() = findImage() && "restore_imgs $srcBoot".sh().isSuccess
 | 
			
		||||
 | 
			
		||||
    protected fun uninstall() = "run_uninstaller $AppApkPath".sh().isSuccess
 | 
			
		||||
 | 
			
		||||
    @WorkerThread
 | 
			
		||||
@@ -608,11 +606,10 @@ abstract class MagiskInstallImpl protected constructor(
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
abstract class MagiskInstaller(
 | 
			
		||||
abstract class ConsoleInstaller(
 | 
			
		||||
    console: MutableList<String>,
 | 
			
		||||
    logs: MutableList<String>
 | 
			
		||||
) : MagiskInstallImpl(console, logs) {
 | 
			
		||||
 | 
			
		||||
    override suspend fun exec(): Boolean {
 | 
			
		||||
        val success = super.exec()
 | 
			
		||||
        if (success) {
 | 
			
		||||
@@ -622,40 +619,51 @@ abstract class MagiskInstaller(
 | 
			
		||||
        }
 | 
			
		||||
        return success
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
abstract class CallBackInstaller : MagiskInstallImpl(DummyList, DummyList) {
 | 
			
		||||
    suspend fun exec(callback: (Boolean) -> Unit): Boolean {
 | 
			
		||||
        val success = exec()
 | 
			
		||||
        callback(success)
 | 
			
		||||
        return success
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class MagiskInstaller {
 | 
			
		||||
 | 
			
		||||
    class Patch(
 | 
			
		||||
        private val uri: Uri,
 | 
			
		||||
        console: MutableList<String>,
 | 
			
		||||
        logs: MutableList<String>
 | 
			
		||||
    ) : MagiskInstaller(console, logs) {
 | 
			
		||||
    ) : ConsoleInstaller(console, logs) {
 | 
			
		||||
        override suspend fun operations() = patchFile(uri)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    class SecondSlot(
 | 
			
		||||
        console: MutableList<String>,
 | 
			
		||||
        logs: MutableList<String>
 | 
			
		||||
    ) : MagiskInstaller(console, logs) {
 | 
			
		||||
    ) : ConsoleInstaller(console, logs) {
 | 
			
		||||
        override suspend fun operations() = secondSlot()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    class Direct(
 | 
			
		||||
        console: MutableList<String>,
 | 
			
		||||
        logs: MutableList<String>
 | 
			
		||||
    ) : MagiskInstaller(console, logs) {
 | 
			
		||||
    ) : ConsoleInstaller(console, logs) {
 | 
			
		||||
        override suspend fun operations() = direct()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    class Emulator(
 | 
			
		||||
        console: MutableList<String>,
 | 
			
		||||
        logs: MutableList<String>
 | 
			
		||||
    ) : MagiskInstaller(console, logs) {
 | 
			
		||||
    ) : ConsoleInstaller(console, logs) {
 | 
			
		||||
        override suspend fun operations() = fixEnv()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    class Uninstall(
 | 
			
		||||
        console: MutableList<String>,
 | 
			
		||||
        logs: MutableList<String>
 | 
			
		||||
    ) : MagiskInstallImpl(console, logs) {
 | 
			
		||||
    ) : ConsoleInstaller(console, logs) {
 | 
			
		||||
        override suspend fun operations() = uninstall()
 | 
			
		||||
 | 
			
		||||
        override suspend fun exec(): Boolean {
 | 
			
		||||
@@ -669,19 +677,11 @@ abstract class MagiskInstaller(
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    class FixEnv(private val callback: () -> Unit) : MagiskInstallImpl(DummyList, DummyList) {
 | 
			
		||||
        override suspend fun operations() = fixEnv()
 | 
			
		||||
    class Restore : CallBackInstaller() {
 | 
			
		||||
        override suspend fun operations() = restore()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
        override suspend fun exec(): Boolean {
 | 
			
		||||
            val success = super.exec()
 | 
			
		||||
            callback()
 | 
			
		||||
            context.toast(
 | 
			
		||||
                if (success) R.string.reboot_delay_toast else R.string.setup_fail,
 | 
			
		||||
                Toast.LENGTH_LONG
 | 
			
		||||
            )
 | 
			
		||||
            if (success)
 | 
			
		||||
                UiThreadHandler.handler.postDelayed(5000) { reboot() }
 | 
			
		||||
            return success
 | 
			
		||||
        }
 | 
			
		||||
    class FixEnv : CallBackInstaller() {
 | 
			
		||||
        override suspend fun operations() = fixEnv()
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,32 @@
 | 
			
		||||
package com.topjohnwu.magisk.core.utils;
 | 
			
		||||
 | 
			
		||||
import android.os.Build;
 | 
			
		||||
 | 
			
		||||
import java.nio.file.attribute.FileTime;
 | 
			
		||||
import java.util.zip.ZipEntry;
 | 
			
		||||
 | 
			
		||||
public class Desugar {
 | 
			
		||||
    public static FileTime getLastModifiedTime(ZipEntry entry) {
 | 
			
		||||
        if (Build.VERSION.SDK_INT >= 26) {
 | 
			
		||||
            return entry.getLastModifiedTime();
 | 
			
		||||
        } else {
 | 
			
		||||
            return FileTime.fromMillis(entry.getTime());
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public static FileTime getLastAccessTime(ZipEntry entry) {
 | 
			
		||||
        if (Build.VERSION.SDK_INT >= 26) {
 | 
			
		||||
            return entry.getLastAccessTime();
 | 
			
		||||
        } else {
 | 
			
		||||
            return null;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public static FileTime getCreationTime(ZipEntry entry) {
 | 
			
		||||
        if (Build.VERSION.SDK_INT >= 26) {
 | 
			
		||||
            return entry.getCreationTime();
 | 
			
		||||
        } else {
 | 
			
		||||
            return null;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -10,10 +10,9 @@
 | 
			
		||||
    <string name="section_theme">עיצוב</string>
 | 
			
		||||
    <string name="denylist">רשימת דחייה</string>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    <!--Home-->
 | 
			
		||||
    <string name="no_connection">אין חיבור זמין</string>
 | 
			
		||||
    <string name="app_changelog">רשימת שינויים</string>
 | 
			
		||||
    <string name="app_changelog">יומן שינויים</string>
 | 
			
		||||
    <string name="loading">טוען…</string>
 | 
			
		||||
    <string name="update">עדכון</string>
 | 
			
		||||
    <string name="not_available">ל/ז</string>
 | 
			
		||||
@@ -45,16 +44,16 @@
 | 
			
		||||
    <string name="install_inactive_slot_msg">ההתקן שלך ייאלץ אתחול לחריץ הלא פעיל הנוכחי שלך לאחר הפעלה מחדש!\nיש להשתמש באפשרות זו רק לאחר ביצוע OTA בלבד.\nלהמשיך?</string>
 | 
			
		||||
    <string name="setup_title">התקנה נוספת</string>
 | 
			
		||||
    <string name="select_patch_file">בחירה והתקנת קובץ</string>
 | 
			
		||||
    <string name="patch_file_msg">בחירת תמונה גולמית (*.img) או ODIN קובץ tar (*.tar)</string>
 | 
			
		||||
    <string name="patch_file_msg">בחירת תמונה גולמית (*.img) או ODIN tarfile (*.tar) או payload.bin (*.bin)</string>
 | 
			
		||||
    <string name="reboot_delay_toast">מאתחל בעוד 5 שניות…</string>
 | 
			
		||||
    <string name="flash_screen_title">התקנה</string>
 | 
			
		||||
 | 
			
		||||
    <!--Superuser-->
 | 
			
		||||
    <string name="su_request_title">בקשות משתמש על</string>
 | 
			
		||||
    <string name="touch_filtered_warning">מכיוון שיישום מסתיר בקשה של משתמש על, Magisk לא יכול לאמת את תגובתך</string>
 | 
			
		||||
    <string name="deny">דחה</string>
 | 
			
		||||
    <string name="deny">דחייה</string>
 | 
			
		||||
    <string name="prompt">מיידי</string>
 | 
			
		||||
    <string name="grant">הענק</string>
 | 
			
		||||
    <string name="grant">הענקה</string>
 | 
			
		||||
    <string name="su_warning">מעניק גישה מלאה להתקן שלך.\nיש לדחות באי וודאות!</string>
 | 
			
		||||
    <string name="forever">לצמיתות</string>
 | 
			
		||||
    <string name="once">פעם אחת</string>
 | 
			
		||||
@@ -62,24 +61,24 @@
 | 
			
		||||
    <string name="twentymin">20 דקות</string>
 | 
			
		||||
    <string name="thirtymin">חצי שעה</string>
 | 
			
		||||
    <string name="sixtymin">שעה</string>
 | 
			
		||||
    <string name="su_allow_toast">%1$s הוענקו הרשאות משתמש עבור</string>
 | 
			
		||||
    <string name="su_deny_toast">%1$s נשללו הרשאות משתמש עבור</string>
 | 
			
		||||
    <string name="su_allow_toast">%1$s קיבל הרשאות משתמש על</string>
 | 
			
		||||
    <string name="su_deny_toast">%1$s נשללו הרשאות משתמש על</string>
 | 
			
		||||
    <string name="su_snack_grant">הרשאות משתמש על עבור %1$s הוענקו</string>
 | 
			
		||||
    <string name="su_snack_deny">הרשאות משתמש על עבור %1$s נשללו</string>
 | 
			
		||||
    <string name="su_snack_notif_on">התראות עבור %1$s פועלות</string>
 | 
			
		||||
    <string name="su_snack_notif_off">התראות עבור %1$s כבויות</string>
 | 
			
		||||
    <string name="su_snack_notif_on">התראות של %1$s מופעלות</string>
 | 
			
		||||
    <string name="su_snack_notif_off">התראות של %1$s מושבתות</string>
 | 
			
		||||
    <string name="su_snack_log_on">יומני רישום עבור %1$s פועלות</string>
 | 
			
		||||
    <string name="su_snack_log_off">יומני רישום עבור %1$s כבויות</string>
 | 
			
		||||
    <string name="su_snack_log_off">יומני רישום עבור %1$s מושבתות</string>
 | 
			
		||||
    <string name="su_revoke_title">להסיר?</string>
 | 
			
		||||
    <string name="su_revoke_msg">נא לאשר שלילת הרשאות עבור %1$s?</string>
 | 
			
		||||
    <string name="toast">הרמת כוסית</string>
 | 
			
		||||
    <string name="none">ללא</string>
 | 
			
		||||
    <string name="superuser_toggle_notification">התראות</string>
 | 
			
		||||
    <string name="superuser_toggle_revoke">הסרה</string>
 | 
			
		||||
    <string name="superuser_policy_none">לא נתבקשו הרשאות משתמש על על ידי שום יישום</string>
 | 
			
		||||
    <string name="superuser_policy_none">טרם נתבקשו הרשאות משתמש על על ידי יישומים</string>
 | 
			
		||||
 | 
			
		||||
    <!--Logs-->
 | 
			
		||||
    <string name="log_data_none">הינך ללא יומן רישום, יש לנסות להשתמש ביישומים מותאמים יותר למשתמש העל שלך</string>
 | 
			
		||||
    <string name="log_data_none">הינך ללא יומן, יש לנסות להשתמש יותר ביישומי השורש שלך</string>
 | 
			
		||||
    <string name="log_data_magisk_none">יומני רישום Magisk ריקים, זה מוזר</string>
 | 
			
		||||
    <string name="menuSaveLog">שמירת יומן רישום</string>
 | 
			
		||||
    <string name="menuClearLog">ניקוי יומן רישום כעת</string>
 | 
			
		||||
@@ -92,7 +91,7 @@
 | 
			
		||||
 | 
			
		||||
    <!--SafetyNet-->
 | 
			
		||||
 | 
			
		||||
    <!-- MagiskHide -->
 | 
			
		||||
    <!--MagiskHide-->
 | 
			
		||||
    <string name="show_system_app">הצגת יישומי מערכת</string>
 | 
			
		||||
    <string name="show_os_app">הצגת יישומי מערכת הפעלה</string>
 | 
			
		||||
    <string name="hide_filter_hint">סינון לפי שם</string>
 | 
			
		||||
@@ -100,14 +99,16 @@
 | 
			
		||||
 | 
			
		||||
    <!--Module-->
 | 
			
		||||
    <string name="no_info_provided">(לא סופק מידע)</string>
 | 
			
		||||
    <string name="reboot_userspace">אתחול מהיר</string>
 | 
			
		||||
    <string name="reboot_userspace">אתחול רך</string>
 | 
			
		||||
    <string name="reboot_recovery">אתחול למצב שחזור</string>
 | 
			
		||||
    <string name="reboot_bootloader">אתחול מצב מנהל האתחול</string>
 | 
			
		||||
    <string name="reboot_download">אתחול מצב הורדה</string>
 | 
			
		||||
    <string name="reboot_bootloader">אתחול לטוען האתחול</string>
 | 
			
		||||
    <string name="reboot_download">אתחול למצב הורדה</string>
 | 
			
		||||
    <string name="reboot_edl">אתחול למצב EDL</string>
 | 
			
		||||
    <string name="reboot_safe_mode">מצב בטוח</string>
 | 
			
		||||
    <string name="module_version_author">%1$s מאת %2$s</string>
 | 
			
		||||
    <string name="module_state_remove">הסרה</string>
 | 
			
		||||
    <string name="module_state_restore">שיחזור</string>
 | 
			
		||||
    <string name="module_action">פעולה</string>
 | 
			
		||||
    <string name="module_state_restore">שחזור</string>
 | 
			
		||||
    <string name="module_action_install_external">התקנה מהאחסון</string>
 | 
			
		||||
    <string name="update_available">עדכונים זמינים</string>
 | 
			
		||||
    <string name="suspend_text_riru">מודול מושעה כי %1$s מופעל</string>
 | 
			
		||||
@@ -117,7 +118,7 @@
 | 
			
		||||
    <string name="confirm_install">להתקין מודול %1$s?</string>
 | 
			
		||||
    <string name="confirm_install_title">אישור התקנה</string>
 | 
			
		||||
 | 
			
		||||
    <!--Settings -->
 | 
			
		||||
    <!--Settings-->
 | 
			
		||||
    <string name="settings_dark_mode_title">מצב עיצוב</string>
 | 
			
		||||
    <string name="settings_dark_mode_message">נא לבחור מצב המתאים ביותר לסגנון שלך!</string>
 | 
			
		||||
    <string name="settings_dark_mode_light">תמיד בהיר</string>
 | 
			
		||||
@@ -128,11 +129,11 @@
 | 
			
		||||
    <string name="settings_hide_app_title">הסתרת היישום Magisk</string>
 | 
			
		||||
    <string name="settings_hide_app_summary">התקנת יישום מתווך עם מזהה חבילה אקראי ותווית שם מותאמת אישית</string>
 | 
			
		||||
    <string name="settings_restore_app_title">שיחזור היישום Magisk</string>
 | 
			
		||||
    <string name="settings_restore_app_summary">יש לבטל את הסתרת היישום ולשחזור אותו ל-APK המקורי</string>
 | 
			
		||||
    <string name="settings_restore_app_summary">ביטול הסתרת היישום ושחזור אל ה-APK המקורי</string>
 | 
			
		||||
    <string name="language">שפה</string>
 | 
			
		||||
    <string name="system_default">(ברירת מחדל מערכת)</string>
 | 
			
		||||
    <string name="settings_check_update_title">בדיקת עדכונים</string>
 | 
			
		||||
    <string name="settings_check_update_summary">בדוק מעת לעת ברקע אם יש עדכונים</string>
 | 
			
		||||
    <string name="settings_check_update_summary">בדיקה מעת לעת ברקע אם יש עדכונים</string>
 | 
			
		||||
    <string name="settings_update_channel_title">ערוץ עדכון</string>
 | 
			
		||||
    <string name="settings_update_stable">יציב</string>
 | 
			
		||||
    <string name="settings_update_beta">בטא</string>
 | 
			
		||||
@@ -161,12 +162,12 @@
 | 
			
		||||
    <string name="settings_su_request_60">60 שניות</string>
 | 
			
		||||
    <string name="superuser_access">גישת משתמש על</string>
 | 
			
		||||
    <string name="auto_response">תגובה אוטומטית</string>
 | 
			
		||||
    <string name="request_timeout">בקש פסק זמן</string>
 | 
			
		||||
    <string name="request_timeout">בקשת פסק זמן</string>
 | 
			
		||||
    <string name="superuser_notification">התראות משתמש על</string>
 | 
			
		||||
    <string name="settings_su_reauth_title">אימות מחדש לאחר שדרוג</string>
 | 
			
		||||
    <string name="settings_su_reauth_summary">אימות מחדש הרשאות של משתמש על לאחר שדרוג יישום</string>
 | 
			
		||||
    <string name="settings_su_tapjack_title">הפעלת הגנת Tapjacking</string>
 | 
			
		||||
    <string name="settings_su_tapjack_summary">תיבת הדו שיח של משתמש העל לא תגיב לקלט כשהיא מוסתרת על ידי חלון או כיסוי אחר</string>
 | 
			
		||||
    <string name="settings_su_tapjack_title">הגנת Tapjacking</string>
 | 
			
		||||
    <string name="settings_su_tapjack_summary">תיבת הדו שיח של משתמש העל לא תגיב לקלט כשהיא מוסתרת על ידי חלון או שכבת על אחרת</string>
 | 
			
		||||
    <string name="settings_su_auth_title">אימות משתמש</string>
 | 
			
		||||
    <string name="settings_su_auth_summary">בקשת אימות משתמש במהלך בקשות משתמש על</string>
 | 
			
		||||
    <string name="settings_su_auth_insecure">לא מוגדרת שיטת אימות בהתקן</string>
 | 
			
		||||
@@ -174,6 +175,8 @@
 | 
			
		||||
    <string name="setting_add_shortcut_summary">הוספת קיצור דרך יפה במסך הבית למקרה שקשה לזהות את השם ואת הסמל לאחר הסתרת היישום</string>
 | 
			
		||||
    <string name="settings_doh_title">DNS על HTTPS</string>
 | 
			
		||||
    <string name="settings_doh_description">עקיפת DNS מורעל במדינות מסוימות</string>
 | 
			
		||||
    <string name="settings_random_name_title">שם פלט אקראי</string>
 | 
			
		||||
    <string name="settings_random_name_description">שם אקראי לקובץ הפלט של תמונות מתוקנות וקבצי tar כדי למנוע זיהוי</string>
 | 
			
		||||
    <string name="multiuser_mode">מצב מרובה משתמשים</string>
 | 
			
		||||
    <string name="settings_owner_only">בעל ההתקן בלבד</string>
 | 
			
		||||
    <string name="settings_owner_manage">אחראי ניהול ההתקן</string>
 | 
			
		||||
@@ -205,10 +208,13 @@
 | 
			
		||||
    <string name="repo_install_title">מתקין %1$s %2$s(%3$d)</string>
 | 
			
		||||
    <string name="download">הורדה</string>
 | 
			
		||||
    <string name="reboot">הפעלה מחדש</string>
 | 
			
		||||
    <string name="close">סגירה</string>
 | 
			
		||||
    <string name="release_notes">הערות שחרור</string>
 | 
			
		||||
    <string name="flashing">צורב…</string>
 | 
			
		||||
    <string name="running">רץ…</string>
 | 
			
		||||
    <string name="done">בוצע!</string>
 | 
			
		||||
    <string name="failure">נכשל</string>
 | 
			
		||||
    <string name="done_action">בוצעה ריצת פעולה של %1$s</string>
 | 
			
		||||
    <string name="failure">נכשל!</string>
 | 
			
		||||
    <string name="hide_app_title">מסתיר את יישום Magisk…</string>
 | 
			
		||||
    <string name="open_link_failed_toast">לא נמצאו יישומים לפתיחת קישור זה</string>
 | 
			
		||||
    <string name="complete_uninstall">הסרה מלאה</string>
 | 
			
		||||
@@ -237,4 +243,5 @@
 | 
			
		||||
    <string name="app_not_found">לא נמצא יישום לטיפול בפעולה זו</string>
 | 
			
		||||
    <string name="reboot_apply_change">ייש להפעיל מחדש כדי להחיל שינויים</string>
 | 
			
		||||
    <string name="restore_app_confirmation">פעולה זו תשחזר את היישום המוסתר חזרה ליישום המקורי. האם בוודאות ברצונך לעשות את זה?</string>
 | 
			
		||||
 | 
			
		||||
</resources>
 | 
			
		||||
 
 | 
			
		||||
@@ -105,6 +105,7 @@
 | 
			
		||||
    <string name="reboot_safe_mode">Modo de segurança</string>
 | 
			
		||||
    <string name="module_version_author">%1$s por %2$s</string>
 | 
			
		||||
    <string name="module_state_remove">Remover</string>
 | 
			
		||||
    <string name="module_action">Ação</string>
 | 
			
		||||
    <string name="module_state_restore">Restaurar</string>
 | 
			
		||||
    <string name="module_action_install_external">Instalar a partir do armazenamento</string>
 | 
			
		||||
    <string name="update_available">Atualização disponível</string>
 | 
			
		||||
@@ -173,7 +174,7 @@
 | 
			
		||||
    <string name="settings_doh_title">DNS sobre HTTPS</string>
 | 
			
		||||
    <string name="settings_doh_description">Solução alternativa para envenenamento de DNS em alguns países</string>
 | 
			
		||||
    <string name="settings_random_name_title">Randomizar nome de saída</string>
 | 
			
		||||
    <string name="settings_random_name_description">Randomize o nome do arquivo de saída de imagens corrigidas e arquivos tar para evitar a detecção</string>
 | 
			
		||||
    <string name="settings_random_name_description">Randomize o nome do arquivo de saída de imagens corrigidas e arquivos tar (*.tar) para evitar a detecção</string>
 | 
			
		||||
    <string name="multiuser_mode">Modo multiusuário</string>
 | 
			
		||||
    <string name="settings_owner_only">Somente proprietário do dispositivo</string>
 | 
			
		||||
    <string name="settings_owner_manage">Gerenciado pelo proprietário do dispositivo</string>
 | 
			
		||||
@@ -205,9 +206,12 @@
 | 
			
		||||
    <string name="repo_install_title">Instalar %1$s %2$s(%3$d)</string>
 | 
			
		||||
    <string name="download">Baixar</string>
 | 
			
		||||
    <string name="reboot">Reiniciar</string>
 | 
			
		||||
    <string name="close">Fechar</string>
 | 
			
		||||
    <string name="release_notes">Notas da atualização</string>
 | 
			
		||||
    <string name="flashing">Flashando…</string>
 | 
			
		||||
    <string name="running">Executando…</string>
 | 
			
		||||
    <string name="done">Concluído!</string>
 | 
			
		||||
    <string name="done_action">Ação de execução de %1$s concluída</string>
 | 
			
		||||
    <string name="failure">Falhou!</string>
 | 
			
		||||
    <string name="hide_app_title">Ocultando o app do Magisk…</string>
 | 
			
		||||
    <string name="open_link_failed_toast">Nenhum app encontrado para abrir o link</string>
 | 
			
		||||
 
 | 
			
		||||
@@ -105,6 +105,7 @@
 | 
			
		||||
    <string name="reboot_safe_mode">Modo de segurança</string>
 | 
			
		||||
    <string name="module_version_author">%1$s por %2$s</string>
 | 
			
		||||
    <string name="module_state_remove">Remover</string>
 | 
			
		||||
    <string name="module_action">Ação</string>
 | 
			
		||||
    <string name="module_state_restore">Restaurar</string>
 | 
			
		||||
    <string name="module_action_install_external">Instalar a partir do armazenamento</string>
 | 
			
		||||
    <string name="update_available">Atualização disponível</string>
 | 
			
		||||
@@ -173,7 +174,7 @@
 | 
			
		||||
    <string name="settings_doh_title">DNS sobre HTTPS</string>
 | 
			
		||||
    <string name="settings_doh_description">Solução alternativa para envenenamento de DNS em alguns países</string>
 | 
			
		||||
    <string name="settings_random_name_title">Randomizar nome de saída</string>
 | 
			
		||||
    <string name="settings_random_name_description">Randomize o nome do arquivo de saída de imagens corrigidas e arquivos tar para evitar a detecção</string>
 | 
			
		||||
    <string name="settings_random_name_description">Randomize o nome do arquivo de saída de imagens corrigidas e arquivos tar (*.tar) para evitar a detecção</string>
 | 
			
		||||
    <string name="multiuser_mode">Modo multiusuário</string>
 | 
			
		||||
    <string name="settings_owner_only">Somente proprietário do dispositivo</string>
 | 
			
		||||
    <string name="settings_owner_manage">Gerenciado pelo proprietário do dispositivo</string>
 | 
			
		||||
@@ -205,9 +206,12 @@
 | 
			
		||||
    <string name="repo_install_title">Instalar %1$s %2$s(%3$d)</string>
 | 
			
		||||
    <string name="download">Baixar</string>
 | 
			
		||||
    <string name="reboot">Reiniciar</string>
 | 
			
		||||
    <string name="close">Fechar</string>
 | 
			
		||||
    <string name="release_notes">Notas da atualização</string>
 | 
			
		||||
    <string name="flashing">Flashando…</string>
 | 
			
		||||
    <string name="running">Executando…</string>
 | 
			
		||||
    <string name="done">Concluído!</string>
 | 
			
		||||
    <string name="done_action">Ação de execução de %1$s concluída</string>
 | 
			
		||||
    <string name="failure">Falhou!</string>
 | 
			
		||||
    <string name="hide_app_title">Ocultando o app do Magisk…</string>
 | 
			
		||||
    <string name="open_link_failed_toast">Nenhum app encontrado para abrir o link</string>
 | 
			
		||||
 
 | 
			
		||||
@@ -87,6 +87,9 @@
 | 
			
		||||
    <string name="logs_cleared">Логи успешно очищены</string>
 | 
			
		||||
    <string name="pid">PID: %1$d</string>
 | 
			
		||||
    <string name="target_uid">Целевой UID: %1$d</string>
 | 
			
		||||
    <string name="target_pid">Целевой PID пространства имён: %s</string>
 | 
			
		||||
    <string name="selinux_context">Контекст SELinux: %s</string>
 | 
			
		||||
    <string name="supp_group">Дополнительная группа: %s</string>
 | 
			
		||||
 | 
			
		||||
    <!--SafetyNet-->
 | 
			
		||||
 | 
			
		||||
@@ -164,11 +167,16 @@
 | 
			
		||||
    <string name="settings_su_reauth_title">Повторная аутентификация</string>
 | 
			
		||||
    <string name="settings_su_reauth_summary">Повторный запрос прав суперпользователя после обновления приложений</string>
 | 
			
		||||
    <string name="settings_su_tapjack_title">Защита от перехвата нажатий</string>
 | 
			
		||||
    <string name="settings_su_auth_title">Аутентификация пользователя</string>
 | 
			
		||||
    <string name="settings_su_auth_summary">Требовать аутентификацию пользователя при запросах Superuser</string>
 | 
			
		||||
    <string name="settings_su_auth_insecure">На устройстве не настроен метод аутентификации</string>
 | 
			
		||||
    <string name="settings_su_tapjack_summary">Окно запроса прав суперпользователя будет неактивно пока активированы наложения экрана</string>
 | 
			
		||||
    <string name="settings_customization">Персонализация</string>
 | 
			
		||||
    <string name="setting_add_shortcut_summary">Добавить ярлык на рабочий стол для удобного восприятия приложения после его скрытия</string>
 | 
			
		||||
    <string name="settings_doh_title">DNS поверх HTTPS</string>
 | 
			
		||||
    <string name="settings_doh_description">Активировать DoH (используйте при проблемах с подключением к сети)</string>
 | 
			
		||||
    <string name="settings_random_name_title">Случайное имя образа</string>
 | 
			
		||||
    <string name="settings_random_name_description">Генерировать случайные имена для патченных образов и tar-файлов для предотвращения обнаружения</string>
 | 
			
		||||
 | 
			
		||||
    <string name="multiuser_mode">Многопользовательский режим</string>
 | 
			
		||||
    <string name="settings_owner_only">Только администратор</string>
 | 
			
		||||
 
 | 
			
		||||
@@ -74,7 +74,6 @@
 | 
			
		||||
    <string name="su_revoke_msg">Konfirmo për të hequr të drejtat e %1$s?</string>
 | 
			
		||||
    <string name="toast">Dolli</string>
 | 
			
		||||
    <string name="none">Asnjë</string>
 | 
			
		||||
 | 
			
		||||
    <string name="superuser_toggle_notification">Njoftimet</string>
 | 
			
		||||
    <string name="superuser_toggle_revoke">Të drejtat</string>
 | 
			
		||||
    <string name="superuser_policy_none">Asnjë aplikacion nuk ka kërkuar akoma akses për super-përdoruesin.</string>
 | 
			
		||||
@@ -91,8 +90,6 @@
 | 
			
		||||
    <string name="selinux_context">Konteksti SELinux: %s</string>
 | 
			
		||||
    <string name="supp_group">Grupi suplementar: %s</string>
 | 
			
		||||
 | 
			
		||||
    <!--SafetyNet-->
 | 
			
		||||
 | 
			
		||||
    <!--MagiskHide-->
 | 
			
		||||
    <string name="show_system_app">Shfaq aplikacionet e sistemit</string>
 | 
			
		||||
    <string name="show_os_app">Shfaq aplikacionet e sistemit operativ</string>
 | 
			
		||||
@@ -109,6 +106,7 @@
 | 
			
		||||
    <string name="reboot_safe_mode">Rinis në safe mode</string>
 | 
			
		||||
    <string name="module_version_author">%1$s nga %2$s</string>
 | 
			
		||||
    <string name="module_state_remove">Hiqe</string>
 | 
			
		||||
    <string name="module_action">Veprim</string>
 | 
			
		||||
    <string name="module_state_restore">Rikëthe</string>
 | 
			
		||||
    <string name="module_action_install_external">Instaloni nga sdcard</string>
 | 
			
		||||
    <string name="update_available">Përditësimi në dispozicion</string>
 | 
			
		||||
@@ -207,9 +205,12 @@
 | 
			
		||||
    <string name="repo_install_title">Instalo %1$s %2$s(%3$d)</string>
 | 
			
		||||
    <string name="download">Shkarko</string>
 | 
			
		||||
    <string name="reboot">Rinis</string>
 | 
			
		||||
    <string name="close">Mbylle</string>
 | 
			
		||||
    <string name="release_notes">Shënimet e lëshimit</string>
 | 
			
		||||
    <string name="flashing">Duke flashuar…</string>
 | 
			
		||||
    <string name="running">Duke vepruar...</string>
 | 
			
		||||
    <string name="done">U krye!</string>
 | 
			
		||||
    <string name="done_action">Veprimi i ekzekutimit të %1$s u krye</string>
 | 
			
		||||
    <string name="failure">Dështoi!</string>
 | 
			
		||||
    <string name="hide_app_title">Fshehja e aplikacionit Magisk…</string>
 | 
			
		||||
    <string name="open_link_failed_toast">Nuk u gjet asnjë aplikacion për të hapur lidhjen</string>
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user