diff --git a/.github/workflows/gradle.yml b/.github/workflows/gradle.yml
index 0e12312..4aa03bd 100644
--- a/.github/workflows/gradle.yml
+++ b/.github/workflows/gradle.yml
@@ -1,6 +1,10 @@
name: Java CI
-on: [push]
+on:
+ push:
+ branches:
+ - main
+ - develop
jobs:
build:
@@ -8,10 +12,10 @@ jobs:
runs-on: ubuntu-latest
steps:
- - uses: actions/checkout@v1
- - name: Set up JDK 1.8
- uses: actions/setup-java@v1
- with:
- java-version: 1.8
- - name: Build with Gradle
- run: ./gradlew build
+ - uses: actions/checkout@v1
+ - name: Set up JDK 17
+ uses: actions/setup-java@v1
+ with:
+ java-version: 17
+ - name: Build with Gradle
+ run: ./gradlew build
diff --git a/.gitignore b/.gitignore
index fd45b12..46712b1 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,11 +1,11 @@
*.iml
.gradle
/local.properties
-/.idea/caches/build_file_checksums.ser
-/.idea/libraries
-/.idea/modules.xml
-/.idea/workspace.xml
+/.idea
.DS_Store
/build
/captures
.externalNativeBuild
+sign.jks
+signing.properties
+lint-baseline.xml
\ No newline at end of file
diff --git a/app/build.gradle b/app/build.gradle
index 242b561..c764937 100644
--- a/app/build.gradle
+++ b/app/build.gradle
@@ -1,10 +1,10 @@
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
-
-apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'//kapt 插件
+apply from: "${rootDir}/common.gradle"
android {
+ namespace "com.wrbug.developerhelper"
lintOptions {
checkReleaseBuilds false
abortOnError false
@@ -14,20 +14,16 @@ android {
releaseConfig
}
}
- compileOptions {
- sourceCompatibility JavaVersion.VERSION_1_8
- targetCompatibility JavaVersion.VERSION_1_8
- }
- compileSdkVersion 28
+ compileSdk 34
defaultConfig {
applicationId "com.wrbug.developerhelper"
- minSdkVersion 21
- targetSdkVersion 27
+ minSdkVersion 23
+ targetSdkVersion 34
versionCode 100040
versionName "1.0.4"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
ndk {
- abiFilters "armeabi-v7a", "arm64-v8a"
+ abiFilters "arm64-v8a","x86_64"
}
}
buildTypes {
@@ -42,36 +38,33 @@ android {
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
- dataBinding {
- enabled = true
- }
-
}
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
- implementation 'androidx.appcompat:appcompat:1.1.0'
- implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
+ implementation 'androidx.appcompat:appcompat:1.7.0'
+ implementation 'androidx.constraintlayout:constraintlayout:2.1.4'
+ implementation 'net.dongliu:apk-parser:2.6.10'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
- implementation 'com.google.code.gson:gson:2.8.5'
+ implementation 'com.google.code.gson:gson:2.10'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
implementation 'com.github.yhaolpz:FloatWindow:1.0.9'
- implementation project(':basecommon')
kapt "com.android.databinding:compiler:3.1.4"
implementation 'com.elvishew:xlog:1.6.1'
implementation 'org.jetbrains.anko:anko-commons:0.10.8'
- //dagger2
- implementation 'com.google.dagger:dagger:2.16'
+ implementation 'com.airbnb.android:lottie:6.5.2'
// https://mvnrepository.com/artifact/dom4j/dom4j
implementation 'dom4j:dom4j:1.6.1'
+ implementation 'com.drakeet.multitype:multitype:4.3.0'
implementation 'com.evrencoskun.library:tableview:0.8.8'
+ implementation 'com.yanzhenjie.recyclerview:x:1.3.2'
implementation 'gdut.bsx:share2:0.9.3'
- kapt 'com.google.dagger:dagger-compiler:2.16'
- implementation project(':xposedmodule')
implementation 'de.blox:graphview:0.5.0'
- implementation project(':basemoduleimport')
+ implementation 'com.github.bumptech.glide:glide:4.16.0'
+ implementation "com.google.android.material:material:1.12.0"
implementation project(':ipc')
+ implementation project(':commonutil')
+ implementation project(':mmkv')
}
kapt {
generateStubs = true
diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index 7ff46b2..3856b16 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -1,14 +1,22 @@
+ xmlns:tools="http://schemas.android.com/tools">
-
-
+
+
+
+
+
-
-
@@ -34,6 +41,7 @@
@@ -52,14 +60,21 @@
android:name=".service.FloatWindowService"
android:enabled="true"
android:exported="true"
+ android:foregroundServiceType="dataSync"
android:process=":floatWindow" />
-
-
+ android:name=".ui.activity.sharedpreferencesedit.SharedPreferenceEditActivity"
+ android:exported="false" />
+
+
+
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/assets/zip.dex b/app/src/main/assets/zip.dex
deleted file mode 100644
index 8cb90dd..0000000
Binary files a/app/src/main/assets/zip.dex and /dev/null differ
diff --git a/basemoduleimport/src/main/java/com/wrbug/developerhelper/basewidgetimport/BaseModule.kt b/app/src/main/java/com/wrbug/developerhelper/BaseModule.kt
similarity index 85%
rename from basemoduleimport/src/main/java/com/wrbug/developerhelper/basewidgetimport/BaseModule.kt
rename to app/src/main/java/com/wrbug/developerhelper/BaseModule.kt
index e47910e..7f045cc 100644
--- a/basemoduleimport/src/main/java/com/wrbug/developerhelper/basewidgetimport/BaseModule.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/BaseModule.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basewidgetimport
+package com.wrbug.developerhelper
import android.app.Application
import com.wrbug.developerhelper.commonutil.CommonUtils
diff --git a/app/src/main/java/com/wrbug/developerhelper/DeveloperApplication.kt b/app/src/main/java/com/wrbug/developerhelper/DeveloperApplication.kt
index ab8083e..76b99d7 100644
--- a/app/src/main/java/com/wrbug/developerhelper/DeveloperApplication.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/DeveloperApplication.kt
@@ -8,22 +8,16 @@ import com.elvishew.xlog.LogConfiguration
import com.elvishew.xlog.LogLevel
import com.elvishew.xlog.XLog
import com.elvishew.xlog.internal.DefaultsFactory
-import com.wrbug.datafinder.startup.LaunchContentProvider
-import com.wrbug.developerhelper.basecommon.BaseApp
-import com.wrbug.developerhelper.basewidgetimport.BaseModule
+import com.wrbug.developerhelper.base.BaseApp
import com.wrbug.developerhelper.commonutil.ProcessUtil
import com.wrbug.developerhelper.commonutil.print
-import com.wrbug.developerhelper.ipc.processshare.tcp.MessageHandler
-import java.io.File
-import java.io.FileOutputStream
-import com.wrbug.developerhelper.ipc.processshare.tcp.TcpManager
import com.wrbug.developerhelper.ipcserver.IpcManager
import com.wrbug.developerhelper.ui.activity.main.MainActivity
-import org.jetbrains.anko.doAsync
-
+import com.wrbug.developerhelper.util.AppStatusRegister
class DeveloperApplication : BaseApp() {
companion object {
+
private lateinit var instance: DeveloperApplication
fun getInstance(): DeveloperApplication {
return instance
@@ -31,7 +25,6 @@ class DeveloperApplication : BaseApp() {
}
override fun attachBaseContext(base: Context?) {
- LaunchContentProvider.setAutoLaunch(false)
super.attachBaseContext(base)
instance = this
}
@@ -39,13 +32,14 @@ class DeveloperApplication : BaseApp() {
override fun onCreate() {
super.onCreate()
XLog.init(
- LogConfiguration.Builder().logLevel(LogLevel.ALL).tag("developerHelper.print-->").build(),
+ LogConfiguration.Builder().logLevel(LogLevel.ALL).tag("developerHelper.print-->")
+ .build(),
DefaultsFactory.createPrinter()
)
registerIpcServer()
BaseModule.init(this)
- releaseAssetsFile()
registerLifecycle()
+ AppStatusRegister.init(this)
}
private fun registerIpcServer() {
@@ -61,51 +55,33 @@ class DeveloperApplication : BaseApp() {
private fun registerLifecycle() {
registerActivityLifecycleCallbacks(object : ActivityLifecycleCallbacks {
private var count = 0
- override fun onActivityPaused(activity: Activity?) {
+ override fun onActivityPaused(activity: Activity) {
}
- override fun onActivityResumed(activity: Activity?) {
+ override fun onActivityResumed(activity: Activity) {
}
- override fun onActivityStarted(activity: Activity?) {
+ override fun onActivityStarted(activity: Activity) {
count++
}
- override fun onActivityDestroyed(activity: Activity?) {
+ override fun onActivityDestroyed(activity: Activity) {
}
- override fun onActivitySaveInstanceState(activity: Activity?, outState: Bundle?) {
+ override fun onActivitySaveInstanceState(activity: Activity, outState: Bundle) {
}
- override fun onActivityStopped(activity: Activity?) {
+ override fun onActivityStopped(activity: Activity) {
count--
- activity?.let {
- if (count == 0 && activity is MainActivity) {
- activity.finish()
- }
+ if (count == 0 && activity is MainActivity) {
+ activity.finish()
}
-
}
- override fun onActivityCreated(activity: Activity?, savedInstanceState: Bundle?) {
+ override fun onActivityCreated(activity: Activity, savedInstanceState: Bundle?) {
}
})
}
-
-
- private fun releaseAssetsFile() {
- doAsync {
- val inputStream = BaseApp.instance.assets.open("zip.dex")
- val file = File(BaseApp.instance.cacheDir, "zip.dex")
- if (file.exists().not()) {
- file.createNewFile()
- }
- val fileOutputStream = FileOutputStream(file)
- fileOutputStream.write(inputStream.readBytes())
- fileOutputStream.flush()
- fileOutputStream.close()
- }
- }
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/base/AndroidExt.kt b/app/src/main/java/com/wrbug/developerhelper/base/AndroidExt.kt
new file mode 100644
index 0000000..f410d8a
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/base/AndroidExt.kt
@@ -0,0 +1,11 @@
+package com.wrbug.developerhelper.base
+
+import android.content.pm.PackageInfo
+import android.os.Build
+
+inline val PackageInfo.versionCodeLong: Long
+ get() = if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.P) {
+ longVersionCode
+ } else {
+ versionCode.toLong()
+ }
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/base/AppCompatActivityExt.kt b/app/src/main/java/com/wrbug/developerhelper/base/AppCompatActivityExt.kt
new file mode 100644
index 0000000..2fa0857
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/base/AppCompatActivityExt.kt
@@ -0,0 +1,130 @@
+/*
+ * Copyright (C) 2017 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.wrbug.developerhelper.base
+
+import android.Manifest
+import android.content.BroadcastReceiver
+import android.content.Context
+import android.content.Intent
+import android.content.IntentFilter
+import android.net.Uri
+import android.os.Build
+import android.os.Environment
+import android.provider.Settings
+import androidx.annotation.IdRes
+import androidx.appcompat.app.ActionBar
+import androidx.appcompat.app.AlertDialog
+import androidx.appcompat.app.AppCompatActivity
+import androidx.appcompat.widget.Toolbar
+import androidx.fragment.app.FragmentManager
+import androidx.fragment.app.FragmentTransaction
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.activityresultcallback.ActResultRequest
+import com.wrbug.developerhelper.base.activityresultcallback.ActivityResultCallback
+
+/**
+ * Various extension functions for AppCompatActivity.
+ */
+
+
+fun AppCompatActivity.setupActionBar(@IdRes toolbarId: Int, action: ActionBar.() -> Unit = {}) {
+ setupActionBar(findViewById(toolbarId), action)
+}
+
+fun AppCompatActivity.setupActionBar(toolbar: Toolbar, action: ActionBar.() -> Unit = {}) {
+ setSupportActionBar(toolbar)
+ supportActionBar?.run {
+ setDisplayHomeAsUpEnabled(false)
+ action()
+ }
+}
+
+/**
+ * Runs a FragmentTransaction, then calls commit().
+ */
+private inline fun FragmentManager.transact(action: FragmentTransaction.() -> Unit) {
+ beginTransaction().apply {
+ action()
+ }.commit()
+}
+
+fun AppCompatActivity.startActivityForResult(intent: Intent, callback: ActivityResultCallback) {
+ ActResultRequest(this).startForResult(intent, callback)
+}
+
+fun AppCompatActivity.startActivityForResultOk(
+ intent: Intent, action: Intent?.() -> Unit
+) {
+ ActResultRequest(this).startForResult(intent, object : ActivityResultCallback() {
+ override fun onActivityResultOk(data: Intent?) {
+ action(data)
+ }
+ })
+}
+
+fun Context.requestStoragePermission(callback: () -> Unit) {
+ if (this !is BaseActivity) {
+ return
+ }
+ when {
+ Build.VERSION.SDK_INT>= Build.VERSION_CODES.R -> {
+ if (!Environment.isExternalStorageManager()) {
+ AlertDialog.Builder(this).setTitle(R.string.notice)
+ .setMessage(getString(R.string.request_write_sdcard_notice))
+ .setNegativeButton(R.string.cancel, null).setPositiveButton(
+ R.string.ok
+ ) { _, _ ->
+ try {
+ val intent =
+ Intent(Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION)
+ intent.addCategory("android.intent.category.DEFAULT")
+ intent.data = Uri.parse(String.format("package:%s", packageName))
+ startActivity(intent)
+ } catch (e: Exception) {
+ val intent = Intent()
+ intent.action = Settings.ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION
+ startActivity(intent)
+ }
+ }.create().show()
+ } else {
+ callback()
+ }
+ }
+
+ else -> {
+ requestPermission(arrayOf(
+ Manifest.permission.WRITE_EXTERNAL_STORAGE,
+ Manifest.permission.READ_EXTERNAL_STORAGE
+ ), object : BaseActivity.PermissionCallback() {
+ override fun granted() {
+ callback()
+ }
+ })
+ }
+ }
+}
+
+fun Context.registerReceiverComp(
+ broadcastReceiver: BroadcastReceiver,
+ intentFilter: IntentFilter,
+ flags: Int = AppCompatActivity.RECEIVER_NOT_EXPORTED
+) {
+ if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.O) {
+ registerReceiver(broadcastReceiver, intentFilter, flags)
+ } else {
+ registerReceiver(broadcastReceiver, intentFilter)
+ }
+}
\ No newline at end of file
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseActivity.kt b/app/src/main/java/com/wrbug/developerhelper/base/BaseActivity.kt
similarity index 78%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseActivity.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/BaseActivity.kt
index 6abe91e..5bb12e6 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseActivity.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/BaseActivity.kt
@@ -1,21 +1,23 @@
-package com.wrbug.developerhelper.basecommon
+package com.wrbug.developerhelper.base
-import android.annotation.TargetApi
import android.content.DialogInterface
import android.content.pm.PackageManager
-import android.os.Build
import android.os.Bundle
import android.view.View
import androidx.annotation.StringRes
import androidx.appcompat.app.AlertDialog
import androidx.appcompat.app.AppCompatActivity
+import androidx.viewbinding.ViewBinding
import com.google.android.material.snackbar.Snackbar
+import io.reactivex.rxjava3.disposables.CompositeDisposable
import java.util.ArrayList
abstract class BaseActivity : AppCompatActivity() {
+
private lateinit var toastRootView: View
protected lateinit var context: BaseActivity
private var mPermissionCallback: PermissionCallback? = null
+ protected lateinit var disposable: CompositeDisposable
companion object {
private const val PERMISSION_REQUEST_CODE = 0xAADF1
@@ -24,11 +26,23 @@ abstract class BaseActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
context = this
super.onCreate(savedInstanceState)
+ disposable = CompositeDisposable()
}
override fun setContentView(layoutResID: Int) {
- toastRootView = layoutInflater.inflate(layoutResID, null)
- setContentView(toastRootView)
+ setContentView(layoutInflater.inflate(layoutResID, null))
+ }
+
+ override fun onDestroy() {
+ super.onDestroy()
+ disposable.dispose()
+ }
+
+ override fun setContentView(view: View?) {
+ view?.let {
+ toastRootView = view
+ }
+ super.setContentView(view)
}
fun showSnack(msg: String) {
@@ -39,13 +53,7 @@ abstract class BaseActivity : AppCompatActivity() {
Snackbar.make(toastRootView, id, Snackbar.LENGTH_SHORT).show()
}
-
- @TargetApi(Build.VERSION_CODES.M)
fun requestPermission(permissions: Array, callback: PermissionCallback) {
- if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) { - callback.granted() - return - } val list = ArrayList()
for (permission in permissions) {
val hasPermission = checkSelfPermission(permission)
@@ -61,7 +69,6 @@ abstract class BaseActivity : AppCompatActivity() {
requestPermissions(list.toTypedArray(), PERMISSION_REQUEST_CODE)
}
-
fun showDialog(
@StringRes title: Int,
@StringRes msg: Int,
@@ -80,24 +87,22 @@ abstract class BaseActivity : AppCompatActivity() {
showDialog(title, msg, positiveText, null, positiveListener, null)
}
-
fun showDialog(
@StringRes title: Int,
@StringRes msg: Int,
@StringRes positiveText: Int,
@StringRes negativeText: Int,
- onPositiveClick: DialogInterface?.(Int) -> Unit,
- onNegativeClick: DialogInterface?.(Int) -> Unit? = {
+ onPositiveClick: DialogInterface.(Int) -> Unit,
+ onNegativeClick: DialogInterface.(Int) -> Unit? = {
}
) {
- showDialog(
- title,
+ showDialog(title,
msg,
positiveText,
negativeText,
- DialogInterface.OnClickListener { dialog, which -> dialog.onPositiveClick(which) },
- DialogInterface.OnClickListener { dialog, which -> dialog.onNegativeClick(which) })
+ { dialog, which -> dialog.onPositiveClick(which) },
+ { dialog, which -> dialog.onNegativeClick(which) })
}
fun showDialog(
@@ -110,13 +115,12 @@ abstract class BaseActivity : AppCompatActivity() {
}
) {
- showDialog(
- title,
+ showDialog(title,
msg,
positiveText,
negativeText,
- DialogInterface.OnClickListener { dialog, which -> dialog.onPositiveClick(which) },
- DialogInterface.OnClickListener { dialog, which -> dialog.onNegativeClick(which) })
+ { dialog, which -> dialog.onPositiveClick(which) },
+ { dialog, which -> dialog.onNegativeClick(which) })
}
private fun showDialog(
@@ -163,8 +167,7 @@ abstract class BaseActivity : AppCompatActivity() {
positiveListener: DialogInterface.OnClickListener,
negativeListener: DialogInterface.OnClickListener?
) {
- val builder = AlertDialog.Builder(this).setMessage(msg)
- .setTitle(title)
+ val builder = AlertDialog.Builder(this).setMessage(msg).setTitle(title)
.setPositiveButton(positiveText, positiveListener)
if (negativeText.isNullOrEmpty().not()) {
builder.setNegativeButton(negativeText, negativeListener)
@@ -172,10 +175,12 @@ abstract class BaseActivity : AppCompatActivity() {
builder.show()
}
+ fun hasPermission(permissions: String): Boolean {
+ return checkSelfPermission(permissions) == PackageManager.PERMISSION_GRANTED
+ }
+
override fun onRequestPermissionsResult(
- requestCode: Int,
- permissions: Array,
- grantResults: IntArray
+ requestCode: Int, permissions: Array, grantResults: IntArray
) {
if (requestCode == PERMISSION_REQUEST_CODE) {
val list = ArrayList()
@@ -195,10 +200,16 @@ abstract class BaseActivity : AppCompatActivity() {
}
abstract class PermissionCallback {
+
abstract fun granted()
open fun denied(permissions: List) {
}
}
+
+ protected fun T.inject(): T {
+ setContentView(this.root)
+ return this
+ }
}
\ No newline at end of file
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseApp.kt b/app/src/main/java/com/wrbug/developerhelper/base/BaseApp.kt
similarity index 83%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseApp.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/BaseApp.kt
index 7c1c2e6..86221e7 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseApp.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/BaseApp.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basecommon
+package com.wrbug.developerhelper.base
import android.app.Application
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseFragment.kt b/app/src/main/java/com/wrbug/developerhelper/base/BaseFragment.kt
similarity index 64%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseFragment.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/BaseFragment.kt
index 894989c..e7d2360 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseFragment.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/BaseFragment.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basecommon
+package com.wrbug.developerhelper.base
import androidx.fragment.app.Fragment
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseVMActivity.kt b/app/src/main/java/com/wrbug/developerhelper/base/BaseVMActivity.kt
similarity index 92%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseVMActivity.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/BaseVMActivity.kt
index fb50ab6..51bff35 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseVMActivity.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/BaseVMActivity.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basecommon
+package com.wrbug.developerhelper.base
import android.os.Bundle
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseViewModel.kt b/app/src/main/java/com/wrbug/developerhelper/base/BaseViewModel.kt
similarity index 90%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseViewModel.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/BaseViewModel.kt
index 3928d9a..a0e716a 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/BaseViewModel.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/BaseViewModel.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basecommon
+package com.wrbug.developerhelper.base
import androidx.lifecycle.LifecycleObserver
import androidx.lifecycle.MutableLiveData
diff --git a/app/src/main/java/com/wrbug/developerhelper/base/ExtraKey.kt b/app/src/main/java/com/wrbug/developerhelper/base/ExtraKey.kt
new file mode 100644
index 0000000..bec3ba2
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/base/ExtraKey.kt
@@ -0,0 +1,13 @@
+package com.wrbug.developerhelper.base
+
+object ExtraKey {
+
+ const val PACKAGE_NAME = "packageName"
+ const val DATA = "data"
+ const val SELECTED = "selected"
+ const val KEY_1 = "key1"
+ const val KEY_2 = "key2"
+ const val KEY_3 = "key3"
+ const val KEY_4 = "key4"
+ const val KEY_5 = "key5"
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/base/ThreadExt.kt b/app/src/main/java/com/wrbug/developerhelper/base/ThreadExt.kt
new file mode 100644
index 0000000..6278628
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/base/ThreadExt.kt
@@ -0,0 +1,21 @@
+package com.wrbug.developerhelper.base
+
+import android.os.Handler
+import android.os.Looper
+import kotlin.concurrent.thread
+
+private val handler by lazy {
+ Handler(Looper.getMainLooper())
+}
+
+fun doAsync(callback: () -> Unit) {
+ thread {
+ callback()
+ }
+}
+
+fun uiThread(callback: () -> Unit) {
+ handler.post {
+ callback()
+ }
+}
\ No newline at end of file
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/ToastExts.kt b/app/src/main/java/com/wrbug/developerhelper/base/ToastExts.kt
similarity index 76%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/ToastExts.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/ToastExts.kt
index e8f1eee..872bcf7 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/ToastExts.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/ToastExts.kt
@@ -1,14 +1,11 @@
-package com.wrbug.developerhelper.basecommon
+package com.wrbug.developerhelper.base
import android.app.Activity
-import android.app.Application
import android.content.Context
-import android.os.Handler
-import android.os.Looper
import android.view.View
import android.widget.Toast
import androidx.annotation.StringRes
-import com.wrbug.developerhelper.commonwidget.flexibletoast.FlexibleToast
+import com.wrbug.developerhelper.ui.widget.flexibletoast.FlexibleToast
fun View.showToast(msg: CharSequence?) {
Toast.makeText(this.context, msg, Toast.LENGTH_SHORT).show()
@@ -21,7 +18,6 @@ fun View.showToast(id: Int) {
fun Context.showToast(msg: CharSequence?) {
FlexibleToast.toastShow(this, msg.toString())
-
}
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/ActResultRequest.kt b/app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/ActResultRequest.kt
similarity index 84%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/ActResultRequest.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/ActResultRequest.kt
index 522648a..8301892 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/ActResultRequest.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/ActResultRequest.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basecommon.activityresultcallback
+package com.wrbug.developerhelper.base.activityresultcallback
import android.content.Intent
import android.os.Bundle
@@ -7,7 +7,7 @@ import androidx.fragment.app.Fragment
import androidx.fragment.app.FragmentManager
class ActResultRequest {
- private lateinit var mFragment: OnActResultEventDispatcherFragment
+ private val mFragment: OnActResultEventDispatcherFragment
constructor(activity: AppCompatActivity) {
mFragment = getEventDispatchFragment(activity)
@@ -29,10 +29,12 @@ class ActResultRequest {
}
private fun addFragment(fragmentManager: FragmentManager): OnActResultEventDispatcherFragment {
- var fragment: OnActResultEventDispatcherFragment? = this.findEventDispatchFragment(fragmentManager)
+ var fragment: OnActResultEventDispatcherFragment? =
+ this.findEventDispatchFragment(fragmentManager)
if (fragment == null) {
fragment = OnActResultEventDispatcherFragment()
- fragmentManager.beginTransaction().add(fragment, "on_act_result_event_dispatcher").commitAllowingStateLoss()
+ fragmentManager.beginTransaction().add(fragment, "on_act_result_event_dispatcher")
+ .commitAllowingStateLoss()
fragmentManager.executePendingTransactions()
}
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/ActivityResultCallback.kt b/app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/ActivityResultCallback.kt
similarity index 54%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/ActivityResultCallback.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/ActivityResultCallback.kt
index 8306c3d..42ef08b 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/ActivityResultCallback.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/ActivityResultCallback.kt
@@ -1,21 +1,22 @@
-package com.wrbug.developerhelper.basecommon.activityresultcallback
+package com.wrbug.developerhelper.base.activityresultcallback
import android.app.Activity
import android.content.Intent
-abstract class ActivityResultCallback() {
- fun dispatchActivityResult(resultCode: Int, data: Intent) {
+abstract class ActivityResultCallback {
+
+ fun dispatchActivityResult(resultCode: Int, data: Intent?) {
onActivityResult(resultCode, data)
if (resultCode == Activity.RESULT_OK) {
onActivityResultOk(data)
}
}
- protected open fun onActivityResult(resultCode: Int, data: Intent) {
+ protected open fun onActivityResult(resultCode: Int, data: Intent?) {
}
- protected open fun onActivityResultOk(data: Intent) {
+ protected open fun onActivityResultOk(data: Intent?) {
}
}
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/OnActResultEventDispatcherFragment.kt b/app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/OnActResultEventDispatcherFragment.kt
similarity index 90%
rename from basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/OnActResultEventDispatcherFragment.kt
rename to app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/OnActResultEventDispatcherFragment.kt
index 70311d2..b44f5ba 100644
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/activityresultcallback/OnActResultEventDispatcherFragment.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/base/activityresultcallback/OnActResultEventDispatcherFragment.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.basecommon.activityresultcallback
+package com.wrbug.developerhelper.base.activityresultcallback
import android.content.Intent
import android.os.Bundle
@@ -15,12 +15,10 @@ class OnActResultEventDispatcherFragment : Fragment() {
this.startActivityForResult(intent, requestCode, options)
}
-
- override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent) {
+ override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
val callback = this.mCallbacks.get(requestCode) as ActivityResultCallback
this.mCallbacks.remove(requestCode)
callback.dispatchActivityResult(resultCode, data)
-
}
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/base/entry/HierarchyNode.kt b/app/src/main/java/com/wrbug/developerhelper/base/entry/HierarchyNode.kt
new file mode 100644
index 0000000..0356af7
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/base/entry/HierarchyNode.kt
@@ -0,0 +1,33 @@
+package com.wrbug.developerhelper.base.entry
+
+import android.graphics.Rect
+import android.os.Parcelable
+import kotlinx.parcelize.Parcelize
+import java.io.Serializable
+
+@Suppress("NULLABILITY_MISMATCH_BASED_ON_JAVA_ANNOTATIONS")
+@Parcelize
+data class HierarchyNode(
+ var id: Long = -1L,
+ var screenBounds: Rect? = null,
+ var parentBounds: Rect? = null,
+ var checkable: Boolean = false,
+ var checked: Boolean = false,
+ var widget: String = "",
+ var clickable: Boolean = false,
+ var contentDesc: String = "",
+ var enabled: Boolean = false,
+ var focusable: Boolean = false,
+ var focused: Boolean = false,
+ var index: String = "",
+ var longClickable: Boolean = false,
+ var packageName: String = "",
+ var password: Boolean = false,
+ var scrollable: Boolean = false,
+ var selected: Boolean = false,
+ var text: String = "",
+ var resourceId: String = "",
+ var idHex: String? = null,
+ var parentId: Long = -1L,
+ var childId: ArrayList = arrayListOf()
+): Parcelable, Serializable
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ipcserver/FileProcessDataImpl.kt b/app/src/main/java/com/wrbug/developerhelper/ipcserver/FileProcessDataImpl.kt
index 8246dac..8371bf1 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ipcserver/FileProcessDataImpl.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ipcserver/FileProcessDataImpl.kt
@@ -1,6 +1,6 @@
package com.wrbug.developerhelper.ipcserver
-import com.wrbug.developerhelper.basecommon.BaseApp
+import com.wrbug.developerhelper.base.BaseApp
import com.wrbug.developerhelper.commonutil.Base64
import com.wrbug.developerhelper.ipc.processshare.TcpUrl
import com.wrbug.developerhelper.ipcserver.annotation.Controller
diff --git a/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppData.kt b/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppData.kt
new file mode 100644
index 0000000..3c1ac5b
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppData.kt
@@ -0,0 +1,13 @@
+package com.wrbug.developerhelper.model.entity
+
+import kotlinx.parcelize.Parcelize
+import java.io.File
+import java.io.Serializable
+
+data class BackupAppData(
+ val appName: String,
+ val packageName: String,
+ val rootDir: File,
+ val backupMap: HashMap,
+ val iconPath: File?
+) : Serializable
diff --git a/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppInfo.kt b/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppInfo.kt
new file mode 100644
index 0000000..91db0d0
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppInfo.kt
@@ -0,0 +1,12 @@
+package com.wrbug.developerhelper.model.entity
+
+import com.google.gson.annotations.SerializedName
+
+data class BackupAppInfo(
+ @SerializedName("appName")
+ var appName: String = "",
+ @SerializedName("packageName")
+ var packageName: String = "",
+ @SerializedName("backupMap")
+ val backupMap: HashMap = hashMapOf()
+)
diff --git a/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppItemInfo.kt b/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppItemInfo.kt
new file mode 100644
index 0000000..613b9f3
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/model/entity/BackupAppItemInfo.kt
@@ -0,0 +1,38 @@
+package com.wrbug.developerhelper.model.entity
+
+import android.os.Parcelable
+import com.google.gson.annotations.SerializedName
+import kotlinx.parcelize.Parcelize
+import java.io.Serializable
+
+@Parcelize
+data class BackupAppItemInfo(
+ @SerializedName("backupFile")
+ var backupFile: String = "",
+ @SerializedName("backupApk")
+ var backupApk: Boolean = false,
+ @SerializedName("backupData")
+ var backupData: Boolean = false,
+ @SerializedName("backupAndroidData")
+ var backupAndroidData: Boolean = false,
+ @SerializedName("apkFile")
+ var apkFile: String = "",
+ @SerializedName("versionName")
+ var versionName: String = "",
+ @SerializedName("versionCode")
+ var versionCode: Long = 0,
+ @SerializedName("packageName")
+ var packageName: String = "",
+ @SerializedName("time")
+ var time: Long = 0,
+ @SerializedName("dataFile")
+ var dataFile: String = "",
+ @SerializedName("androidDataFile")
+ var androidDataFile: String = "",
+ @SerializedName("memo")
+ var memo: String = ""
+) : Parcelable, Serializable {
+ companion object {
+ val EMPTY = BackupAppItemInfo()
+ }
+}
diff --git a/app/src/main/java/com/wrbug/developerhelper/model/entity/SharedPreferenceItemInfo.kt b/app/src/main/java/com/wrbug/developerhelper/model/entity/SharedPreferenceItemInfo.kt
index 6bf2927..49859ce 100644
--- a/app/src/main/java/com/wrbug/developerhelper/model/entity/SharedPreferenceItemInfo.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/model/entity/SharedPreferenceItemInfo.kt
@@ -10,7 +10,7 @@ class SharedPreferenceItemInfo {
fun isValueValid(): Boolean {
- when (type.toLowerCase()) {
+ when (type.lowercase()) {
"string" -> {
return JsonHelper.fromJson(value) == null || JsonHelper.fromJson(newValue) != null
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/service/AccessibilityManager.kt b/app/src/main/java/com/wrbug/developerhelper/service/AccessibilityManager.kt
index 418707e..3c9c6ce 100644
--- a/app/src/main/java/com/wrbug/developerhelper/service/AccessibilityManager.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/service/AccessibilityManager.kt
@@ -4,32 +4,23 @@ import android.content.Context
import android.content.Intent
import android.provider.Settings
import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.showToast
-import com.wrbug.developerhelper.commonutil.shell.Callback
+import com.wrbug.developerhelper.base.showToast
import com.wrbug.developerhelper.commonutil.shell.ShellManager
+import com.wrbug.developerhelper.util.getString
+import io.reactivex.rxjava3.core.Single
object AccessibilityManager {
- fun startService(context: Context?, callback: Callback? = null) {
- context?.run {
- ShellManager.openAccessibilityService(object : Callback {
- override fun onSuccess(data: Boolean) {
- if (!data) {
- showToast(getString(R.string.please_open_accessbility_service))
- startAccessibilitySetting(context)
- }
- callback?.onSuccess(data)
- }
-
- override fun onFailed(msg: String) {
- callback?.onFailed(msg)
- startAccessibilitySetting(context)
- }
-
- })
+ fun startService(context: Context?): Single {
+ return ShellManager.openAccessibilityService().doOnSuccess {
+ if (!it) {
+ context?.showToast(getString(R.string.please_open_accessbility_service))
+ }
}
}
- fun startAccessibilitySetting(context: Context) {
+
+ fun startAccessibilitySetting(context: Context?) {
+ context ?: return
val intent = Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS)
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
context.startActivity(intent)
diff --git a/app/src/main/java/com/wrbug/developerhelper/service/DeveloperHelperAccessibilityService.kt b/app/src/main/java/com/wrbug/developerhelper/service/DeveloperHelperAccessibilityService.kt
index be8a751..7c8e99f 100644
--- a/app/src/main/java/com/wrbug/developerhelper/service/DeveloperHelperAccessibilityService.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/service/DeveloperHelperAccessibilityService.kt
@@ -1,25 +1,25 @@
package com.wrbug.developerhelper.service
import android.accessibilityservice.AccessibilityService
+import android.annotation.SuppressLint
import android.content.BroadcastReceiver
+import android.content.ComponentName
import android.content.Context
import android.content.Intent
import android.content.IntentFilter
import android.graphics.Rect
+import android.os.Build
import android.provider.Settings
import android.text.TextUtils
import android.view.accessibility.AccessibilityEvent
import android.view.accessibility.AccessibilityNodeInfo
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseApp
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
-import com.wrbug.developerhelper.basecommon.showToast
+import com.wrbug.developerhelper.base.BaseApp
+import com.wrbug.developerhelper.base.entry.HierarchyNode
+import com.wrbug.developerhelper.base.registerReceiverComp
import com.wrbug.developerhelper.commonutil.AppInfoManager
import com.wrbug.developerhelper.commonutil.UiUtils
import com.wrbug.developerhelper.commonutil.entity.ApkInfo
import com.wrbug.developerhelper.commonutil.entity.TopActivityInfo
-import com.wrbug.developerhelper.commonutil.shell.Callback
-import com.wrbug.developerhelper.commonutil.shell.ShellManager
import com.wrbug.developerhelper.constant.ReceiverConstant
import com.wrbug.developerhelper.ui.activity.hierachy.HierarchyActivity
@@ -29,6 +29,7 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
private var nodeId = 0L
private var currentAppInfo: ApkInfo? = null
private var topActivity: TopActivityInfo? = null
+ private val activityMap = hashMapOf()
companion object {
internal var serviceRunning = false
@@ -41,7 +42,7 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
BaseApp.instance.applicationContext.contentResolver,
android.provider.Settings.Secure.ACCESSIBILITY_ENABLED
)
- } catch (e: Settings.SettingNotFoundException) {
+ } catch (_: Settings.SettingNotFoundException) {
}
val mStringColonSplitter = TextUtils.SimpleStringSplitter(':')
@@ -64,7 +65,6 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
return false
}
- //todo java.lang.RuntimeException:android.os.TransactionTooLargeException
val nodeMap: HashMap = hashMapOf()
}
@@ -72,7 +72,22 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
}
override fun onAccessibilityEvent(event: AccessibilityEvent?) {
-
+ if (event?.eventType != AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED
+ || event.className.isNullOrEmpty()
+ || event.className?.startsWith("android.") == true
+ || event.packageName.isNullOrEmpty()
+ ) {
+ return
+ }
+ runCatching {
+ val info = packageManager.getActivityInfo(
+ ComponentName(
+ event.packageName.toString(),
+ event.className.toString()
+ ), 0
+ )
+ activityMap[info.packageName] = info.name
+ }
}
fun readNode(): ArrayList {
@@ -88,6 +103,7 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
return hierarchyNodes
}
+
private fun getDecorViewNode(node: AccessibilityNodeInfo): AccessibilityNodeInfo? {
for (index in 0 until node.childCount) {
val child = node.getChild(index)
@@ -102,18 +118,18 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
return null
}
+ @SuppressLint("UnspecifiedRegisterReceiverFlag")
override fun onCreate() {
super.onCreate()
val filter = IntentFilter()
filter.addAction(ReceiverConstant.ACTION_HIERARCHY_VIEW)
- registerReceiver(receiver, filter)
+ registerReceiverComp(receiver, filter)
sendStatusBroadcast(true)
serviceRunning = true
nodeMap.clear()
}
-
override fun onDestroy() {
super.onDestroy()
nodeMap.clear()
@@ -205,17 +221,11 @@ class DeveloperHelperAccessibilityService : AccessibilityService() {
inner class DeveloperHelperAccessibilityReceiver : BroadcastReceiver() {
- override fun onReceive(context: Context?, data: Intent?) {
- showToast(getString(R.string.getting_app_info))
- ShellManager.getTopActivity(object : Callback {
-
- override fun onSuccess(data: TopActivityInfo?) {
- topActivity = data
- val nodesInfo = readNode()
- HierarchyActivity.start(context, currentAppInfo, nodesInfo, topActivity)
- }
- })
-
+ override fun onReceive(context: Context, data: Intent?) {
+ val nodesInfo = readNode()
+ currentAppInfo?.topActivity =
+ activityMap[currentAppInfo?.packageInfo?.packageName].orEmpty()
+ HierarchyActivity.start(context, currentAppInfo, nodesInfo)
}
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/service/FloatWindowService.kt b/app/src/main/java/com/wrbug/developerhelper/service/FloatWindowService.kt
index da75c41..5404f33 100644
--- a/app/src/main/java/com/wrbug/developerhelper/service/FloatWindowService.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/service/FloatWindowService.kt
@@ -1,27 +1,41 @@
package com.wrbug.developerhelper.service
+import android.annotation.SuppressLint
import android.app.*
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import android.content.IntentFilter
+import android.content.pm.ServiceInfo
import android.os.Build
import android.os.IBinder
+import android.util.Log
import android.view.LayoutInflater
import android.widget.RemoteViews
import androidx.core.app.NotificationCompat
+import androidx.core.app.ServiceCompat
import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.registerReceiverComp
+import com.wrbug.developerhelper.commonutil.UiUtils
+import com.wrbug.developerhelper.commonutil.addTo
+import com.wrbug.developerhelper.commonutil.dpInt
import com.wrbug.developerhelper.constant.ReceiverConstant
-import com.wrbug.developerhelper.commonutil.shell.Callback
import com.wrbug.developerhelper.commonutil.shell.ShellManager
+import com.wrbug.developerhelper.util.setOnDoubleCheckClickListener
import com.wrbug.developerhelper.ui.activity.main.MainActivity
+import com.wrbug.developerhelper.util.DeviceUtils
+import com.wrbug.developerhelper.util.isPortrait
import com.yhao.floatwindow.FloatWindow
import com.yhao.floatwindow.Screen
-
+import com.yhao.floatwindow.ViewStateListener
+import com.yhao.floatwindow.ViewStateListenerAdapter
+import io.reactivex.rxjava3.disposables.CompositeDisposable
+import org.jetbrains.anko.toast
class FloatWindowService : Service() {
companion object {
+
const val FLOAT_BUTTON = "floatButton"
private const val CHANNEL_ID = "DEMON"
fun start(context: Context) {
@@ -43,51 +57,76 @@ class FloatWindowService : Service() {
RemoteViews(packageName, R.layout.view_float_custom).apply {
setOnClickPendingIntent(
R.id.adbWifiContainer, PendingIntent.getBroadcast(
- applicationContext, 0,
- Intent(ReceiverConstant.ACTION_ADB_WIFI_CLICKED), PendingIntent.FLAG_UPDATE_CURRENT
+ applicationContext,
+ 0,
+ Intent(ReceiverConstant.ACTION_ADB_WIFI_CLICKED),
+ PendingIntent.FLAG_UPDATE_CURRENT
)
)
}
}
+ private lateinit var disposable: CompositeDisposable
private lateinit var notification: Notification
private val receiver = Receiver()
override fun onCreate() {
super.onCreate()
+ disposable = CompositeDisposable()
initReceiver()
LayoutInflater.from(this).inflate(R.layout.layout_float_window_button, null)?.let {
- it.setOnClickListener {
+ it.setOnDoubleCheckClickListener {
if (!DeveloperHelperAccessibilityService.isAccessibilitySettingsOn()) {
- AccessibilityManager.startService(this, object : Callback {
- override fun onSuccess(data: Boolean) {
- if (data) {
- it.postDelayed({
- sendBroadcast(Intent(ReceiverConstant.ACTION_HIERARCHY_VIEW))
- }, 500)
- }
+ AccessibilityManager.startService(this).subscribe({ data ->
+ if (data) {
+ it.postDelayed({
+ sendBroadcast(
+ Intent(ReceiverConstant.ACTION_HIERARCHY_VIEW).setPackage(
+ packageName
+ )
+ )
+ }, 500)
}
+ }, {
- })
- return@setOnClickListener
+ }).addTo(disposable)
+ return@setOnDoubleCheckClickListener
}
- sendBroadcast(Intent(ReceiverConstant.ACTION_HIERARCHY_VIEW))
+ sendBroadcast(Intent(ReceiverConstant.ACTION_HIERARCHY_VIEW).setPackage(packageName))
}
- FloatWindow
- .with(applicationContext)
- .setView(it)
- .setWidth(Screen.width, 0.1f) //设置控件宽高
- .setHeight(Screen.width, 0.1f)
- .setY(Screen.height, 0.3f)
- .setTag(FLOAT_BUTTON)
- .setDesktopShow(true) //桌面显示
- .build()
-
+ val screen = if (isPortrait()) {
+ UiUtils.getDeviceWidth() * 0.1
+ } else {
+ UiUtils.getDeviceHeight() * 0.1
+ }.toInt()
+ FloatWindow.with(applicationContext).setView(it).setWidth(screen)
+ .setHeight(screen).setY(Screen.height, 0.3f).setTag(FLOAT_BUTTON)
+ .setDesktopShow(true).setViewStateListener(object : ViewStateListenerAdapter() {
+ override fun onPositionUpdate(x: Int, y: Int) {
+ val minY = UiUtils.getStatusHeight() + 10.dpInt(applicationContext)
+ val maxY = UiUtils.getDeviceHeight() - 60.dpInt(applicationContext)
+ if (y < minY) { + FloatWindow.get(FLOAT_BUTTON).updateY(y) + return + } + if (y> maxY) {
+ FloatWindow.get(FLOAT_BUTTON).updateY(maxY)
+ }
+ }
+ }).build()
}
}
-
private fun updateNotification() {
- startForeground(0x10000, notification)
+ val type = if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.R) {
+ ServiceInfo.FOREGROUND_SERVICE_TYPE_DATA_SYNC
+ } else {
+ 0
+ }
+ runCatching {
+ ServiceCompat.startForeground(this, 0x10000, notification, type)
+ }.getOrElse {
+ it.printStackTrace()
+ }
}
private fun updateNotificationContent(text: String) {
@@ -99,28 +138,35 @@ class FloatWindowService : Service() {
floatCustomView.setImageViewResource(R.id.adbWifiIv, id)
updateNotification()
}
+
+ @SuppressLint("UnspecifiedImmutableFlag")
private fun initNotification() {
- val notificationManager = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
+ val notificationManager =
+ getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.O) {
- val channel =
- NotificationChannel(CHANNEL_ID, getString(R.string.demon_process), NotificationManager.IMPORTANCE_LOW)
+ val channel = NotificationChannel(
+ CHANNEL_ID, getString(R.string.demon_process), NotificationManager.IMPORTANCE_LOW
+ )
channel.enableLights(true)
channel.setShowBadge(true)
notificationManager.createNotificationChannel(channel)
}
val intent = Intent(this, MainActivity::class.java)
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
- val pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT)
- val builder = NotificationCompat.Builder(this, CHANNEL_ID)
- .setAutoCancel(false)
- .setContentIntent(pendingIntent)
- .setContentTitle(getString(R.string.app_name))
+ val pendingIntent = if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.M) {
+ PendingIntent.getActivity(
+ this, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT or PendingIntent.FLAG_IMMUTABLE
+ )
+ } else {
+ PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT)
+ }
+ val builder = NotificationCompat.Builder(this, CHANNEL_ID).setAutoCancel(false)
+ .setContentIntent(pendingIntent).setContentTitle(getString(R.string.app_name))
.setContentText(getString(R.string.demon_process_content))
- .setSmallIcon(R.drawable.ic_launcher_notify)
- .setVibrate(null)
+ .setSmallIcon(R.drawable.ic_launcher_notify).setVibrate(null)
notification = builder.build()
- notification.flags = Notification.FLAG_ONGOING_EVENT or Notification.FLAG_NO_CLEAR or
- Notification.FLAG_FOREGROUND_SERVICE
+ notification.flags =
+ Notification.FLAG_ONGOING_EVENT or Notification.FLAG_NO_CLEAR or Notification.FLAG_FOREGROUND_SERVICE
updateNotification()
}
@@ -135,7 +181,7 @@ class FloatWindowService : Service() {
private fun initReceiver() {
val filter = IntentFilter(ReceiverConstant.ACTION_SET_FLOAT_BUTTON_VISIBLE)
filter.addAction(ReceiverConstant.ACTION_ADB_WIFI_CLICKED)
- registerReceiver(receiver, filter)
+ registerReceiverComp(receiver, filter)
}
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
@@ -146,6 +192,7 @@ class FloatWindowService : Service() {
override fun onDestroy() {
try {
+ disposable.dispose()
FloatWindow.destroy(FLOAT_BUTTON)
unregisterReceiver(receiver)
stopForeground(true)
@@ -158,8 +205,8 @@ class FloatWindowService : Service() {
override fun onBind(intent: Intent): IBinder = null!!
-
private inner class Receiver : BroadcastReceiver() {
+
override fun onReceive(context: Context?, intent: Intent?) {
when (intent?.action) {
ReceiverConstant.ACTION_SET_FLOAT_BUTTON_VISIBLE -> {
@@ -170,6 +217,7 @@ class FloatWindowService : Service() {
hideFloatButton()
}
}
+
ReceiverConstant.ACTION_ADB_WIFI_CLICKED -> {
updateNotificationContent("正在开启adb wifi")
val success = ShellManager.openAdbWifi()
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/AppBackupDetailActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/AppBackupDetailActivity.kt
new file mode 100644
index 0000000..0dd9509
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/AppBackupDetailActivity.kt
@@ -0,0 +1,125 @@
+package com.wrbug.developerhelper.ui.activity.appbackup
+
+import android.app.ActionBar.LayoutParams
+import android.app.backup.BackupManager
+import android.content.Context
+import android.content.Intent
+import android.os.Bundle
+import android.view.View
+import androidx.core.content.IntentCompat
+import androidx.recyclerview.widget.LinearLayoutManager
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.BaseActivity
+import com.wrbug.developerhelper.base.ExtraKey
+import com.wrbug.developerhelper.commonutil.dpInt
+import com.wrbug.developerhelper.databinding.ActivityAppBackupDetailBinding
+import com.wrbug.developerhelper.model.entity.BackupAppData
+import com.wrbug.developerhelper.model.entity.BackupAppItemInfo
+import com.wrbug.developerhelper.ui.adapter.ExMultiTypeAdapter
+import com.wrbug.developerhelper.ui.decoration.SpaceItemDecoration
+import com.wrbug.developerhelper.util.BackupUtils
+import com.wrbug.developerhelper.util.loadImage
+import com.yanzhenjie.recyclerview.SwipeMenuItem
+
+class AppBackupDetailActivity : BaseActivity() {
+
+ companion object {
+ fun start(context: Context, info: BackupAppData) {
+ context.startActivity(Intent(context, AppBackupDetailActivity::class.java).apply {
+ putExtra(ExtraKey.DATA, info)
+ })
+ }
+ }
+
+
+ private val adapter by ExMultiTypeAdapter.get()
+
+ private val info by lazy {
+ IntentCompat.getSerializableExtra(intent, ExtraKey.DATA, BackupAppData::class.java)
+ }
+ private val binding by lazy {
+ ActivityAppBackupDetailBinding.inflate(layoutInflater)
+ }
+ private var listCache = arrayListOf()
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ setContentView(binding.root)
+ initView()
+ loadData(info?.backupMap)
+ }
+
+ private fun loadData(map: Map?) {
+ val list = map?.values?.sortedByDescending { it.time } ?: emptyList()
+ listCache.clear()
+ listCache.addAll(list)
+ setupList()
+ }
+
+ private fun setupList() {
+ if (listCache.isEmpty()) {
+ adapter.showEmpty()
+ } else {
+ adapter.loadData(listCache)
+ }
+ }
+
+ private fun initView() {
+ binding.appBar.setSubTitle(info?.appName)
+ binding.rvAppBackupList.layoutManager = LinearLayoutManager(this)
+ binding.rvAppBackupList.addItemDecoration(SpaceItemDecoration.standard)
+ adapter.register(BackupDetailDelegate(info?.appName.orEmpty()))
+ binding.rvAppBackupList.setSwipeMenuCreator { leftMenu, rightMenu, position ->
+ rightMenu.addMenuItem(SwipeMenuItem(this).apply {
+ text = getString(R.string.item_swipe_menu_meme)
+ width = 56.dpInt()
+ height = LayoutParams.MATCH_PARENT
+ setTextColorResource(R.color.material_text_color_white_text)
+ setBackgroundColorResource(R.color.colorAccent)
+ })
+ rightMenu.addMenuItem(SwipeMenuItem(this).apply {
+ text = getString(R.string.item_swipe_menu_delete)
+ width = 56.dpInt()
+ height = LayoutParams.MATCH_PARENT
+ setTextColorResource(R.color.material_text_color_white_text)
+ setBackgroundColorResource(R.color.material_color_red_600)
+ })
+ }
+ binding.rvAppBackupList.setOnItemMenuClickListener { menuBridge, adapterPosition ->
+ menuBridge.closeMenu()
+ when (menuBridge.position) {
+ 0 -> {
+
+ }
+
+ 1 -> {
+ showDialog(
+ R.string.notice,
+ R.string.delete_backup_notice,
+ R.string.ok,
+ R.string.cancel,
+ {
+ deleteBackup(adapter.items[adapterPosition] as? BackupAppItemInfo)
+ dismiss()
+ },
+ {
+ dismiss()
+ })
+ }
+
+ else -> {}
+ }
+ }
+ binding.rvAppBackupList.adapter = adapter
+ }
+
+ private fun deleteBackup(backupAppItemInfo: BackupAppItemInfo?) {
+ backupAppItemInfo ?: return
+ BackupUtils.deleteBackupItem(backupAppItemInfo.packageName, backupAppItemInfo.backupFile)
+ .subscribe({
+ loadData(it.backupMap)
+ }, {
+ showSnack(getString(R.string.delete_backup_failed_retry))
+ })
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupAppActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupAppActivity.kt
new file mode 100644
index 0000000..535cfd6
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupAppActivity.kt
@@ -0,0 +1,55 @@
+package com.wrbug.developerhelper.ui.activity.appbackup
+
+import android.os.Bundle
+import androidx.recyclerview.widget.LinearLayoutManager
+import com.drakeet.multitype.MultiTypeAdapter
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.BaseActivity
+import com.wrbug.developerhelper.base.setupActionBar
+import com.wrbug.developerhelper.commonutil.addTo
+import com.wrbug.developerhelper.commonutil.dpInt
+import com.wrbug.developerhelper.commonutil.runOnIO
+import com.wrbug.developerhelper.databinding.ActivityBackupAppBinding
+import com.wrbug.developerhelper.ui.adapter.ExMultiTypeAdapter
+import com.wrbug.developerhelper.ui.decoration.SpaceItemDecoration
+import com.wrbug.developerhelper.util.BackupUtils
+
+class BackupAppActivity : BaseActivity() {
+ private val binding by lazy {
+ ActivityBackupAppBinding.inflate(layoutInflater)
+ }
+ private val adapter by ExMultiTypeAdapter.get()
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ setContentView(binding.root)
+ initView()
+ }
+
+ private fun initView() {
+ binding.rvAppList.layoutManager = LinearLayoutManager(this)
+ binding.rvAppList.adapter = adapter
+ binding.rvAppList.addItemDecoration(SpaceItemDecoration.standard)
+ adapter.register(BackupInfoItemDelegate {
+ AppBackupDetailActivity.start(this, it)
+ })
+ }
+
+ override fun onStart() {
+ super.onStart()
+ loadData()
+ }
+
+ private fun loadData() {
+ adapter.showLoading()
+ BackupUtils.getAllBackupInfo().runOnIO().subscribe({
+ if (it.isEmpty()) {
+ adapter.showEmpty()
+ return@subscribe
+ }
+ adapter.loadData(it)
+ }, {
+ adapter.showEmpty()
+ }).addTo(disposable)
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupDetailDelegate.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupDetailDelegate.kt
new file mode 100644
index 0000000..ff9a1c9
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupDetailDelegate.kt
@@ -0,0 +1,45 @@
+package com.wrbug.developerhelper.ui.activity.appbackup
+
+import android.view.LayoutInflater
+import android.view.ViewGroup
+import android.widget.TextView
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.databinding.ItemBackupDetailInfoBinding
+import com.wrbug.developerhelper.model.entity.BackupAppItemInfo
+import com.wrbug.developerhelper.ui.adapter.delegate.BaseItemViewBindingDelegate
+import com.wrbug.developerhelper.util.format
+import com.wrbug.developerhelper.util.getColor
+import com.wrbug.developerhelper.util.getString
+
+class BackupDetailDelegate(private val appName: String) :
+ BaseItemViewBindingDelegate() {
+ override fun onBindViewHolder(binding: ItemBackupDetailInfoBinding, item: BackupAppItemInfo) {
+ binding.tvTime.text = item.time.format()
+ binding.tvTitle.text = item.memo.ifEmpty {
+ binding.root.context.getString(
+ R.string.backup_default_meme,
+ appName
+ )
+ }
+ binding.tvVersion.text = "${item.versionName}(${item.versionCode})"
+ binding.tvBackupApk.setStatusColor(item.backupApk)
+ binding.tvBackupAndroidData.setStatusColor(item.backupAndroidData)
+ binding.tvBackupData.setStatusColor(item.backupData)
+ }
+
+ private fun TextView.setStatusColor(enable: Boolean) {
+ val color = if (enable) {
+ R.color.material_color_green_500.getColor()
+ } else {
+ R.color.material_color_red_500.getColor()
+ }
+ setTextColor(color)
+ }
+
+ override fun onCreateViewBinding(
+ inflater: LayoutInflater,
+ parent: ViewGroup
+ ): ItemBackupDetailInfoBinding {
+ return ItemBackupDetailInfoBinding.inflate(inflater, parent, false)
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupInfoItemDelegate.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupInfoItemDelegate.kt
new file mode 100644
index 0000000..f5cf1b1
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/appbackup/BackupInfoItemDelegate.kt
@@ -0,0 +1,39 @@
+package com.wrbug.developerhelper.ui.activity.appbackup
+
+import android.view.LayoutInflater
+import android.view.ViewGroup
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.commonutil.SpannableBuilder
+import com.wrbug.developerhelper.databinding.ItemBackupAppInfoBinding
+import com.wrbug.developerhelper.model.entity.BackupAppData
+import com.wrbug.developerhelper.ui.adapter.delegate.BaseItemViewBindingDelegate
+import com.wrbug.developerhelper.util.format
+import com.wrbug.developerhelper.util.getString
+import com.wrbug.developerhelper.util.loadImage
+import com.wrbug.developerhelper.util.setOnDoubleCheckClickListener
+
+class BackupInfoItemDelegate(private val listener: (BackupAppData) -> Unit) :
+ BaseItemViewBindingDelegate() {
+ override fun onBindViewHolder(binding: ItemBackupAppInfoBinding, item: BackupAppData) {
+ binding.tvAppName.text = item.appName
+ binding.tvAppPackageName.text = item.packageName
+ val size = item.backupMap.size
+ binding.tvBackupCount.text = SpannableBuilder.with(
+ binding.root.context,
+ R.string.app_info_backup_count.getString(size)
+ ).addSpanWithBold(size.toString()).build()
+ val time = item.backupMap.values.maxByOrNull { it.time }?.time ?: 0
+ binding.tvLastBackupTime.text = R.string.last_backup_time.getString(time.format())
+ binding.ivIcon.loadImage(item.iconPath, R.drawable.ic_default_app_ico_place_holder)
+ binding.root.setOnDoubleCheckClickListener {
+ listener(item)
+ }
+ }
+
+ override fun onCreateViewBinding(
+ inflater: LayoutInflater,
+ parent: ViewGroup
+ ): ItemBackupAppInfoBinding {
+ return ItemBackupAppInfoBinding.inflate(inflater, parent, false)
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseEditActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseEditActivity.kt
index cdfe075..d8f805a 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseEditActivity.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseEditActivity.kt
@@ -10,23 +10,23 @@ import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
import com.evrencoskun.tableview.listener.ITableViewListener
import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.basecommon.setupActionBar
-import com.wrbug.developerhelper.basecommon.showToast
+import com.wrbug.developerhelper.base.*
import com.wrbug.developerhelper.model.entity.DatabaseTableInfo
import com.wrbug.developerhelper.commonutil.shell.ShellManager
import com.wrbug.developerhelper.util.DatabaseUtils
import com.wrbug.developerhelper.commonutil.dp2px
-import kotlinx.android.synthetic.main.activity_database_edit.*
+import com.wrbug.developerhelper.databinding.ActivityDatabaseEditBinding
import org.jetbrains.anko.doAsync
import org.jetbrains.anko.uiThread
import java.io.File
import java.util.*
import kotlin.collections.ArrayList
-class DatabaseEditActivity : BaseActivity() {
+class DatabaseEditActivity: BaseActivity() {
+
private var filePath: String? = ""
private lateinit var dbPath: File
+ private lateinit var dataBinding: ActivityDatabaseEditBinding
private val tableNames = ArrayList()
private var dbMap: Map = TreeMap()
private val adapter = DatabaseTableAdapter(this)
@@ -40,6 +40,7 @@ class DatabaseEditActivity : BaseActivity() {
}
companion object {
+
fun start(context: Context, filePath: String) {
val intent = Intent(context, DatabaseEditActivity::class.java)
intent.putExtra("filePath", filePath)
@@ -49,7 +50,8 @@ class DatabaseEditActivity : BaseActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_database_edit)
+ dataBinding = ActivityDatabaseEditBinding.inflate(layoutInflater)
+ setContentView(dataBinding.root)
intent?.run {
filePath = getStringExtra("filePath")
}
@@ -67,8 +69,8 @@ class DatabaseEditActivity : BaseActivity() {
}
private fun initTableView() {
- tableView.adapter = adapter
- tableView.tableViewListener = object : ITableViewListener {
+ dataBinding.tableView.adapter = adapter
+ dataBinding.tableView.tableViewListener = object: ITableViewListener {
override fun onCellLongPressed(p0: RecyclerView.ViewHolder, p1: Int, p2: Int) {
}
@@ -96,7 +98,8 @@ class DatabaseEditActivity : BaseActivity() {
if (dstDir.exists().not()) {
dstDir.mkdir()
}
- val success = ShellManager.cpFile(dbPath.absolutePath, "${dstDir.absolutePath}/${dbPath.name}")
+ val success =
+ ShellManager.cpFile(dbPath.absolutePath, "${dstDir.absolutePath}/${dbPath.name}")
if (!success) {
uiThread {
showToast(getString(R.string.get_database_failed))
@@ -115,11 +118,13 @@ class DatabaseEditActivity : BaseActivity() {
}
}
-
private fun setTableContainer() {
- tableNameContainer.removeAllViews()
+ dataBinding.tableNameContainer.removeAllViews()
val layoutParams =
- LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT)
+ LinearLayout.LayoutParams(
+ ViewGroup.LayoutParams.WRAP_CONTENT,
+ ViewGroup.LayoutParams.WRAP_CONTENT
+ )
val dp10 = dp2px(10F)
val dp20 = dp2px(20F)
tableNames.forEachIndexed { index, name ->
@@ -135,7 +140,7 @@ class DatabaseEditActivity : BaseActivity() {
tv.setOnClickListener {
selectTable(it.tag as Int)
}
- tableNameContainer.addView(tv, layoutParams)
+ dataBinding.tableNameContainer.addView(tv, layoutParams)
}
}
@@ -161,9 +166,9 @@ class DatabaseEditActivity : BaseActivity() {
uiThread {
setHasData(!cell.isEmpty())
adapter.setAllItems(keyList, rowHeaders, cell)
- var tv = tableNameContainer.getChildAt(selectedIndex) as TextView
+ var tv = dataBinding.tableNameContainer.getChildAt(selectedIndex) as TextView
tv.setTextColor(resources.getColor(R.color.text_color_666666))
- tv = tableNameContainer.getChildAt(position) as TextView
+ tv = dataBinding.tableNameContainer.getChildAt(position) as TextView
tv.setTextColor(resources.getColor(R.color.colorPrimary))
selectedIndex = position
}
@@ -172,7 +177,7 @@ class DatabaseEditActivity : BaseActivity() {
}
private fun setHasData(hasData: Boolean) {
- noDataTv.visibility = if (hasData) View.GONE else View.VISIBLE
- tableView.visibility = if (hasData) View.VISIBLE else View.GONE
+ dataBinding.noDataTv.visibility = if (hasData) View.GONE else View.VISIBLE
+ dataBinding.tableView.visibility = if (hasData) View.VISIBLE else View.GONE
}
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseTableAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseTableAdapter.kt
index 9ee8514..680b799 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseTableAdapter.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/databaseedit/DatabaseTableAdapter.kt
@@ -1,5 +1,6 @@
package com.wrbug.developerhelper.ui.activity.databaseedit
+import android.annotation.SuppressLint
import android.content.Context
import android.view.View
import android.view.ViewGroup
@@ -55,6 +56,7 @@ class DatabaseTableAdapter(val context: Context) : AbstractTableAdapter? = null
-
- override fun onCreate(savedInstanceState: Bundle?) {
- super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_guide)
- mSectionsPagerAdapter = SectionsPagerAdapter(supportFragmentManager)
- viewPager.adapter = mSectionsPagerAdapter
- viewPager.addOnPageChangeListener(object : ViewPager.OnPageChangeListener {
- override fun onPageSelected(position: Int) {
- currentPosition = position
- updateIndicators(position)
- viewPager.setBackgroundColor(bgColors[position])
- buttonPre.visibility = if (position == 0) View.GONE else View.VISIBLE
- buttonNext.visibility = if (position == 2) View.GONE else View.VISIBLE
- buttonFinish.visibility = if (position == 2) View.VISIBLE else View.GONE
- }
-
- override fun onPageScrollStateChanged(position: Int) {
-
- }
-
- override fun onPageScrolled(p0: Int, p1: Float, p2: Int) {
- val colorUpdate = ArgbEvaluator().evaluate(
- p1,
- bgColors[p0],
- bgColors[if (p0 == 2) p0 else p0 + 1]
- ) as Int
- viewPager.setBackgroundColor(colorUpdate)
- }
-
- })
- indicators = arrayOf(imageViewIndicator0 as View, imageViewIndicator1 as View, imageViewIndicator2 as View)
- }
-
- private fun updateIndicators(position: Int) {
- for (i in 0 until indicators?.size!!) {
- indicators?.get(i)?.setBackgroundResource(
- if (i == position) R.drawable.onboarding_indicator_selected else R.drawable.onboarding_indicator_unselected
- )
- }
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepAccessibilityFragment.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepAccessibilityFragment.kt
deleted file mode 100644
index 1466d40..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepAccessibilityFragment.kt
+++ /dev/null
@@ -1,45 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.guide
-
-import android.content.Intent
-import android.os.Bundle
-import android.provider.Settings
-import android.view.View
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.service.AccessibilityManager
-import com.wrbug.developerhelper.service.DeveloperHelperAccessibilityService
-import kotlinx.android.synthetic.main.fragment_guide.*
-
-class GuideStepAccessibilityFragment : GuideStepFragment() {
- override fun getLabelText(): String {
- return "无障碍功能"
- }
-
- override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
- checkIsOpen()
- icoIv.setImageResource(R.drawable.ic_accessibility_black)
- contentTv.setOnClickListener {
- if (DeveloperHelperAccessibilityService.serviceRunning) {
- return@setOnClickListener
- }
- AccessibilityManager.startService(activity)
-
- }
- }
-
- private fun checkIsOpen() {
- contentTv.text =
- if (DeveloperHelperAccessibilityService.serviceRunning) "无障碍辅助已开启" else "无障碍辅助已关闭,将无法分析布局和页面信息,点击开启"
- }
-
- override fun onResume() {
- super.onResume()
- checkIsOpen()
- }
-
- companion object {
- fun instance(): GuideStepAccessibilityFragment {
- val fragment = GuideStepAccessibilityFragment()
- return fragment
- }
- }
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepFloatWindowFragment.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepFloatWindowFragment.kt
deleted file mode 100644
index 3e5fd57..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepFloatWindowFragment.kt
+++ /dev/null
@@ -1,57 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.guide
-
-import android.content.Intent
-import android.net.Uri
-import android.os.Bundle
-import android.provider.Settings
-import android.view.View
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.service.FloatWindowService
-import com.wrbug.developerhelper.util.DeviceUtils
-import kotlinx.android.synthetic.main.fragment_guide.*
-
-class GuideStepFloatWindowFragment : GuideStepFragment() {
- override fun getLabelText(): String {
- return getString(R.string.float_window_permission)
- }
-
- override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
- icoIv.setImageResource(R.drawable.ic_float_air_bubble)
- contentTv.setOnClickListener {
- if (DeviceUtils.isFloatWindowOpened(activity!!)) {
- return@setOnClickListener
- }
- startActivityForResult(
- Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:${activity?.packageName}")),
- 0
- )
- }
- }
-
- override fun onResume() {
- super.onResume()
- checkFloatWindow()
- }
-
- private fun checkFloatWindow() {
- if (activity != null && DeviceUtils.isFloatWindowOpened(activity!!)) {
- contentTv.text = getString(R.string.float_window_opened)
- FloatWindowService.start(activity!!)
- } else {
- contentTv.text = getString(R.string.float_window_closed)
- }
- }
-
- companion object {
- fun instance(): GuideStepFloatWindowFragment {
- val fragment = GuideStepFloatWindowFragment()
- return fragment
- }
- }
-
- override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
- if (requestCode == 0) {
- checkFloatWindow()
- }
- }
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepFragment.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepFragment.kt
deleted file mode 100644
index 772c1d3..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/GuideStepFragment.kt
+++ /dev/null
@@ -1,22 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.guide
-
-import android.os.Bundle
-import android.view.LayoutInflater
-import android.view.View
-import android.view.ViewGroup
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseFragment
-import kotlinx.android.synthetic.main.fragment_guide.view.*
-
-
- abstract class GuideStepFragment : BaseFragment() {
- override fun onCreateView(
- inflater: LayoutInflater, container: ViewGroup?,
- savedInstanceState: Bundle?
- ): View? {
- val rootView = inflater.inflate(R.layout.fragment_guide, container, false)
- rootView.labelTv.text = getLabelText()
- return rootView
- }
- abstract fun getLabelText(): String
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/SectionsPagerAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/SectionsPagerAdapter.kt
deleted file mode 100644
index 9515409..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/guide/SectionsPagerAdapter.kt
+++ /dev/null
@@ -1,20 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.guide
-
-import androidx.fragment.app.Fragment
-import androidx.fragment.app.FragmentManager
-import androidx.fragment.app.FragmentPagerAdapter
-
-class SectionsPagerAdapter(fm: FragmentManager) : FragmentPagerAdapter(fm) {
- private val fragments = arrayOf(
- GuideStepFloatWindowFragment.instance(),
- GuideStepAccessibilityFragment.instance()
- )
-
- override fun getItem(position: Int): Fragment {
- return fragments[position]
- }
-
- override fun getCount(): Int {
- return fragments.size
- }
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoDialog.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoDialog.kt
index e3f193b..0e85d4c 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoDialog.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoDialog.kt
@@ -9,22 +9,27 @@ import android.view.ViewGroup
import androidx.fragment.app.DialogFragment
import com.wrbug.developerhelper.R
import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.commonutil.entity.TopActivityInfo
import com.wrbug.developerhelper.commonutil.UiUtils
import com.wrbug.developerhelper.commonutil.dp2px
-import kotlinx.android.synthetic.main.dialog_apk_info.*
+import com.wrbug.developerhelper.databinding.DialogApkInfoBinding
+import com.wrbug.developerhelper.util.isPortrait
+import io.reactivex.rxjava3.disposables.CompositeDisposable
class AppInfoDialog : DialogFragment() {
- private var apkInfo: ApkInfo? = null
- private var topActivity: TopActivityInfo? = null
+
+ private val apkInfo: ApkInfo? by lazy {
+ arguments?.getParcelable("apkInfo")
+ }
private var listener: AppInfoDialogEventListener? = null
+ private lateinit var binding: DialogApkInfoBinding
+ private lateinit var disposable: CompositeDisposable
+ private val pagerAdapter by lazy {
+ AppInfoPagerAdapter(this, disposable)
+ }
+
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
- setStyle(DialogFragment.STYLE_NORMAL, R.style.FullScreenDialog)
- arguments?.let {
- apkInfo = it.getParcelable("apkInfo")
- topActivity = it.getParcelable("topActivity")
- }
+ setStyle(STYLE_NORMAL, R.style.FullScreenDialog)
}
override fun onAttach(activity: Activity) {
@@ -34,39 +39,44 @@ class AppInfoDialog : DialogFragment() {
}
}
- override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
- val view = inflater.inflate(R.layout.dialog_apk_info, container, false)
- dialog?.window.takeUnless {
- it == null
- }?.run {
+ override fun onCreateView(
+ inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?
+ ): View {
+ disposable = CompositeDisposable()
+ binding = DialogApkInfoBinding.inflate(inflater, container, false)
+ dialog?.window?.run {
val layoutParams = attributes
- layoutParams.width = ViewGroup.LayoutParams.MATCH_PARENT
- layoutParams.height = UiUtils.getDeviceHeight() / 2 + dp2px(40F)
+ if (isPortrait()) {
+ layoutParams.width = ViewGroup.LayoutParams.MATCH_PARENT
+ layoutParams.height = UiUtils.getDeviceHeight() / 2 + dp2px(40F)
+ setGravity(Gravity.TOP)
+ } else {
+ layoutParams.width = UiUtils.getDeviceWidth() / 2
+ layoutParams.height = ViewGroup.LayoutParams.MATCH_PARENT
+ setGravity(Gravity.END)
+ }
attributes = layoutParams
- setGravity(Gravity.TOP)
}
- return view
+ return binding.root
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
- titleContainer.setPadding(0, UiUtils.getStatusHeight(), 0, 0)
- activity?.let {
- val pagerAdapter = AppInfoPagerAdapter(this, apkInfo, topActivity)
- pagerAdapter.listener = listener
- viewPager.adapter = pagerAdapter
- tabLayout.setupWithViewPager(viewPager)
- }
- apkInfo?.let { it ->
- logoIv.setImageDrawable(it.getIco())
- titleTv.text = it.getAppName()
- subTitleTv.text = it.applicationInfo.packageName
+ binding.titleContainer.setPadding(0, UiUtils.getStatusHeight(), 0, 0)
+ pagerAdapter.listener = listener
+ binding.viewPager.adapter = pagerAdapter
+ binding.tabLayout.setupWithViewPager(binding.viewPager)
+ apkInfo?.let {
+ binding.logoIv.setImageDrawable(it.getIco())
+ binding.titleTv.text = it.getAppName()
+ binding.subTitleTv.text = it.applicationInfo.packageName
}
+ pagerAdapter.loadData(apkInfo)
}
-
override fun onDestroyView() {
- super.onDestroyView()
listener?.close()
+ disposable.dispose()
+ super.onDestroyView()
}
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoPagerAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoPagerAdapter.kt
index 1b0a383..44079d1 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoPagerAdapter.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/AppInfoPagerAdapter.kt
@@ -1,8 +1,6 @@
package com.wrbug.developerhelper.ui.activity.hierachy
import android.content.Context
-import android.content.pm.ApplicationInfo
-import android.content.pm.PackageInfo
import android.view.View
import android.view.ViewGroup
import androidx.annotation.Keep
@@ -10,60 +8,66 @@ import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import androidx.viewpager.widget.PagerAdapter
import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.versionCodeLong
import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.commonutil.entity.TopActivityInfo
import com.wrbug.developerhelper.commonutil.shell.ShellManager
import com.wrbug.developerhelper.ui.decoration.SpaceItemDecoration
import com.wrbug.developerhelper.ui.widget.appdatainfoview.AppDataInfoView
import com.wrbug.developerhelper.ui.widget.appsettingview.AppSettingView
import com.wrbug.developerhelper.ui.widget.layoutinfoview.infopage.InfoAdapter
import com.wrbug.developerhelper.ui.widget.layoutinfoview.infopage.ItemInfo
-import com.wrbug.developerhelper.util.EnforceUtils
import com.wrbug.developerhelper.commonutil.UiUtils
-import com.wrbug.developerhelper.ipc.processshare.manager.AppXposedProcessDataManager
import com.wrbug.developerhelper.util.format
import com.wrbug.developerhelper.util.getString
-import org.jetbrains.anko.doAsync
-import org.jetbrains.anko.uiThread
+import io.reactivex.rxjava3.disposables.CompositeDisposable
import java.util.ArrayList
@Keep
class AppInfoPagerAdapter(
- private val dialog: AppInfoDialog,
- private val apkInfo: ApkInfo?,
- private val topActivity: TopActivityInfo?
-) :
- PagerAdapter() {
- private val context: Context = dialog.activity!!
+ private val dialog: AppInfoDialog, private val disposable: CompositeDisposable
+) : PagerAdapter() {
+
+ private val context: Context = dialog.requireContext()
private val tabList = arrayListOf()
private val viewList = arrayListOf()
- private val adapter = InfoAdapter(context)
- private val enforceItem =
- ItemInfo(context.getString(R.string.enforce_type), context.getString(R.string.analyzing))
+ private val adapter by lazy {
+ InfoAdapter(context, analyzeItem)
+ }
var listener: AppInfoDialogEventListener? = null
- private val itemInfos = ArrayList()
+ private val itemInfos = ArrayList()
+ private val analyzeItem by lazy {
+ ItemInfo(getString(R.string.page_analyze), getString(R.string.click_to_analyze)).apply {
+ showCopy = false
+ textColor = context.resources.getColor(R.color.colorPrimaryDark)
+ setOnClickListener(View.OnClickListener {
+ listener?.showHierachyView()
+ dialog.dismissAllowingStateLoss()
+ })
+ }
+ }
- init {
- initAppInfoTab()
- initAppDataInfoTab()
- initAppSettingTab()
+ fun loadData(apkInfo: ApkInfo?) {
+ initAppInfoTab(apkInfo)
+ initAppDataInfoTab(apkInfo)
+ initAppSettingTab(apkInfo)
+ notifyDataSetChanged()
}
- private fun initAppSettingTab() {
+ private fun initAppSettingTab(apkInfo: ApkInfo?) {
tabList.add(context.getString(R.string.app_setting))
val view = AppSettingView(context)
- view.apkInfo = apkInfo
+ view.setApkInfo(apkInfo)
viewList.add(view)
}
- private fun initAppDataInfoTab() {
+ private fun initAppDataInfoTab(apkInfo: ApkInfo?) {
tabList.add(context.getString(R.string.data_info))
val appDataInfoView = AppDataInfoView(context)
viewList.add(appDataInfoView)
appDataInfoView.apkInfo = apkInfo
}
- private fun initAppInfoTab() {
+ private fun initAppInfoTab(apkInfo: ApkInfo?) {
tabList.add(context.getString(R.string.base_info))
val rv = RecyclerView(context)
viewList.add(rv)
@@ -74,83 +78,31 @@ class AppInfoPagerAdapter(
itemDecoration.setFirstTopPadding(UiUtils.dp2px(context, 10F))
rv.addItemDecoration(itemDecoration)
apkInfo?.let { it ->
- val item =
- ItemInfo(getString(R.string.page_analyze), getString(R.string.click_to_analyze))
- item.setOnClickListener(View.OnClickListener {
- listener?.showHierachyView()
- dialog.dismissAllowingStateLoss()
- })
- item.textColor = context.resources.getColor(R.color.colorPrimaryDark)
- itemInfos.add(item)
- headerItemHook(it.packageInfo, it.applicationInfo, itemInfos)
- itemInfos.add(ItemInfo("VersionCode", it.packageInfo.versionCode))
- itemInfos.add(ItemInfo("VersionName", it.packageInfo.versionName))
- topActivity?.let {
- itemInfos.add(ItemInfo("Activity", it.activity))
- it.fragments.takeUnless { fragments ->
- fragments.isNullOrEmpty()
- }?.forEach {
- if (it.hidden.not()) {
- itemInfos.add(ItemInfo("Fragment", it.name))
- }
- }
- }
+ itemInfos.add(ItemInfo("PackageName", it.packageInfo.packageName))
it.applicationInfo.className?.let { name ->
itemInfos.add(ItemInfo("Application", name))
}
- itemInfos.add(enforceItem)
+ it.topActivity.takeIf { it.isNotEmpty() }?.let {
+ itemInfos.add(ItemInfo("Activity", it))
+ }
+ itemInfos.add(ItemInfo("VersionName", it.packageInfo.versionName))
+ itemInfos.add(ItemInfo("VersionCode", it.packageInfo.versionCodeLong))
itemInfos.add(ItemInfo("uid", it.applicationInfo.uid))
- ShellManager.getPid(it.packageInfo.packageName).takeUnless {
- it.isEmpty()
- }?.let {
+ ShellManager.getPid(it.packageInfo.packageName).takeUnless { it.isEmpty() }?.let {
itemInfos.add(ItemInfo("Pid", it))
}
itemInfos.add(
ItemInfo(
- getString(R.string.first_install_time),
- it.packageInfo.firstInstallTime.format()
+ getString(R.string.first_install_time), it.packageInfo.firstInstallTime.format()
)
)
itemInfos.add(
ItemInfo(
- getString(R.string.last_update_time),
- it.packageInfo.lastUpdateTime.format()
+ getString(R.string.last_update_time), it.packageInfo.lastUpdateTime.format()
)
)
itemInfos.add(ItemInfo("DataDir", it.applicationInfo.dataDir))
adapter.setItems(itemInfos)
- getEnforce(it.packageInfo.packageName)
- }
- }
-
- private fun headerItemHook(
- i: PackageInfo,
- info: ApplicationInfo,
- itemInfo: ArrayList
- ) {
- }
-
- fun findItemById(id: String): ItemInfo? {
- itemInfos.forEach {
- if (it.id == id) {
- return it
- }
- }
- return null
- }
-
- private fun setEnforceType(type: EnforceUtils.EnforceType) {
- enforceItem.content = type.type
- }
-
- private fun getEnforce(packageName: String) {
- doAsync {
- val type = EnforceUtils.getEnforceType(packageName)
- uiThread {
- setEnforceType(type)
- adapter.notifyItemChanged(enforceItem)
- }
-
}
}
@@ -162,7 +114,6 @@ class AppInfoPagerAdapter(
return tabList.size
}
-
override fun instantiateItem(container: ViewGroup, position: Int): Any {
container.addView(viewList[position])
return viewList[position]
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/HierarchyActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/HierarchyActivity.kt
index e76802b..b148447 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/HierarchyActivity.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/hierachy/HierarchyActivity.kt
@@ -7,44 +7,38 @@ import android.content.Intent
import android.content.IntentFilter
import android.os.Bundle
import android.view.View
-import com.google.gson.reflect.TypeToken
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.constant.ReceiverConstant
+import com.wrbug.developerhelper.base.BaseActivity
+import com.wrbug.developerhelper.base.entry.HierarchyNode
+import com.wrbug.developerhelper.base.registerReceiverComp
import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
-import com.wrbug.developerhelper.commonutil.entity.TopActivityInfo
-import com.wrbug.developerhelper.ui.widget.hierarchyView.HierarchyView
-import com.wrbug.developerhelper.commonutil.JsonHelper
import com.wrbug.developerhelper.constant.ReceiverConstant.ACTION_FINISH_HIERACHY_Activity
+import com.wrbug.developerhelper.databinding.ActivityHierarchyBinding
import com.wrbug.developerhelper.service.FloatWindowService
-import com.wrbug.developerhelper.ui.widget.layoutinfoview.LayoutInfoView
-import com.wrbug.developerhelper.ui.widget.layoutinfoview.OnNodeChangedListener
-import kotlinx.android.synthetic.main.activity_hierarchy.*
+import com.wrbug.developerhelper.ui.widget.hierarchyView.HierarchyView
+import com.wrbug.developerhelper.ui.widget.layoutinfoview.LayoutInfoDialog
import java.lang.ref.WeakReference
-class HierarchyActivity : BaseActivity(), AppInfoDialogEventListener, OnNodeChangedListener {
-
+class HierarchyActivity : BaseActivity(), AppInfoDialogEventListener {
- private var apkInfo: ApkInfo? = null
- private var nodeList: ArrayList? = null
- private var nodeMap: HashMap? = null
+ private val apkInfo: ApkInfo? by lazy {
+ intent?.getParcelableExtra("apkInfo")
+ }
+ private val nodeList: ArrayList? by lazy {
+ intent?.getParcelableArrayListExtra("node")
+ }
private var showHierachyView = false
- private var topActivity: TopActivityInfo? = null
+ private lateinit var binding: ActivityHierarchyBinding
companion object {
+
fun start(
- context: Context?,
- apkInfo: ApkInfo?,
- node: ArrayList?,
- topActivity: TopActivityInfo?
+ context: Context?, apkInfo: ApkInfo?, node: ArrayList
) {
val intent = Intent(context, HierarchyActivity::class.java)
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
val bundle = Bundle()
bundle.putParcelable("apkInfo", apkInfo)
bundle.putParcelableArrayList("node", node)
- bundle.putParcelable("topActivity", topActivity)
intent.putExtras(bundle)
context?.startActivity(intent)
}
@@ -57,7 +51,7 @@ class HierarchyActivity : BaseActivity(), AppInfoDialogEventListener, OnNodeChan
override fun onReceive(context: Context?, intent: Intent?) {
when (intent?.action) {
- ReceiverConstant.ACTION_FINISH_HIERACHY_Activity -> {
+ ACTION_FINISH_HIERACHY_Activity -> {
reference?.get()?.finish()
}
@@ -69,35 +63,26 @@ class HierarchyActivity : BaseActivity(), AppInfoDialogEventListener, OnNodeChan
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_hierarchy)
- intent?.run {
- apkInfo = getParcelableExtra("apkInfo")
- nodeList = getParcelableArrayListExtra("node")
- checkNodeList()
- topActivity = getParcelableExtra("topActivity")
- val json = getStringExtra("nodeMap")
- nodeMap = JsonHelper.fromJson(
- json,
- object : TypeToken>() {}.type
- )
- }
+ binding = ActivityHierarchyBinding.inflate(layoutInflater)
+ setContentView(binding.root)
+ checkNodeList()
val filter = IntentFilter(ACTION_FINISH_HIERACHY_Activity)
receiver.setActivity(this)
- registerReceiver(receiver, filter)
+ registerReceiverComp(receiver, filter)
showAppInfoDialog()
FloatWindowService.setFloatButtonVisible(this, false)
}
private fun checkNodeList() {
nodeList ?: return
- if (nodeList?.size ?: 0 <= 1) { + if ((nodeList?.size ?: 0) <= 1) { return } var hierarchyNode: HierarchyNode? = null nodeList?.forEach { if (hierarchyNode == null) { hierarchyNode = it - } else if (it.screenBounds?.contains(hierarchyNode?.screenBounds) == true) { + } else if (hierarchyNode?.screenBounds?.let { it1 -> it.screenBounds?.contains(it1) } == true) {
hierarchyNode = it
}
}
@@ -111,35 +96,30 @@ class HierarchyActivity : BaseActivity(), AppInfoDialogEventListener, OnNodeChan
val dialog = AppInfoDialog()
val bundle = Bundle()
bundle.putParcelable("apkInfo", apkInfo)
- bundle.putParcelable("topActivity", topActivity)
dialog.arguments = bundle
dialog.show(supportFragmentManager, "")
}
override fun showHierachyView() {
showHierachyView = true
- hierarchyView.setHierarchyNodes(nodeList)
- hierarchyView.setOnHierarchyNodeClickListener(object :
+ binding.hierarchyView.setHierarchyNodes(nodeList)
+ binding.hierarchyView.setOnHierarchyNodeClickListener(object :
HierarchyView.OnHierarchyNodeClickListener {
override fun onClick(node: HierarchyNode, parentNode: HierarchyNode?) {
- hierarchyDetailView.visibility = View.VISIBLE
- hierarchyDetailView.setNode(node, parentNode)
- val layoutInfoView = LayoutInfoView(context, nodeList, node)
- layoutInfoView.setOnNodeChangedListener(this@HierarchyActivity)
- layoutInfoView.show()
+ binding.hierarchyDetailView.visibility = View.VISIBLE
+ binding.hierarchyDetailView.setNode(node, parentNode)
+ LayoutInfoDialog.show(supportFragmentManager, nodeList, node) { node, parentNode ->
+ binding.hierarchyDetailView.setNode(node, parentNode)
+ }
}
override fun onSelectedNodeChanged(node: HierarchyNode, parentNode: HierarchyNode?) {
- hierarchyDetailView.visibility = View.VISIBLE
- hierarchyDetailView.setNode(node, parentNode)
+ binding.hierarchyDetailView.visibility = View.VISIBLE
+ binding.hierarchyDetailView.setNode(node, parentNode)
}
})
}
- override fun onChanged(node: HierarchyNode, parentNode: HierarchyNode?) {
- hierarchyDetailView.setNode(node, parentNode)
- }
-
override fun onDestroy() {
FloatWindowService.setFloatButtonVisible(this, true)
unregisterReceiver(receiver)
@@ -153,18 +133,17 @@ class HierarchyActivity : BaseActivity(), AppInfoDialogEventListener, OnNodeChan
}
override fun onBackPressed() {
- if (hierarchyDetailView.visibility == View.VISIBLE) {
- hierarchyDetailView.visibility = View.GONE
+ if (binding.hierarchyDetailView.visibility == View.VISIBLE) {
+ binding.hierarchyDetailView.visibility = View.GONE
return
}
if (showHierachyView) {
showHierachyView = false
- hierarchyView.visibility = View.GONE
+ binding.hierarchyView.visibility = View.GONE
showAppInfoDialog()
return
}
super.onBackPressed()
}
-
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/MainActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/MainActivity.kt
index 07854fa..d6bb8b7 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/MainActivity.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/MainActivity.kt
@@ -1,201 +1,220 @@
package com.wrbug.developerhelper.ui.activity.main
+import android.Manifest
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import android.content.IntentFilter
import android.net.Uri
+import android.os.Build
import android.os.Bundle
import android.provider.Settings
import android.view.Menu
import android.view.MenuItem
-import android.widget.CompoundButton
import androidx.appcompat.app.AlertDialog
-import androidx.databinding.DataBindingUtil
-import com.wrbug.developerhelper.BuildConfig
+import androidx.core.view.isVisible
import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseVMActivity
-import com.wrbug.developerhelper.basecommon.obtainViewModel
-import com.wrbug.developerhelper.basecommon.setupActionBar
+import com.wrbug.developerhelper.base.BaseActivity
+import com.wrbug.developerhelper.base.registerReceiverComp
+import com.wrbug.developerhelper.base.requestStoragePermission
+import com.wrbug.developerhelper.base.setupActionBar
import com.wrbug.developerhelper.commonutil.ClipboardUtils
-import com.wrbug.developerhelper.commonutil.shell.Callback
import com.wrbug.developerhelper.commonutil.shell.ShellManager
-import com.wrbug.developerhelper.commonutil.toInt
+import com.wrbug.developerhelper.util.setOnDoubleCheckClickListener
import com.wrbug.developerhelper.constant.ReceiverConstant
import com.wrbug.developerhelper.databinding.ActivityMainBinding
+import com.wrbug.developerhelper.mmkv.ConfigKv
+import com.wrbug.developerhelper.mmkv.manager.MMKVManager
import com.wrbug.developerhelper.model.entity.VersionInfo
import com.wrbug.developerhelper.service.AccessibilityManager
+import com.wrbug.developerhelper.service.DeveloperHelperAccessibilityService
import com.wrbug.developerhelper.service.FloatWindowService
-import com.wrbug.developerhelper.ui.activity.main.viewmodel.MainViewModel
-import com.wrbug.developerhelper.ui.activity.xposed.xposedsetting.XposedSettingActivity
-import com.wrbug.developerhelper.ui.widget.settingitemview.SettingItemView
+import com.wrbug.developerhelper.ui.activity.appbackup.BackupAppActivity
import com.wrbug.developerhelper.util.DeviceUtils
-import com.wrbug.developerhelper.util.UpdateUtils
-import kotlinx.android.synthetic.main.activity_main.*
-
-
-class MainActivity : BaseVMActivity() {
+class MainActivity : BaseActivity() {
+ private val configKv: ConfigKv = MMKVManager.get(ConfigKv::class.java)
+ private val binding: ActivityMainBinding by lazy {
+ ActivityMainBinding.inflate(layoutInflater)
+ }
- lateinit var binding: ActivityMainBinding
- lateinit var xposedSettingView: SettingItemView
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
if (DeviceUtils.isFloatWindowOpened()) {
FloatWindowService.start(this)
- "".toInt()
- }
- binding = DataBindingUtil.setContentView(this, R.layout.activity_main)
- binding.presenter = Presenter()
- xposedSettingView = binding.xposedSettingView
- binding.mainVm = vm
- setupActionBar(R.id.toolbar) {
-
}
+ setContentView(binding.root)
+ setupActionBar(R.id.toolbar)
ShellManager.openAccessibilityService()
+ initView()
initListener()
val filter = IntentFilter(ReceiverConstant.ACTION_ACCESSIBILITY_SERVICE_STATUS_CHANGED)
- registerReceiver(receiver, filter)
+ registerReceiverComp(receiver, filter)
+ }
+
+ private fun initView() {
+ if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.TIRAMISU) {
+ binding.notificationSettingView.checkable = false
+ binding.notificationSettingView.isVisible = true
+ binding.notificationSettingView.checked =
+ hasPermission(Manifest.permission.POST_NOTIFICATIONS)
+ } else {
+ binding.notificationSettingView.isVisible = false
+ }
}
private fun initListener() {
- floatWindowSettingView.setOnCheckedChangeListener(CompoundButton.OnCheckedChangeListener { _, isChecked ->
+ binding.floatWindowSettingView.setOnCheckedChangeListener { _, isChecked ->
if (isChecked && DeviceUtils.isFloatWindowOpened()) {
FloatWindowService.start(this)
} else {
FloatWindowService.stop(this)
}
- })
- rootSettingView.setOnCheckedChangeListener(CompoundButton.OnCheckedChangeListener { _, _ ->
-
- })
- xposedSettingView.setOnClickListener {
- if (xposedSettingView.isChecked().not()) {
- showSnack(getString(R.string.open_xposed_first))
- return@setOnClickListener
+ }
+ binding.backupAppSettingView.setOnDoubleCheckClickListener {
+ requestStoragePermission {
+ startActivity(Intent(this, BackupAppActivity::class.java))
}
- startActivity(Intent(this, XposedSettingActivity::class.java))
}
- }
-
- override fun getViewModel(): MainViewModel {
- return obtainViewModel(MainViewModel::class.java)
- }
-
- override fun onDestroy() {
- super.onDestroy()
- unregisterReceiver(receiver)
- }
-
- inner class Presenter {
- fun onAccessibilityClick() {
- if (!accessibilitySettingView.checked) {
+ binding.accessibilitySettingView.setOnDoubleCheckClickListener {
+ if (!binding.accessibilitySettingView.checked) {
AccessibilityManager.startAccessibilitySetting(context)
} else {
showSnack(getString(R.string.accessibility_service_opened))
}
}
-
- fun onFloatWindowClick() {
- if (!floatWindowSettingView.checked) {
+ binding.floatWindowSettingView.setOnDoubleCheckClickListener {
+ if (!binding.floatWindowSettingView.checked) {
startActivityForResult(
- Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:$packageName")),
- 0
+ Intent(
+ Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:$packageName")
+ ), 0
)
} else {
showSnack(getString(R.string.float_window_opened))
}
}
+ binding.rootSettingView.setOnDoubleCheckClickListener {
+ if (!DeviceUtils.isRoot()) {
+ showSnack(R.string.devices_is_not_root)
+ return@setOnDoubleCheckClickListener
+ }
+ val isRootEnable = binding.rootSettingView.isChecked()
+ binding.rootSettingView.checked = !isRootEnable
+ configKv.setOpenRoot(!isRootEnable)
+ }
+ binding.notificationSettingView.setOnSwitcherClickListener {
+ if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.TIRAMISU) {
+ requestPermission(arrayOf(Manifest.permission.POST_NOTIFICATIONS),
+ object : PermissionCallback() {
+ override fun granted() {
+ binding.notificationSettingView.checked = true
+ }
+ })
+ }
+ }
+ }
+
+
+ override fun onResume() {
+ super.onResume()
+ checkStatus()
+ }
+
- fun onRootClick() {
- vm.toggleRootPermission()
+ private fun checkStatus() {
+ binding.accessibilitySettingView.checked =
+ DeveloperHelperAccessibilityService.serviceRunning
+ binding.floatWindowSettingView.checked = DeviceUtils.isFloatWindowOpened()
+ if (configKv.isOpenRoot()) {
+ if (DeviceUtils.isRoot()) {
+ binding.rootSettingView.checked = true
+ } else {
+ binding.rootSettingView.checked = false
+ configKv.setOpenRoot(false)
+ }
+ }
+ if (DeviceUtils.isFloatWindowOpened()) {
+ FloatWindowService.start(application)
}
}
+ override fun onDestroy() {
+ super.onDestroy()
+ unregisterReceiver(receiver)
+ }
+
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
menuInflater.inflate(R.menu.menu_main, menu)
return super.onCreateOptionsMenu(menu)
}
- override fun onOptionsItemSelected(item: MenuItem?): Boolean {
- item?.let {
- when (it.itemId) {
- R.id.about_menu -> {
- showAboutDialog()
- }
- R.id.exit_menu -> {
- showExitMenuDialog()
- }
+ override fun onOptionsItemSelected(item: MenuItem): Boolean {
+ when (item.itemId) {
+ R.id.about_menu -> {
+ showAboutDialog()
}
- }
+ R.id.exit_menu -> {
+ showExitMenuDialog()
+ }
+ }
return super.onOptionsItemSelected(item)
}
- private fun showExitMenuDialog() = AlertDialog.Builder(this)
- .setTitle(R.string.notice)
+ private fun showExitMenuDialog() = AlertDialog.Builder(this).setTitle(R.string.notice)
.setMessage(getString(R.string.exit_content))
.setPositiveButton(getString(R.string.ok)) { _, _ ->
FloatWindowService.stop(this)
finish()
- }
- .setNegativeButton(getString(R.string.cancel), null)
- .create()
- .show()
+ }.setNegativeButton(getString(R.string.cancel), null).create().show()
- private fun showAboutDialog() = AlertDialog.Builder(this)
- .setTitle(R.string.about)
+ private fun showAboutDialog() = AlertDialog.Builder(this).setTitle(R.string.about)
.setMessage(getString(R.string.about_content))
.setPositiveButton(getString(R.string.copy_group_number)) { _, _ ->
ClipboardUtils.saveClipboardText(this, "627962572")
showSnack(R.string.copy_success)
- }
- .setNeutralButton(getString(R.string.check_update)) { _, _ ->
+ }.setNeutralButton(getString(R.string.check_update)) { _, _ ->
checkUpdate(true)
- }
- .create().show()
-
- override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
-
- }
+ }.create().show()
private fun checkUpdate(showSnack: Boolean = false) {
- if (showSnack) {
- showSnack(getString(R.string.checking_update))
- }
- UpdateUtils.checkUpdate(object : Callback {
- override fun onSuccess(data: VersionInfo) {
- if (BuildConfig.VERSION_NAME == data.versionName) {
- showSnack(getString(R.string.no_new_version))
- return
- }
- showUpdateDialog(data)
- }
-
- override fun onFailed(msg: String) {
- if (showSnack) {
- showSnack(getString(R.string.check_update_failed))
- }
- }
- })
+// if (showSnack) {
+// showSnack(getString(R.string.checking_update))
+// }
+// UpdateUtils.checkUpdate(object : Callback {
+// override fun onSuccess(data: VersionInfo) {
+// if (BuildConfig.VERSION_NAME == data.versionName) {
+// showSnack(getString(R.string.no_new_version))
+// return
+// }
+// showUpdateDialog(data)
+// }
+//
+// override fun onFailed(msg: String) {
+// if (showSnack) {
+// showSnack(getString(R.string.check_update_failed))
+// }
+// }
+// })
}
- private fun showUpdateDialog(data: VersionInfo) = AlertDialog.Builder(this)
- .setTitle(getString(R.string.find_new_version))
- .setMessage("版本号:${data.versionName}\n更新时间:${data.updateDate}\n大小:${data.size}\n版本说明:\n${data.feature}")
- .setPositiveButton(getString(R.string.download)) { _, _ ->
- val intent = Intent(Intent.ACTION_VIEW)
- val uri = Uri.parse(data.downloadUrl)
- intent.data = uri
- startActivity(intent)
- }
- .create().show()
+ private fun showUpdateDialog(data: VersionInfo) =
+ AlertDialog.Builder(this).setTitle(getString(R.string.find_new_version))
+ .setMessage("版本号:${data.versionName}\n更新时间:${data.updateDate}\n大小:${data.size}\n版本说明:\n${data.feature}")
+ .setPositiveButton(getString(R.string.download)) { _, _ ->
+ val intent = Intent(Intent.ACTION_VIEW)
+ val uri = Uri.parse(data.downloadUrl)
+ intent.data = uri
+ startActivity(intent)
+ }.create().show()
private val receiver = object : BroadcastReceiver() {
override fun onReceive(context: Context?, intent: Intent?) {
intent?.run {
if (action == ReceiverConstant.ACTION_ACCESSIBILITY_SERVICE_STATUS_CHANGED) {
- accessibilitySettingView.checked = intent.getBooleanExtra("status", false)
+ binding.accessibilitySettingView.checked =
+ intent.getBooleanExtra("status", false)
}
}
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/viewmodel/MainViewModel.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/viewmodel/MainViewModel.kt
deleted file mode 100644
index c6959ae..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/main/viewmodel/MainViewModel.kt
+++ /dev/null
@@ -1,50 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.main.viewmodel
-
-import androidx.databinding.ObservableBoolean
-import androidx.databinding.ObservableField
-import androidx.lifecycle.Lifecycle
-import androidx.lifecycle.OnLifecycleEvent
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseViewModel
-import com.wrbug.developerhelper.mmkv.ConfigKv
-import com.wrbug.developerhelper.mmkv.manager.MMKVManager
-import com.wrbug.developerhelper.service.DeveloperHelperAccessibilityService
-import com.wrbug.developerhelper.service.FloatWindowService
-import com.wrbug.developerhelper.util.DeviceUtils
-import javax.inject.Inject
-
-class MainViewModel @Inject constructor() : BaseViewModel() {
- val openAccessibility = ObservableBoolean()
- val openFloatWindow = ObservableBoolean()
- val openRoot = ObservableBoolean()
- private val configKv: ConfigKv = MMKVManager.get(ConfigKv::class.java)
- private fun checkStatus() {
- openAccessibility.set(DeveloperHelperAccessibilityService.serviceRunning)
- openFloatWindow.set(DeviceUtils.isFloatWindowOpened())
- if (configKv.isOpenRoot()) {
- if (DeviceUtils.isRoot()) {
- openRoot.set(true)
- } else {
- openRoot.set(false)
- configKv.setOpenRoot(false)
- }
- }
- if (DeviceUtils.isFloatWindowOpened()) {
- FloatWindowService.start(application)
- }
- }
-
- @OnLifecycleEvent(Lifecycle.Event.ON_RESUME)
- fun onResume() {
- checkStatus()
- }
-
- fun toggleRootPermission() {
- if (!DeviceUtils.isRoot()) {
- showSnack(R.string.devices_is_not_root)
- return
- }
- openRoot.set(!openRoot.get())
- configKv.setOpenRoot(openRoot.get())
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceEditActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceEditActivity.kt
index eeaf954..04089b9 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceEditActivity.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceEditActivity.kt
@@ -6,31 +6,40 @@ import android.os.Bundle
import android.view.Menu
import android.view.MenuItem
import androidx.appcompat.app.AlertDialog
+import androidx.core.view.isVisible
import androidx.recyclerview.widget.LinearLayoutManager
import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.basecommon.setupActionBar
+import com.wrbug.developerhelper.base.BaseActivity
+import com.wrbug.developerhelper.base.setupActionBar
import com.wrbug.developerhelper.commonutil.AppManagerUtils
-import com.wrbug.developerhelper.commonutil.shell.ShellManager
+import com.wrbug.developerhelper.commonutil.addTo
+import com.wrbug.developerhelper.commonutil.dpInt
+import com.wrbug.developerhelper.commonutil.runOnIO
+import com.wrbug.developerhelper.util.startPageLoading
+import com.wrbug.developerhelper.util.stopPageLoading
+import com.wrbug.developerhelper.databinding.ActivitySharedPreferenceEditBinding
import com.wrbug.developerhelper.ui.decoration.SpaceItemDecoration
-import com.wrbug.developerhelper.util.OutSharedPreferenceManager
-import com.wrbug.developerhelper.util.XmlUtil
-import com.wrbug.developerhelper.commonutil.dp2px
-import kotlinx.android.synthetic.main.activity_shared_preference_edit.*
-import org.jetbrains.anko.doAsync
-import org.jetbrains.anko.uiThread
+import com.wrbug.developerhelper.util.OutSharedPreference
import java.io.File
-class SharedPreferenceEditActivity : BaseActivity(),
- SharedPreferenceListAdapter.OnValueChangedListener {
+class SharedPreferenceEditActivity : BaseActivity() {
- private var filePath: String = ""
- private var filePackageName = ""
- private var appName = ""
+ private val filePath by lazy {
+ intent?.getStringExtra(KEY_FILE_PATH).orEmpty()
+ }
+ private val filePackageName by lazy {
+ intent?.getStringExtra(KEY_PACKAGE_NAME).orEmpty()
+ }
+ private val appName by lazy {
+ intent?.getStringExtra(KEY_APP_NAME).orEmpty()
+ }
private lateinit var adapter: SharedPreferenceListAdapter
private var saveMenuItem: MenuItem? = null
+ private lateinit var binding: ActivitySharedPreferenceEditBinding
+ private lateinit var outSharedPreference: OutSharedPreference
companion object {
+
private const val KEY_FILE_PATH = "filePath"
private const val KEY_PACKAGE_NAME = "packageName"
private const val KEY_APP_NAME = "appName"
@@ -46,41 +55,42 @@ class SharedPreferenceEditActivity : BaseActivity(),
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_shared_preference_edit)
- intent?.let {
- filePath = it.getStringExtra(KEY_FILE_PATH)
- filePackageName = it.getStringExtra(KEY_PACKAGE_NAME)
- appName = it.getStringExtra(KEY_APP_NAME)
- }
+ binding = ActivitySharedPreferenceEditBinding.inflate(layoutInflater).inject()
setupActionBar(R.id.toolbar) {
if (filePath.isNotEmpty()) {
title = File(filePath).name
}
}
-
- sprefRv.layoutManager = LinearLayoutManager(this)
+ outSharedPreference = OutSharedPreference(this, filePath)
+ binding.sprefRv.layoutManager = LinearLayoutManager(this)
adapter = SharedPreferenceListAdapter(this)
- sprefRv.adapter = adapter
- adapter.setOnValueChangedListener(this)
- val spaceItemDecoration = SpaceItemDecoration(dp2px(10F))
- sprefRv.addItemDecoration(spaceItemDecoration)
+ binding.sprefRv.adapter = adapter
+ adapter.setOnValueChangedListener {
+ saveMenuItem?.isVisible = it
+ }
+ val spaceItemDecoration = SpaceItemDecoration(
+ 24.dpInt(context),
+ 16.dpInt(context),
+ 24.dpInt(context),
+ 0,
+ 16.dpInt(context),
+ 40.dpInt(context)
+ )
+ binding.sprefRv.addItemDecoration(spaceItemDecoration)
parseXml()
-
- }
-
- override fun onChanged(changed: Boolean) {
- saveMenuItem?.isVisible = changed
}
private fun parseXml() {
- doAsync {
- val xml = ShellManager.catFile(filePath)
- val list = XmlUtil.parseSharedPreference(xml)
- uiThread {
- saveMenuItem?.isVisible = false
- adapter.setData(list)
- }
- }
+ binding.flLoading.startPageLoading()
+ outSharedPreference.parse().runOnIO().subscribe({
+ binding.emptyView.isVisible = it.isEmpty()
+ binding.sprefRv.isVisible = it.isNotEmpty()
+ saveMenuItem?.isVisible = false
+ adapter.setData(it)
+ binding.flLoading.stopPageLoading()
+ }, {
+ binding.flLoading.stopPageLoading()
+ }).addTo(disposable)
}
override fun onBackPressed() {
@@ -95,69 +105,50 @@ class SharedPreferenceEditActivity : BaseActivity(),
private fun showSaveDialog() {
AlertDialog.Builder(this).setMessage(getString(R.string.confirm_shared_preference_save))
- .setTitle(R.string.notice)
- .setPositiveButton(R.string.save_and_exit) { _, _ ->
- if (doSave()) {
- showDialog(
- R.string.notice,
- getString(R.string.save_shared_preference_success, appName),
- R.string.ok,
- R.string.cancel,
- {
- AppManagerUtils.restartApp(
- this@SharedPreferenceEditActivity,
- filePackageName
- )
- finish()
- }, {
- finish()
- }
- )
- } else {
- showSnack(getString(R.string.save_shared_preference_failed))
- }
- }
- .setNegativeButton(getString(R.string.do_not_save)) { _, _ ->
+ .setTitle(R.string.notice).setPositiveButton(R.string.save_and_exit) { _, _ ->
+ doSave()
+ }.setNegativeButton(getString(R.string.do_not_save)) { _, _ ->
finish()
}.show()
}
- private fun doSave(): Boolean {
+ private fun doSave() {
val data = adapter.getData()
- val file = OutSharedPreferenceManager.saveToFile(this, data)
- val success = ShellManager.catFile(file.absolutePath, filePath, "666")
- file.delete()
- return success
+ outSharedPreference.saveToFile(this, data).runOnIO().subscribe({
+ if (it) {
+ parseXml()
+ showDialog(R.string.notice,
+ getString(R.string.save_shared_preference_success, appName),
+ R.string.ok,
+ R.string.cancel,
+ {
+ AppManagerUtils.restartApp(
+ this@SharedPreferenceEditActivity, filePackageName
+ )
+ finish()
+ })
+ } else {
+ showSnack(getString(R.string.save_shared_preference_failed))
+ }
+ }, {
+
+ }).addTo(disposable)
}
- override fun onOptionsItemSelected(item: MenuItem?): Boolean {
- item?.run {
- when (itemId) {
- R.id.save_menu -> {
- if (!doSave()) {
- showSnack(getString(R.string.save_shared_preference_failed))
- return@run
- }
- parseXml()
- showDialog(
- R.string.notice,
- getString(R.string.save_shared_preference_success, appName),
- R.string.ok,
- R.string.cancel,
- {
- AppManagerUtils.restartApp(
- this@SharedPreferenceEditActivity,
- filePackageName
- )
- finish()
- }
- )
- }
+ override fun onOptionsItemSelected(item: MenuItem): Boolean {
+ when (item.itemId) {
+ R.id.save_menu -> {
+ doSave()
}
}
return super.onOptionsItemSelected(item)
}
+ override fun onDestroy() {
+ super.onDestroy()
+ outSharedPreference.deleteTmpFile()
+ }
+
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
menuInflater.inflate(R.menu.menu_shared_preference_edit, menu)
saveMenuItem = menu?.findItem(R.id.save_menu)
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceListAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceListAdapter.kt
index a952e6e..0e68495 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceListAdapter.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/activity/sharedpreferencesedit/SharedPreferenceListAdapter.kt
@@ -4,33 +4,34 @@ import android.content.Context
import android.text.Editable
import android.text.TextWatcher
import android.view.LayoutInflater
-import android.view.View
import android.view.ViewGroup
-import android.widget.EditText
-import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.util.inVisible
+import com.wrbug.developerhelper.databinding.ItemSharedPreferenceInfoBinding
import com.wrbug.developerhelper.model.entity.SharedPreferenceItemInfo
class SharedPreferenceListAdapter(val context: Context) :
RecyclerView.Adapter() {
private val data: ArrayList = arrayListOf()
- private var onValueChangedListener: OnValueChangedListener? = null
+ private var onValueChangedListener: ((Boolean) -> Unit)? = null
private var changedFlag: Long = 0
- fun setOnValueChangedListener(listener: OnValueChangedListener) {
+ fun setOnValueChangedListener(listener: (Boolean) -> Unit) {
onValueChangedListener = listener
}
fun setData(array: Array) {
data.clear()
changedFlag = 0
- data.addAll(array.sortedBy { it.key })
+ data.addAll(array.sortedBy { it.key.lowercase() })
notifyDataSetChanged()
}
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder {
- return ViewHolder(LayoutInflater.from(context).inflate(R.layout.item_shared_preference_info, parent, false))
+ return ViewHolder(
+ ItemSharedPreferenceInfoBinding.inflate(LayoutInflater.from(context), parent, false)
+ )
}
override fun getItemCount(): Int {
@@ -39,18 +40,15 @@ class SharedPreferenceListAdapter(val context: Context) :
override fun onBindViewHolder(holder: ViewHolder, position: Int) {
val sharedPreferenceItemInfo = data[position]
- holder.titleTv?.text = sharedPreferenceItemInfo.key
- holder.contentTv?.setText(sharedPreferenceItemInfo.value)
- holder.restoreTv?.visibility =
- if (sharedPreferenceItemInfo.value == sharedPreferenceItemInfo.newValue) {
- View.INVISIBLE
- } else {
- View.VISIBLE
- }
+ holder.removeTextChangedListener()
+ holder.binding.titleTv.text = sharedPreferenceItemInfo.key
+ holder.binding.contentEt.setText(sharedPreferenceItemInfo.newValue)
+ holder.binding.restoreTv.inVisible =
+ sharedPreferenceItemInfo.value == sharedPreferenceItemInfo.newValue
if (sharedPreferenceItemInfo.isValueValid()) {
- holder.contentTv?.error = null
+ holder.binding.contentEt.error = null
} else {
- holder.contentTv?.error = context.getString(R.string.input_error)
+ holder.binding.contentEt.error = context.getString(R.string.input_error)
}
holder.tag = sharedPreferenceItemInfo
holder.addTextChangedListener()
@@ -61,21 +59,26 @@ class SharedPreferenceListAdapter(val context: Context) :
}
- inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
- val titleTv: TextView? = itemView.findViewById(R.id.titleTv)
- val contentTv: EditText? = itemView.findViewById(R.id.contentEt)
- var restoreTv: TextView? = itemView.findViewById(R.id.restoreTv)
+ inner class ViewHolder(val binding: ItemSharedPreferenceInfoBinding) :
+ RecyclerView.ViewHolder(binding.root) {
var tag: SharedPreferenceItemInfo? = null
var textWatcher: TextWatcher? = null
init {
- restoreTv?.setOnClickListener {
+ binding.restoreTv.setOnClickListener {
tag?.run {
- contentTv?.setText(value)
+ binding.contentEt.setText(value)
}
}
}
+ fun removeTextChangedListener() {
+ if (textWatcher != null) {
+ binding.contentEt.removeTextChangedListener(textWatcher)
+ textWatcher = null
+ }
+ }
+
fun addTextChangedListener() {
if (textWatcher != null) {
return
@@ -90,9 +93,9 @@ class SharedPreferenceListAdapter(val context: Context) :
tag?.let {
it.newValue = toString()
if (it.isValueValid()) {
- contentTv?.error = null
+ binding.contentEt.error = null
} else {
- contentTv?.error = context.getString(R.string.input_error)
+ binding.contentEt.error = context.getString(R.string.input_error)
}
}
changedFlag = if (toString() == tag?.value) {
@@ -100,30 +103,22 @@ class SharedPreferenceListAdapter(val context: Context) :
} else {
changedFlag or 1L shl index
}
- onValueChangedListener?.onChanged(changedFlag != 0L)
+ onValueChangedListener?.invoke(changedFlag != 0L)
}
- restoreTv?.visibility =
- if (tag?.value == tag?.newValue) {
- View.INVISIBLE
- } else {
- View.VISIBLE
- }
+ binding.restoreTv.inVisible = tag?.value == tag?.newValue
}
- override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
+ override fun beforeTextChanged(
+ s: CharSequence?, start: Int, count: Int, after: Int
+ ) {
}
override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
}
}
- contentTv?.addTextChangedListener(textWatcher)
+ binding.contentEt.addTextChangedListener(textWatcher)
}
}
-
-
- interface OnValueChangedListener {
- fun onChanged(changed: Boolean)
- }
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/BaseXposedAppManagerActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/BaseXposedAppManagerActivity.kt
deleted file mode 100644
index 8e6ab49..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/BaseXposedAppManagerActivity.kt
+++ /dev/null
@@ -1,67 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed
-
-import android.os.Bundle
-import androidx.recyclerview.widget.DividerItemDecoration
-import androidx.recyclerview.widget.LinearLayoutManager
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.basecommon.setupActionBar
-import com.wrbug.developerhelper.commonutil.UiUtils
-import com.wrbug.developerhelper.ui.decoration.SpaceItemDecoration
-import kotlinx.android.synthetic.main.activity_base_xposed_app_manager.*
-import org.jetbrains.anko.doAsync
-import org.jetbrains.anko.uiThread
-
-abstract class BaseXposedAppManagerActivity : BaseActivity(),
- XposedAppListAdapter.OnItemChangedListener {
- private val adapter: XposedAppListAdapter by lazy {
- XposedAppListAdapter(this).apply {
- setOnItemChangedListener(this@BaseXposedAppManagerActivity)
- }
- }
-
- override fun onCreate(savedInstanceState: Bundle?) {
- super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_base_xposed_app_manager)
- setupActionBar(R.id.toolbar) {
- title = getManagerTitle()
- }
- swipeLayout.setOnRefreshListener {
- getData(true)
- }
- initRv()
- getData()
- }
-
- private fun initRv() {
- rv.layoutManager = LinearLayoutManager(this)
- val dp8 = UiUtils.dp2px(this, 8F)
- val dp16 = UiUtils.dp2px(this, 16F)
- val dp4 = UiUtils.dp2px(this, 4F)
- rv.addItemDecoration(SpaceItemDecoration(dp4, dp8, dp4, dp8).apply {
- setLastBottomPadding(dp16)
- })
- rv.addItemDecoration(DividerItemDecoration(this, DividerItemDecoration.VERTICAL))
- rv.adapter = adapter
- }
-
- abstract fun getManagerTitle(): String
-
- abstract fun getAppEnableStatus(): Map
-
- private fun getData(force: Boolean = false) {
- swipeLayout.isRefreshing = true
- doAsync {
- val map = getAppEnableStatus()
- uiThread {
- swipeLayout.isRefreshing = false
- setData(map, force)
- }
- }
- }
-
- private fun setData(map: Map, force: Boolean) {
- swipeLayout.isRefreshing = false
- adapter.setAppEnableStatus(map, force)
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/XposedAppListAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/XposedAppListAdapter.kt
deleted file mode 100644
index a5d81f2..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/XposedAppListAdapter.kt
+++ /dev/null
@@ -1,101 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed
-
-import android.content.Context
-import android.view.LayoutInflater
-import android.view.View
-import android.view.ViewGroup
-import android.widget.ImageView
-import android.widget.Switch
-import android.widget.TextView
-import androidx.appcompat.widget.SwitchCompat
-import androidx.recyclerview.widget.RecyclerView
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.commonutil.AppInfoManager
-import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.ui.widget.bottommenu.BottomMenu
-import com.wrbug.developerhelper.ui.widget.bottommenu.OnItemClickListener
-import com.wrbug.developerhelper.ipc.processshare.DumpDexListProcessData
-import com.wrbug.developerhelper.ipc.processshare.ProcessDataCreator
-
-class XposedAppListAdapter(val context: Context) :
- RecyclerView.Adapter() {
- companion object {
- private val cache = ArrayList()
- }
-
- private val list = ArrayList()
- private val appEnableStatusMap = HashMap()
- private var listener: OnItemChangedListener? = null
-
- init {
- initApkInfo()
- }
-
- private fun initApkInfo(force: Boolean = false) {
- if (force || list.isEmpty()) {
- cache.clear()
- list.clear()
- cache.addAll(AppInfoManager.getAllApps().values)
- cache.sortBy { it.getAppName() }
- list.addAll(cache)
- }
- }
-
- override fun onCreateViewHolder(parent: ViewGroup, viewType: Int) =
- ViewHolder(
- LayoutInflater.from(context).inflate(
- R.layout.item_shell_app_info,
- parent,
- false
- )
- )
-
- fun setAppEnableStatus(data: Map, force: Boolean = false) {
- initApkInfo(force)
- appEnableStatusMap.clear()
- appEnableStatusMap.putAll(data)
- list.sortBy { appEnableStatusMap[it.applicationInfo.packageName] != true }
- notifyDataSetChanged()
- }
-
- override fun getItemCount(): Int = list.size
- fun setOnItemChangedListener(listener: OnItemChangedListener) {
- this.listener = listener
- }
-
- override fun onBindViewHolder(holder: ViewHolder, position: Int) {
- val apkInfo = list[position]
- holder.removeListener()
- holder.icoIv.setImageDrawable(apkInfo.getIco())
- holder.appNameTv.text = apkInfo.getAppName()
- holder.packageNameTv.text = apkInfo.packageInfo.packageName
- holder.toggle.isChecked = appEnableStatusMap[apkInfo.applicationInfo.packageName] == true
- holder.apkInfo = apkInfo
- holder.setListener()
- }
-
- inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
- var icoIv: ImageView = itemView.findViewById(R.id.icoIv)
- var appNameTv: TextView = itemView.findViewById(R.id.appNameTv)
- var packageNameTv: TextView = itemView.findViewById(R.id.packageNameTv)
- var apkInfo: ApkInfo? = null
- val toggle: SwitchCompat = itemView.findViewById(R.id.toggle)
-
- fun removeListener() {
- toggle.setOnCheckedChangeListener(null)
- }
-
- fun setListener() {
- toggle.setOnCheckedChangeListener { _, isChecked ->
- apkInfo?.let {
- listener?.onChanged(this@XposedAppListAdapter, it, isChecked)
- appEnableStatusMap[it.applicationInfo.packageName] = isChecked
- }
- }
- }
- }
-
- interface OnItemChangedListener {
- fun onChanged(adapter: XposedAppListAdapter, apkInfo: ApkInfo, isChecked: Boolean)
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/appxposedmodulemanager/AppXposedModuleManagerActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/appxposedmodulemanager/AppXposedModuleManagerActivity.kt
deleted file mode 100644
index bb63185..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/appxposedmodulemanager/AppXposedModuleManagerActivity.kt
+++ /dev/null
@@ -1,13 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed.appxposedmodulemanager
-
-import androidx.appcompat.app.AppCompatActivity
-import android.os.Bundle
-import com.wrbug.developerhelper.R
-
-class AppXposedModuleManagerActivity : AppCompatActivity() {
-
- override fun onCreate(savedInstanceState: Bundle?) {
- super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_app_xposed_module_manager)
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/appxposedsetting/AppXposedSettingActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/appxposedsetting/AppXposedSettingActivity.kt
deleted file mode 100644
index a177647..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/appxposedsetting/AppXposedSettingActivity.kt
+++ /dev/null
@@ -1,34 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed.appxposedsetting
-
-import android.content.Context
-import android.content.Intent
-import androidx.appcompat.app.AppCompatActivity
-import android.os.Bundle
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.ipc.processshare.manager.AppXposedProcessDataManager
-import com.wrbug.developerhelper.ipc.processshare.manager.DumpDexListProcessDataManager
-import com.wrbug.developerhelper.ui.activity.xposed.BaseXposedAppManagerActivity
-import com.wrbug.developerhelper.ui.activity.xposed.XposedAppListAdapter
-import com.wrbug.developerhelper.ui.activity.xposed.shellmanager.ShellAppManagerActivity
-
-class AppXposedSettingActivity : BaseXposedAppManagerActivity() {
-
- override fun getManagerTitle(): String = getString(R.string.app_xposed_function_manager)
-
- override fun getAppEnableStatus(): Map {
- val packageNames = AppXposedProcessDataManager.instance.getAppXposedStatusList()
- return packageNames
- }
-
- override fun onChanged(adapter: XposedAppListAdapter, apkInfo: ApkInfo, isChecked: Boolean) {
- AppXposedProcessDataManager.instance.setAppXposedStatusList(apkInfo.applicationInfo.packageName to isChecked)
- }
-
- companion object {
- fun start(context: Context) {
- context.startActivity(Intent(context, AppXposedSettingActivity::class.java))
- }
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/datafindermanager/DataFinderManagerActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/datafindermanager/DataFinderManagerActivity.kt
deleted file mode 100644
index 20a46d1..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/datafindermanager/DataFinderManagerActivity.kt
+++ /dev/null
@@ -1,30 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed.datafindermanager
-
-import android.content.Context
-import android.content.Intent
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.ipc.processshare.manager.DataFinderListProcessDataManager
-import com.wrbug.developerhelper.ipc.processshare.manager.DumpDexListProcessDataManager
-import com.wrbug.developerhelper.ui.activity.xposed.BaseXposedAppManagerActivity
-import com.wrbug.developerhelper.ui.activity.xposed.XposedAppListAdapter
-
-class DataFinderManagerActivity : BaseXposedAppManagerActivity() {
- override fun getManagerTitle(): String = getString(R.string.data_finder_manager)
-
- override fun getAppEnableStatus(): Map {
- val packageNames = DataFinderListProcessDataManager.instance.getData()
- return packageNames
- }
-
- override fun onChanged(adapter: XposedAppListAdapter, apkInfo: ApkInfo, isChecked: Boolean) {
- DataFinderListProcessDataManager.instance.setData(apkInfo.applicationInfo.packageName to isChecked)
- }
-
- companion object {
- fun start(context: Context) {
- context.startActivity(Intent(context, DataFinderManagerActivity::class.java))
- }
- }
-
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/shellmanager/ShellAppManagerActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/shellmanager/ShellAppManagerActivity.kt
deleted file mode 100644
index 5374803..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/shellmanager/ShellAppManagerActivity.kt
+++ /dev/null
@@ -1,29 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed.shellmanager
-
-import android.content.Context
-import android.content.Intent
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.commonutil.entity.ApkInfo
-import com.wrbug.developerhelper.ipc.processshare.manager.DumpDexListProcessDataManager
-import com.wrbug.developerhelper.ui.activity.xposed.BaseXposedAppManagerActivity
-import com.wrbug.developerhelper.ui.activity.xposed.XposedAppListAdapter
-
-class ShellAppManagerActivity : BaseXposedAppManagerActivity() {
- override fun getManagerTitle(): String = getString(R.string.shell_app_manager)
-
- override fun getAppEnableStatus(): Map {
- val packageNames = DumpDexListProcessDataManager.instance.getData()
- return packageNames
- }
-
- override fun onChanged(adapter: XposedAppListAdapter, apkInfo: ApkInfo, isChecked: Boolean) {
- DumpDexListProcessDataManager.instance.setData(apkInfo.applicationInfo.packageName to isChecked)
- }
-
- companion object {
- fun start(context: Context) {
- context.startActivity(Intent(context, ShellAppManagerActivity::class.java))
- }
- }
-
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/xposedsetting/XposedSettingActivity.kt b/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/xposedsetting/XposedSettingActivity.kt
deleted file mode 100644
index 535d7c2..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/activity/xposed/xposedsetting/XposedSettingActivity.kt
+++ /dev/null
@@ -1,28 +0,0 @@
-package com.wrbug.developerhelper.ui.activity.xposed.xposedsetting
-
-import android.os.Bundle
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.basecommon.setupActionBar
-import com.wrbug.developerhelper.ipc.processshare.manager.FileProcessDataManager
-import com.wrbug.developerhelper.ui.activity.xposed.appxposedsetting.AppXposedSettingActivity
-import com.wrbug.developerhelper.ui.activity.xposed.shellmanager.ShellAppManagerActivity
-import kotlinx.android.synthetic.main.activity_xposed_setting.*
-import java.io.File
-
-class XposedSettingActivity : BaseActivity() {
-
- override fun onCreate(savedInstanceState: Bundle?) {
- super.onCreate(savedInstanceState)
- setContentView(R.layout.activity_xposed_setting)
- setupActionBar(R.id.toolbar) {
- title = getString(R.string.xposed_setting)
- }
- appXposedSettingItemView.setOnClickListener {
- AppXposedSettingActivity.start(this)
- }
- shellSettingItemView.setOnClickListener {
- ShellAppManagerActivity.start(this)
- }
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/adapter/ExMultiTypeAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/ExMultiTypeAdapter.kt
new file mode 100644
index 0000000..fc192c7
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/ExMultiTypeAdapter.kt
@@ -0,0 +1,36 @@
+package com.wrbug.developerhelper.ui.adapter
+
+import com.drakeet.multitype.MultiTypeAdapter
+import com.wrbug.developerhelper.ui.adapter.bean.EmptyViewItemData
+import com.wrbug.developerhelper.ui.adapter.bean.LoadingViewItemData
+import com.wrbug.developerhelper.ui.adapter.delegate.EmptyViewItemDelegate
+import com.wrbug.developerhelper.ui.adapter.delegate.LoadingItemDelegate
+
+class ExMultiTypeAdapter : MultiTypeAdapter() {
+
+ companion object {
+ fun get() = lazy {
+ ExMultiTypeAdapter()
+ }
+ }
+
+ init {
+ register(EmptyViewItemDelegate())
+ register(LoadingItemDelegate())
+ }
+
+ fun showEmpty(title: String = "") {
+ items = listOf(EmptyViewItemData(title))
+ notifyDataSetChanged()
+ }
+
+ fun showLoading() {
+ items = listOf(LoadingViewItemData)
+ notifyDataSetChanged()
+ }
+
+ fun loadData(list: List) {
+ items = list.toList()
+ notifyDataSetChanged()
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/adapter/bean/EmptyViewItemData.kt b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/bean/EmptyViewItemData.kt
new file mode 100644
index 0000000..190efdb
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/bean/EmptyViewItemData.kt
@@ -0,0 +1,5 @@
+package com.wrbug.developerhelper.ui.adapter.bean
+
+class EmptyViewItemData(val title: String = "")
+
+object LoadingViewItemData
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/BaseItemViewBindingDelegate.kt b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/BaseItemViewBindingDelegate.kt
new file mode 100644
index 0000000..ad8b57c
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/BaseItemViewBindingDelegate.kt
@@ -0,0 +1,27 @@
+package com.wrbug.developerhelper.ui.adapter.delegate
+
+import android.content.Context
+import android.view.LayoutInflater
+import android.view.ViewGroup
+import androidx.viewbinding.ViewBinding
+import com.drakeet.multitype.ItemViewDelegate
+
+abstract class BaseItemViewBindingDelegate :
+ ItemViewDelegate>() {
+
+ final override fun onBindViewHolder(holder: ViewBindingHolder, item: T) {
+ onBindViewHolder(holder.binding, item)
+ }
+
+ final override fun onCreateViewHolder(
+ context: Context,
+ parent: ViewGroup
+ ): ViewBindingHolder {
+ return ViewBindingHolder(onCreateViewBinding(LayoutInflater.from(context), parent))
+ }
+
+ abstract fun onCreateViewBinding(inflater: LayoutInflater, parent: ViewGroup): B
+
+ abstract fun onBindViewHolder(binding: B, item: T)
+
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/EmptyViewItemDelegate.kt b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/EmptyViewItemDelegate.kt
new file mode 100644
index 0000000..e191a21
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/EmptyViewItemDelegate.kt
@@ -0,0 +1,27 @@
+package com.wrbug.developerhelper.ui.adapter.delegate
+
+import android.content.Context
+import android.view.LayoutInflater
+import android.view.ViewGroup
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.databinding.ItemEmptyDataBinding
+import com.wrbug.developerhelper.ui.adapter.bean.EmptyViewItemData
+import com.wrbug.developerhelper.util.getString
+
+class EmptyViewItemDelegate :
+ BaseItemViewBindingDelegate() {
+ override fun onBindViewHolder(binding: ItemEmptyDataBinding, item: EmptyViewItemData) {
+ if (item.title.isNotEmpty()) {
+ binding.emptyView.setTitle(item.title)
+ } else {
+ binding.emptyView.setTitle(R.string.no_data.getString())
+ }
+ }
+
+ override fun onCreateViewBinding(
+ inflater: LayoutInflater,
+ parent: ViewGroup
+ ): ItemEmptyDataBinding {
+ return ItemEmptyDataBinding.inflate(inflater, parent, false)
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/LoadingItemDelegate.kt b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/LoadingItemDelegate.kt
new file mode 100644
index 0000000..d778443
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/LoadingItemDelegate.kt
@@ -0,0 +1,20 @@
+package com.wrbug.developerhelper.ui.adapter.delegate
+
+import android.view.LayoutInflater
+import android.view.ViewGroup
+import com.wrbug.developerhelper.databinding.ItemLoadingDataBinding
+import com.wrbug.developerhelper.ui.adapter.bean.LoadingViewItemData
+
+class LoadingItemDelegate :
+ BaseItemViewBindingDelegate() {
+ override fun onBindViewHolder(binding: ItemLoadingDataBinding, item: LoadingViewItemData) {
+
+ }
+
+ override fun onCreateViewBinding(
+ inflater: LayoutInflater,
+ parent: ViewGroup
+ ): ItemLoadingDataBinding {
+ return ItemLoadingDataBinding.inflate(inflater, parent, false)
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/ViewBindingHolder.kt b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/ViewBindingHolder.kt
new file mode 100644
index 0000000..4eb89fe
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/adapter/delegate/ViewBindingHolder.kt
@@ -0,0 +1,7 @@
+package com.wrbug.developerhelper.ui.adapter.delegate
+
+import androidx.recyclerview.widget.RecyclerView
+import androidx.viewbinding.ViewBinding
+
+class ViewBindingHolder(val binding: T) :
+ RecyclerView.ViewHolder(binding.root)
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/decoration/SpaceItemDecoration.kt b/app/src/main/java/com/wrbug/developerhelper/ui/decoration/SpaceItemDecoration.kt
index d31ab71..5fb18cf 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/decoration/SpaceItemDecoration.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/decoration/SpaceItemDecoration.kt
@@ -3,6 +3,7 @@ package com.wrbug.developerhelper.ui.decoration
import android.graphics.Rect
import android.view.View
import androidx.recyclerview.widget.RecyclerView
+import com.wrbug.developerhelper.commonutil.dpInt
/**
* SpaceItemDecoration
@@ -10,15 +11,37 @@ import androidx.recyclerview.widget.RecyclerView
* @author wrbug
* @since 2017年9月29日
*/
-class SpaceItemDecoration : RecyclerView.ItemDecoration {
- private var leftSpace: Int = 0
- private var topSpace: Int = 0
- private var rightSpace: Int = 0
- private var bottomSpace: Int = 0
- private var firstTopSpace = 0
- private var lastBottomSpace: Int = 0
-
- override fun getItemOffsets(outRect: Rect, view: View, parent: RecyclerView, state: RecyclerView.State) {
+class SpaceItemDecoration(
+ private val leftSpace: Int = 0,
+ private val topSpace: Int = 0,
+ private val rightSpace: Int = 0,
+ private val bottomSpace: Int = 0,
+ private var firstTopSpace: Int = 0,
+ private var lastBottomSpace: Int = 0,
+) : RecyclerView.ItemDecoration() {
+ companion object {
+ val standard = SpaceItemDecoration(
+ 24.dpInt(),
+ 12.dpInt(),
+ 24.dpInt(),
+ 12.dpInt(),
+ 24.dpInt(),
+ 40.dpInt()
+ )
+ }
+
+ constructor(space: Int) : this(
+ bottomSpace = space,
+ topSpace = space,
+ rightSpace = space,
+ leftSpace = space,
+ lastBottomSpace = space,
+ firstTopSpace = space
+ )
+
+ override fun getItemOffsets(
+ outRect: Rect, view: View, parent: RecyclerView, state: RecyclerView.State
+ ) {
super.getItemOffsets(outRect, view, parent, state)
outRect.left = leftSpace
outRect.right = rightSpace
@@ -35,14 +58,6 @@ class SpaceItemDecoration : RecyclerView.ItemDecoration {
}
- constructor(space: Int) {
- bottomSpace = space
- topSpace = space
- rightSpace = space
- leftSpace = space
- lastBottomSpace = space
- firstTopSpace = space
- }
fun setLastBottomPadding(space: Int) {
lastBottomSpace = space
@@ -51,13 +66,4 @@ class SpaceItemDecoration : RecyclerView.ItemDecoration {
fun setFirstTopPadding(space: Int) {
firstTopSpace = space
}
-
- constructor(leftSpace: Int, topSpace: Int, rightSpace: Int, bottomSpace: Int) {
- this.leftSpace = leftSpace
- this.topSpace = topSpace
- this.rightSpace = rightSpace
- this.bottomSpace = bottomSpace
- lastBottomSpace = bottomSpace
- firstTopSpace = topSpace
- }
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/appbar/AppBar.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appbar/AppBar.kt
new file mode 100644
index 0000000..c208920
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appbar/AppBar.kt
@@ -0,0 +1,88 @@
+package com.wrbug.developerhelper.ui.widget.appbar
+
+import android.content.Context
+import android.util.AttributeSet
+import android.view.LayoutInflater
+import android.view.View
+import android.view.ViewGroup
+import android.widget.FrameLayout
+import androidx.constraintlayout.widget.ConstraintLayout
+import androidx.core.view.isGone
+import androidx.core.view.isInvisible
+import androidx.core.view.isVisible
+import androidx.fragment.app.FragmentActivity
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.databinding.ViewAppBarBinding
+import com.wrbug.developerhelper.util.getColor
+import com.wrbug.developerhelper.util.setOnDoubleCheckClickListener
+
+class AppBar @JvmOverloads constructor(
+ context: Context, attrs: AttributeSet? = null
+) : FrameLayout(context, attrs) {
+
+ private val binding = ViewAppBarBinding.inflate(LayoutInflater.from(context), this, false)
+
+ init {
+ setBackgroundColor(R.color.colorPrimaryDark.getColor(context))
+ addView(binding.root)
+ binding.ivBack.setOnDoubleCheckClickListener {
+ if (context is FragmentActivity) {
+ context.onBackPressedDispatcher.onBackPressed()
+ }
+ }
+ attrs?.let {
+ initAttr(it)
+ }
+ }
+
+ private fun initAttr(attrs: AttributeSet) {
+ with(context.obtainStyledAttributes(attrs, R.styleable.AppBar)) {
+ val title = getString(R.styleable.AppBar_title)
+ setTitle(title)
+ val subTitle = getString(R.styleable.AppBar_subTitle)
+ setSubTitle(subTitle)
+ val showBack = getBoolean(R.styleable.AppBar_showBack, false)
+ showBack(showBack)
+ recycle()
+ }
+ }
+
+ fun setSubTitle(subTitle: CharSequence?) {
+ binding.tvSubTitle.text = subTitle
+ binding.tvSubTitle.isGone = subTitle.isNullOrEmpty()
+ }
+
+ fun showBack(showBack: Boolean) {
+ binding.ivBack.isInvisible = !showBack
+ }
+
+ fun setTitle(title: CharSequence?) {
+ binding.tvTitle.text = title
+ }
+
+ override fun addView(child: View?, index: Int, params: ViewGroup.LayoutParams?) {
+ if (child == binding.root) {
+ super.addView(child, index, params)
+ return
+ }
+ binding.flRight.addView(child, index, params)
+ }
+
+ override fun addViewInLayout(
+ child: View?,
+ index: Int,
+ params: ViewGroup.LayoutParams?,
+ preventRequestLayout: Boolean
+ ): Boolean {
+ return super.addViewInLayout(child, index, params, preventRequestLayout)
+ }
+
+
+// fun setMenu(title: String?, listener: (View) -> Unit) {
+// binding.tvRightBtn.text = title
+// binding.tvRightBtn.isGone = title.isNullOrEmpty()
+// binding.tvRightBtn.setOnDoubleCheckClickListener {
+// listener(it)
+// }
+// }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/appdatainfoview/AppDataInfoView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appdatainfoview/AppDataInfoView.kt
index 1ad8f19..292ab52 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/appdatainfoview/AppDataInfoView.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appdatainfoview/AppDataInfoView.kt
@@ -14,37 +14,39 @@ import com.wrbug.developerhelper.commonutil.shell.ShellManager
import com.wrbug.developerhelper.ui.activity.databaseedit.DatabaseEditActivity
import com.wrbug.developerhelper.ui.activity.sharedpreferencesedit.SharedPreferenceEditActivity
import com.wrbug.developerhelper.commonutil.UiUtils
-import kotlinx.android.synthetic.main.view_app_data_info.view.*
+import com.wrbug.developerhelper.databinding.ViewAppDataInfoBinding
import org.jetbrains.anko.doAsync
import org.jetbrains.anko.uiThread
import java.io.File
-class AppDataInfoView : FrameLayout {
+class AppDataInfoView: FrameLayout {
+
+ private lateinit var binding: ViewAppDataInfoBinding
var apkInfo: ApkInfo? = null
set(value) {
field = value
loadData()
}
- constructor(context: Context) : super(context) {
+ constructor(context: Context): super(context) {
initView()
}
- constructor(context: Context, attributeSet: AttributeSet) : super(context, attributeSet) {
+ constructor(context: Context, attributeSet: AttributeSet): super(context, attributeSet) {
initView()
}
private fun initView() {
- LayoutInflater.from(context).inflate(R.layout.view_app_data_info, this)
+ binding = ViewAppDataInfoBinding.inflate(LayoutInflater.from(context), this, true)
}
private fun loadData() {
apkInfo?.run {
- apkPathTv.text = applicationInfo.publicSourceDir
+ binding.apkPathTv.text = applicationInfo.publicSourceDir
val apkSignInfo = ApkUtils.getApkSignInfo(context, applicationInfo.packageName)
- apkSha1Tv.text = apkSignInfo.sha1
- apkMd5Tv.text = apkSignInfo.md5
+ binding.apkSha1Tv.text = apkSignInfo.sha1
+ binding.apkMd5Tv.text = apkSignInfo.md5
getSharedPreferencesFiles(applicationInfo.packageName)
getDatabaseFiles(applicationInfo.packageName)
}
@@ -55,10 +57,10 @@ class AppDataInfoView : FrameLayout {
val sqliteFiles = ShellManager.getSqliteFiles(packageName)
uiThread {
if (sqliteFiles.isEmpty()) {
- defaultDbTv.setText(R.string.none)
+ binding.defaultDbTv.setText(R.string.none)
return@uiThread
}
- databaseContainer.removeAllViews()
+ binding.databaseContainer.removeAllViews()
val params =
LinearLayout.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT)
for (sqliteFile in sqliteFiles) {
@@ -68,7 +70,7 @@ class AppDataInfoView : FrameLayout {
textView.setTextColor(resources.getColor(R.color.item_content_text))
textView.setPadding(0, UiUtils.dp2px(context, 8F), 0, 0)
textView.tag = sqliteFile
- databaseContainer.addView(textView, params)
+ binding.databaseContainer.addView(textView, params)
textView.setOnClickListener {
DatabaseEditActivity.start(context, (it.tag as File).absolutePath)
}
@@ -83,10 +85,10 @@ class AppDataInfoView : FrameLayout {
val files = AppInfoManager.getSharedPreferencesFiles(packageName)
uiThread {
if (files.isEmpty()) {
- defaultSpTv.setText(R.string.none)
+ binding.defaultSpTv.setText(R.string.none)
return@uiThread
}
- sharedPreferenceContainer.removeAllViews()
+ binding.sharedPreferenceContainer.removeAllViews()
val params =
LinearLayout.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT)
files.forEach {
@@ -96,13 +98,13 @@ class AppDataInfoView : FrameLayout {
textView.setTextColor(resources.getColor(R.color.item_content_text))
textView.setPadding(0, UiUtils.dp2px(context, 8F), 0, 0)
textView.tag = it
- sharedPreferenceContainer.addView(textView, params)
+ binding.sharedPreferenceContainer.addView(textView, params)
textView.setOnClickListener {
SharedPreferenceEditActivity.start(
context,
(it.tag as File).absolutePath,
packageName,
- apkInfo?.getAppName()?:""
+ apkInfo?.getAppName() ?: ""
)
}
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/AppSettingView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/AppSettingView.kt
index c41471a..aaeb840 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/AppSettingView.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/AppSettingView.kt
@@ -1,39 +1,32 @@
package com.wrbug.developerhelper.ui.widget.appsettingview
-import android.Manifest
import android.app.Activity
-import android.app.AlertDialog
import android.content.Context
-import android.content.DialogInterface
import android.util.AttributeSet
import android.view.LayoutInflater
import android.widget.ScrollView
+import androidx.appcompat.app.AlertDialog
+import androidx.fragment.app.FragmentActivity
+import com.wrbug.developerhelper.BuildConfig
import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.showToast
+import com.wrbug.developerhelper.base.requestStoragePermission
+import com.wrbug.developerhelper.base.showToast
import com.wrbug.developerhelper.commonutil.AppManagerUtils
import com.wrbug.developerhelper.commonutil.entity.ApkInfo
+import com.wrbug.developerhelper.util.setOnRootCheckClickListener
+import com.wrbug.developerhelper.util.visible
+import com.wrbug.developerhelper.databinding.DialogBackupAppSelectBinding
+import com.wrbug.developerhelper.databinding.ViewAppSettingBinding
import com.wrbug.developerhelper.mmkv.ConfigKv
import com.wrbug.developerhelper.mmkv.manager.MMKVManager
-import kotlinx.android.synthetic.main.view_app_setting.view.*
-import android.content.Intent
-import android.net.Uri
-import androidx.appcompat.widget.AppCompatButton
-import com.wrbug.developerhelper.basecommon.BaseActivity
-import com.wrbug.developerhelper.commonutil.shell.ShellManager
-import com.wrbug.developerhelper.commonutil.zip
-import com.wrbug.developerhelper.util.BackupUtils
-import com.wrbug.developerhelper.commonutil.toUri
-import gdut.bsx.share2.Share2
-import gdut.bsx.share2.ShareContentType
-import org.jetbrains.anko.doAsync
-import org.jetbrains.anko.uiThread
-import java.io.File
-
+import io.reactivex.rxjava3.disposables.CompositeDisposable
class AppSettingView : ScrollView {
- var apkInfo: ApkInfo? = null
+
+ private var apkInfo: ApkInfo? = null
private val configKv = MMKVManager.get(ConfigKv::class.java)
- private var exportDexBtn: AppCompatButton? = null
+ private lateinit var binding: ViewAppSettingBinding
+ private lateinit var disposable: CompositeDisposable
constructor(context: Context) : super(context) {
initView()
@@ -43,241 +36,137 @@ class AppSettingView : ScrollView {
initView()
}
- private fun initView() {
- LayoutInflater.from(context).inflate(R.layout.view_app_setting, this)
- exportDexBtn = findViewById(R.id.exportDexBtn)
- if (configKv.isOpenRoot().not()) {
- backupApkBtn.isEnabled = false
- backupApkDataDirBtn.isEnabled = false
- restartAppBtn.isEnabled = false
- stopAppBtn.isEnabled = false
- deleteAppDataBtn.isEnabled = false
- exportDexBtn?.isEnabled = false
+ fun setApkInfo(apkInfo: ApkInfo?) {
+ this.apkInfo = apkInfo
+ if (apkInfo?.applicationInfo?.packageName == BuildConfig.APPLICATION_ID) {
+ binding.uninstallAppBtn.visible = false
}
+ }
+
+ private fun initView() {
+ binding = ViewAppSettingBinding.inflate(LayoutInflater.from(context), this, true)
initListener()
}
private fun initListener() {
- backupApkBtn.setOnClickListener {
- doBackupApk()
+ binding.backupAppBtn.setOnRootCheckClickListener {
+ showBackupSelect()
}
- backupApkDataDirBtn.setOnClickListener {
- doBackupDataDir()
+ binding.restoreAppBtn.setOnRootCheckClickListener {
}
- restartAppBtn.setOnClickListener {
+ binding.restartAppBtn.setOnRootCheckClickListener {
doRestartApp()
}
- stopAppBtn.setOnClickListener {
+ binding.stopAppBtn.setOnRootCheckClickListener {
doStopApp()
}
- deleteAppDataBtn.setOnClickListener {
+ binding.deleteAppDataBtn.setOnRootCheckClickListener {
doDeleteAppData()
}
- uninstallAppBtn.setOnClickListener {
+ binding.uninstallAppBtn.setOnRootCheckClickListener {
doUninstallApp()
}
-
- exportDexBtn?.setOnClickListener {
- doBackupDexData()
- }
}
- private fun doBackupDexData() {
- apkInfo?.apply {
- showToast(context.getString(R.string.packing_files))
- doAsync {
- val dir = File(context.externalCacheDir, "dex/${applicationInfo.packageName}")
- if (dir.exists()) {
- ShellManager.rmFile(dir.absolutePath)
- }
- dir.mkdirs()
- val dexDir = "/data/data/${applicationInfo.packageName}/dump"
- val lsDir = ShellManager.lsDir(dexDir)
- if (lsDir.isEmpty()) {
- uiThread {
- showToast(context.getString(R.string.no_dex_files))
- }
- return@doAsync
- }
- if (ShellManager.cpFile(dexDir, dir.absolutePath)) {
- val zipFile =
- File(context.externalCacheDir, "${apkInfo?.getAppName() ?: ""}-dex.zip")
- dir.zip(zipFile)
- val uri = zipFile.toUri(context)
- if (uri == null) {
- showToast(R.string.export_failed)
- return@doAsync
- }
- uiThread {
- showShareDexNotice(uri)
- }
- } else {
- uiThread {
- showToast(context.getString(R.string.export_failed))
- }
- }
+ private fun showBackupSelect() {
+ context.requestStoragePermission {
+ val selected = booleanArrayOf(false, false, false)
+ val binding = DialogBackupAppSelectBinding.inflate(LayoutInflater.from(context))
+ binding.cbApk.setOnCheckedChangeListener { _, isChecked ->
+ selected[0] = isChecked
+ }
+ binding.cbData.setOnCheckedChangeListener { _, isChecked ->
+ selected[1] = isChecked
}
+ binding.cbAndroidData.setOnCheckedChangeListener { _, isChecked ->
+ selected[2] = isChecked
+ }
+ AlertDialog.Builder(context)
+ .setTitle(R.string.backup_app_file)
+ .setView(binding.root)
+ .setNegativeButton(R.string.cancel, null)
+ .setPositiveButton(
+ R.string.ok
+ ) { _, _ ->
+ doBackup(selected)
+ }.create().show()
}
}
- private fun showShareDexNotice(uri: Uri) {
- Share2.Builder(context as Activity)
- .setContentType(ShareContentType.FILE)
- .setShareFileUri(uri)
- .setOnActivityResult(10)
- .build()
- .shareBySystem()
+ override fun onAttachedToWindow() {
+ super.onAttachedToWindow()
+ disposable = CompositeDisposable()
}
- private fun doUninstallApp() {
- apkInfo?.apply {
- AppManagerUtils.uninstallApp(context, applicationInfo.packageName)
- }
+ override fun onDetachedFromWindow() {
+ super.onDetachedFromWindow()
+ disposable.dispose()
}
- private fun doDeleteAppData() {
- if (checkRoot().not()) {
+ private fun doBackup(selected: BooleanArray) {
+ val activity = context as? FragmentActivity ?: return
+ if (selected.find { it } == null) {
return
}
- apkInfo?.apply {
- showNotice(
- context.getString(R.string.confirm_delete_app_data),
- DialogInterface.OnClickListener { _, _ ->
- if (AppManagerUtils.clearAppData(applicationInfo.packageName)) {
- activityFinish()
- showToast(context.getString(R.string.clear_complete))
- }
- })
-
- }
+ BackupAppDialog.show(
+ activity.supportFragmentManager,
+ apkInfo,
+ selected[0],
+ selected[1],
+ selected[2]
+ )
}
- private fun doStopApp() {
- if (checkRoot().not()) {
- return
- }
+ private fun doUninstallApp() {
apkInfo?.apply {
- showNotice(
- context.getString(R.string.confirm_stop_app),
- DialogInterface.OnClickListener { _, _ ->
- if (AppManagerUtils.forceStopApp(applicationInfo.packageName)) {
- activityFinish()
- }
- })
+ AppManagerUtils.uninstallApp(context, applicationInfo.packageName)
}
}
- private fun doRestartApp() {
+ private fun doDeleteAppData() {
if (checkRoot().not()) {
return
}
apkInfo?.apply {
showNotice(
- context.getString(R.string.confirm_restart_app),
- DialogInterface.OnClickListener { _, _ ->
- AppManagerUtils.restartApp(context, applicationInfo.packageName)
+ context.getString(R.string.confirm_delete_app_data)
+ ) {
+ if (AppManagerUtils.clearAppData(applicationInfo.packageName)) {
activityFinish()
- })
+ showToast(context.getString(R.string.clear_complete))
+ }
+ }
+
}
}
- private fun doBackupDataDir() {
+ private fun doStopApp() {
if (checkRoot().not()) {
return
}
apkInfo?.apply {
- val backupAppData =
- BackupUtils.backupAppData(applicationInfo.packageName, applicationInfo.dataDir)
- if (backupAppData == null) {
- showToast(context.getString(R.string.backup_failed))
- return
- }
- if (context !is BaseActivity) {
- showToast(context.getString(R.string.backup_success_msg))
- return
+ showNotice(context.getString(R.string.confirm_stop_app)) {
+ if (AppManagerUtils.forceStopApp(applicationInfo.packageName)) {
+ activityFinish()
+ }
}
- showShareDataNotice(backupAppData)
}
}
- private fun showShareDataNotice(backupAppData: File) {
- showNotice(
- context.getString(R.string.backup_success_and_share_msg),
- DialogInterface.OnClickListener { _, _ ->
- (context as BaseActivity).requestPermission(arrayOf(Manifest.permission.READ_EXTERNAL_STORAGE),
- object : BaseActivity.PermissionCallback() {
- override fun granted() {
- val zipFile = File(
- context.externalCacheDir,
- "${apkInfo?.getAppName() ?: ""}-data.zip"
- )
- backupAppData.zip(zipFile)
- val uri = zipFile.toUri(context)
- if (uri == null) {
- showToast(context.getString(R.string.share_failed))
- return
- }
- activityFinish()
- Share2.Builder(context as Activity)
- .setContentType(ShareContentType.FILE)
- .setShareFileUri(uri)
- .setOnActivityResult(10)
- .build()
- .shareBySystem()
- }
-
- })
-
- })
- }
-
- private fun doBackupApk() {
+ private fun doRestartApp() {
if (checkRoot().not()) {
return
}
apkInfo?.apply {
- val uri = BackupUtils.backupApk(
- applicationInfo.packageName,
- applicationInfo.publicSourceDir,
- "${getAppName()}_${packageInfo.versionName}.apk"
- )
- if (uri == null) {
- showToast(context.getString(R.string.backup_failed))
- return
- }
- if (context !is BaseActivity) {
- showToast(context.getString(R.string.backup_success_msg))
- return
+ showNotice(
+ context.getString(R.string.confirm_restart_app)
+ ) {
+ AppManagerUtils.restartApp(context, applicationInfo.packageName)
+ activityFinish()
}
- showShareApkDialog(uri)
-
}
}
- private fun showShareApkDialog(uri: Uri) {
- showNotice(
- context.getString(R.string.backup_success_and_share_msg),
- DialogInterface.OnClickListener { _, _ ->
- (context as BaseActivity).requestPermission(arrayOf(
- Manifest.permission.READ_EXTERNAL_STORAGE,
- Manifest.permission.WRITE_EXTERNAL_STORAGE
- ),
- object : BaseActivity.PermissionCallback() {
- override fun granted() {
- activityFinish()
- Share2.Builder(context as Activity)
- .setContentType(ShareContentType.FILE)
- .setShareFileUri(uri)
- .setOnActivityResult(10)
- .build()
- .shareBySystem()
- }
-
- })
-
- })
- }
-
private fun checkRoot(): Boolean {
if (configKv.isOpenRoot().not()) {
showToast(context.getString(R.string.please_open_root))
@@ -286,18 +175,17 @@ class AppSettingView : ScrollView {
return true
}
- fun activityFinish() {
+ private fun activityFinish() {
if (context is Activity) {
(context as Activity).finish()
}
}
- private fun showNotice(msg: String, listener: DialogInterface.OnClickListener) {
- AlertDialog.Builder(context)
- .setTitle(R.string.notice)
- .setMessage(msg)
- .setNegativeButton(R.string.cancel, null)
- .setPositiveButton(R.string.ok, listener)
- .create().show()
+ private fun showNotice(msg: String, listener: () -> Unit) {
+ AlertDialog.Builder(context).setTitle(R.string.notice).setMessage(msg)
+ .setNegativeButton(R.string.cancel, null).setPositiveButton(
+ R.string.ok
+ ) { _, _ -> listener() }.create().show()
}
+
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/BackupAppDialog.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/BackupAppDialog.kt
new file mode 100644
index 0000000..2173665
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/appsettingview/BackupAppDialog.kt
@@ -0,0 +1,270 @@
+package com.wrbug.developerhelper.ui.widget.appsettingview
+
+import android.app.Dialog
+import android.os.Bundle
+import android.view.LayoutInflater
+import android.view.View
+import android.view.ViewGroup
+import androidx.core.content.ContextCompat
+import androidx.core.view.isVisible
+import androidx.fragment.app.FragmentManager
+import com.google.android.material.bottomsheet.BottomSheetDialogFragment
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.ExtraKey
+import com.wrbug.developerhelper.base.versionCodeLong
+import com.wrbug.developerhelper.commonutil.addTo
+import com.wrbug.developerhelper.commonutil.entity.ApkInfo
+import com.wrbug.developerhelper.commonutil.observeOnMain
+import com.wrbug.developerhelper.commonutil.shell.ShellManager
+import com.wrbug.developerhelper.util.setOnDoubleCheckClickListener
+import com.wrbug.developerhelper.databinding.DialogBackupAppBinding
+import com.wrbug.developerhelper.model.entity.BackupAppItemInfo
+import com.wrbug.developerhelper.ui.widget.backupprogress.BackupProgressView
+import com.wrbug.developerhelper.util.BackupUtils
+import com.wrbug.developerhelper.util.format
+import io.reactivex.rxjava3.core.Single
+import io.reactivex.rxjava3.disposables.CompositeDisposable
+import io.reactivex.rxjava3.schedulers.Schedulers
+
+
+class BackupAppDialog : BottomSheetDialogFragment() {
+ companion object {
+ fun show(
+ fragmentManager: FragmentManager,
+ apkInfo: ApkInfo?,
+ backupApk: Boolean,
+ backupData: Boolean,
+ backupAndroidData: Boolean
+ ) {
+ BackupAppDialog().apply {
+ arguments = Bundle().apply {
+ putParcelable(ExtraKey.DATA, apkInfo)
+ putBoolean(ExtraKey.KEY_1, backupApk)
+ putBoolean(ExtraKey.KEY_2, backupData)
+ putBoolean(ExtraKey.KEY_3, backupAndroidData)
+ }
+ }.show(fragmentManager, "BackupAppDialog")
+ }
+ }
+
+ private lateinit var binding: DialogBackupAppBinding
+ private lateinit var disposable: CompositeDisposable
+ private lateinit var dateDir: String
+ private var backupTimeStamp: Long = 0
+
+ private val apkInfo: ApkInfo? by lazy {
+ arguments?.getParcelable(ExtraKey.DATA)
+ }
+
+ private val backupApk by lazy {
+ arguments?.getBoolean(ExtraKey.KEY_1) ?: false
+ }
+
+
+ private val backupData by lazy {
+ arguments?.getBoolean(ExtraKey.KEY_2) ?: false
+ }
+
+
+ private val backupAndroidData by lazy {
+ arguments?.getBoolean(ExtraKey.KEY_3) ?: false
+ }
+
+ private var successCount = 0
+ set(value) {
+ field = value
+ checkIsAllSuccess()
+ }
+ private var hasError = false
+ set(value) {
+ field = value
+ checkIsAllSuccess()
+ }
+
+ private fun checkIsAllSuccess() {
+ if (hasError) {
+ binding.tvNotice.isVisible = false
+ binding.btnExit.isVisible = true
+ binding.zipFileProgress.setStatus(BackupProgressView.Status.Canceled)
+ return
+ }
+ if (successCount < 3) { + return + } + createInfoFile() + } + + private fun createInfoFile() { + binding.zipFileProgress.setStatus(BackupProgressView.Status.Processing) + runOnIo().map { apkInfo ->
+ BackupUtils.zipBackupFile(apkInfo.packageInfo.packageName.orEmpty(), dateDir)
+ ?.let { apkInfo to it }
+ ?: throw Exception()
+ }.map {
+ val info = BackupAppItemInfo(
+ it.second.name,
+ backupApk,
+ backupData,
+ backupAndroidData,
+ it.first.generateBackupApkFileName(),
+ it.first.packageInfo.versionName.orEmpty(),
+ it.first.packageInfo.versionCodeLong,
+ it.first.packageInfo.packageName.orEmpty(),
+ backupTimeStamp
+ )
+ val success =
+ BackupUtils.saveBackupInfo(it.first, info, it.second.name)
+ if (success) {
+ return@map it.second.absolutePath
+ }
+ throw Exception()
+ }.observeOnMain().subscribe({
+ binding.zipFileProgress.setStatus(BackupProgressView.Status.Success, it)
+ showSuccess()
+ }, {
+ hasError = true
+ binding.zipFileProgress.setStatus(BackupProgressView.Status.Failed)
+ }).addTo(disposable)
+ }
+
+ private fun showSuccess() {
+ binding.btnExit.isVisible = true
+ binding.tvNotice.text = getString(
+ R.string.backup_success_notice,
+ BackupUtils.getAppBackupDir(
+ apkInfo?.packageInfo?.packageName.orEmpty()
+ ).absolutePath
+ )
+ binding.tvNotice.setTextColor(
+ ContextCompat.getColor(
+ requireContext(),
+ R.color.material_color_green_600
+ )
+ )
+ }
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ successCount = 0
+ backupTimeStamp = System.currentTimeMillis()
+ dateDir = backupTimeStamp.format("yyyy-MM-dd-HH_mm_ss")
+ }
+
+ override fun onCreateView(
+ inflater: LayoutInflater,
+ container: ViewGroup?,
+ savedInstanceState: Bundle?
+ ): View {
+ disposable = CompositeDisposable()
+ binding = DialogBackupAppBinding.inflate(inflater, container, false)
+ return binding.root
+ }
+
+ override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
+ return super.onCreateDialog(savedInstanceState).apply {
+ setCancelable(false)
+ setCanceledOnTouchOutside(false)
+ }
+ }
+
+ override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
+ binding.apkProgress.setTitle(R.string.item_backup_apk)
+ binding.androidDataProgress.setTitle(R.string.item_backup_android_data)
+ binding.dataProgress.setTitle(R.string.item_backup_data)
+ binding.zipFileProgress.setTitle(R.string.item_zip_backup_file)
+ binding.zipFileProgress.setStatus(BackupProgressView.Status.Waiting)
+ binding.btnExit.setOnDoubleCheckClickListener {
+ dismissAllowingStateLoss()
+ }
+ backupApk()
+ backupData()
+ backupAndroidData()
+ }
+
+ private fun backupApk() {
+ if (!backupApk) {
+ successCount++
+ binding.apkProgress.setStatus(BackupProgressView.Status.Ignore)
+ return
+ }
+ binding.apkProgress.setStatus(BackupProgressView.Status.Processing)
+ runOnIo().map {
+ BackupUtils.backupApk(
+ it.applicationInfo.packageName,
+ dateDir,
+ it.applicationInfo.publicSourceDir,
+ it.generateBackupApkFileName()
+ ) ?: throw Exception()
+ }.observeOnMain().subscribe({
+ successCount++
+ binding.apkProgress.setStatus(BackupProgressView.Status.Success, it)
+ }, {
+ hasError = true
+ binding.apkProgress.setStatus(BackupProgressView.Status.Failed)
+ }).addTo(disposable)
+ }
+
+ override fun onDestroyView() {
+ super.onDestroyView()
+ disposable.dispose()
+ }
+
+ override fun onDestroy() {
+ super.onDestroy()
+ if (hasError) {
+ ShellManager.rmFile(
+ BackupUtils.getCurrentAppBackupDir(
+ apkInfo?.packageInfo?.packageName.orEmpty(),
+ dateDir
+ ).absolutePath
+ )
+ }
+ }
+
+ private fun backupAndroidData() {
+ if (!backupAndroidData) {
+ successCount++
+ binding.androidDataProgress.setStatus(BackupProgressView.Status.Ignore)
+ return
+ }
+ binding.androidDataProgress.setStatus(BackupProgressView.Status.Processing)
+ runOnIo().map {
+ BackupUtils.backupAppAndroidData(dateDir, it.applicationInfo.packageName)
+ ?: throw Exception()
+ }.observeOnMain().subscribe({
+ successCount++
+ binding.androidDataProgress.setStatus(
+ BackupProgressView.Status.Success,
+ it.ifEmpty { getString(R.string.no_need_to_backup) }
+ )
+ }, {
+ hasError = true
+ binding.androidDataProgress.setStatus(BackupProgressView.Status.Failed)
+ }).addTo(disposable)
+ }
+
+ private fun backupData() {
+ if (!backupData) {
+ successCount++
+ binding.dataProgress.setStatus(BackupProgressView.Status.Ignore)
+ return
+ }
+ binding.dataProgress.setStatus(BackupProgressView.Status.Processing)
+ runOnIo().map {
+ BackupUtils.backupAppData(dateDir, it.applicationInfo.packageName) ?: throw Exception()
+ }.observeOnMain().subscribe({
+ successCount++
+ binding.dataProgress.setStatus(BackupProgressView.Status.Success, it.absolutePath)
+ }, {
+ hasError = true
+ binding.dataProgress.setStatus(BackupProgressView.Status.Failed)
+ }).addTo(disposable)
+ }
+
+
+ private fun runOnIo() = if (apkInfo == null) {
+ Single.error(Exception())
+ } else {
+ Single.just(apkInfo!!).subscribeOn(Schedulers.newThread())
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/backupprogress/BackupProgressView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/backupprogress/BackupProgressView.kt
new file mode 100644
index 0000000..698bf87
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/backupprogress/BackupProgressView.kt
@@ -0,0 +1,98 @@
+package com.wrbug.developerhelper.ui.widget.backupprogress
+
+import android.content.Context
+import android.content.res.ColorStateList
+import android.util.AttributeSet
+import android.view.LayoutInflater
+import androidx.constraintlayout.widget.ConstraintLayout
+import androidx.core.content.ContextCompat
+import androidx.core.view.isVisible
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.databinding.ViewBackupProgressBinding
+import com.wrbug.developerhelper.util.getString
+
+class BackupProgressView @JvmOverloads constructor(
+ context: Context, attrs: AttributeSet? = null
+) : ConstraintLayout(context, attrs) {
+ private val binding = ViewBackupProgressBinding.inflate(LayoutInflater.from(context), this)
+
+
+ fun setTitle(title: Int) {
+ binding.tvTitle.text = getString(title)
+ }
+
+ fun setStatus(status: Status, successPath: String = "") {
+ when (status) {
+ Status.Processing -> {
+ binding.progress.isVisible = true
+ binding.ivStatus.isVisible = false
+ binding.tvStatus.text = context.getString(R.string.backuping_data)
+ }
+
+ Status.Success -> {
+ binding.progress.isVisible = false
+ binding.ivStatus.isVisible = true
+ binding.ivStatus.setImageResource(R.drawable.ic_success)
+ binding.ivStatus.imageTintList = ColorStateList.valueOf(
+ ContextCompat.getColor(
+ context,
+ R.color.material_color_green_400
+ )
+ )
+ binding.tvStatus.text = successPath
+ }
+
+ Status.Failed -> {
+ binding.progress.isVisible = false
+ binding.ivStatus.isVisible = true
+ binding.tvStatus.text = context.getString(R.string.backup_fail)
+ binding.ivStatus.setImageResource(R.drawable.ic_fail)
+ binding.ivStatus.imageTintList = ColorStateList.valueOf(
+ ContextCompat.getColor(
+ context,
+ R.color.material_color_red_400
+ )
+ )
+
+ }
+
+ Status.Ignore -> {
+ binding.progress.isVisible = false
+ binding.ivStatus.isVisible = true
+ binding.tvStatus.text = context.getString(R.string.ignore_backup)
+ binding.ivStatus.setImageResource(R.drawable.ic_ignore)
+ binding.ivStatus.imageTintList = ColorStateList.valueOf(
+ ContextCompat.getColor(
+ context,
+ R.color.material_color_grey_400
+ )
+ )
+ }
+
+ Status.Waiting -> {
+ binding.progress.isVisible = true
+ binding.ivStatus.isVisible = true
+ binding.tvStatus.text = context.getString(R.string.backup_waiting)
+ binding.ivStatus.setImageResource(R.drawable.ic_waiting)
+ binding.ivStatus.imageTintList = ColorStateList.valueOf(
+ ContextCompat.getColor(context, R.color.material_color_grey_400)
+ )
+ }
+
+ Status.Canceled -> {
+ binding.progress.isVisible = false
+ binding.ivStatus.isVisible = true
+ binding.ivStatus.setImageResource(R.drawable.ic_canceled)
+ binding.tvStatus.text = context.getString(R.string.canceled)
+ binding.ivStatus.imageTintList = ColorStateList.valueOf(
+ ContextCompat.getColor(context, R.color.material_color_grey_400)
+ )
+ }
+ }
+ }
+
+
+ enum class Status {
+ Processing, Success, Failed, Ignore, Waiting, Canceled
+ }
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/BottomMenu.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/BottomMenu.kt
deleted file mode 100644
index fcbc77c..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/BottomMenu.kt
+++ /dev/null
@@ -1,93 +0,0 @@
-package com.wrbug.developerhelper.ui.widget.bottommenu
-
-import android.content.Context
-import android.graphics.drawable.ColorDrawable
-import android.os.Bundle
-import android.view.View
-import androidx.recyclerview.widget.DividerItemDecoration
-import androidx.recyclerview.widget.LinearLayoutManager
-import com.google.android.material.bottomsheet.BottomSheetDialog
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.commonutil.UiUtils
-import com.wrbug.developerhelper.ui.decoration.SpaceItemDecoration
-import kotlinx.android.synthetic.main.dialog_bottom_menu.*
-
-class BottomMenu(context: Context) : BottomSheetDialog(context), OnItemClickListener {
-
- private var title = ""
- val adapter: BottomMenuItemAdapter by lazy {
- BottomMenuItemAdapter(context)
- }
-
- private var items: Array = arrayOf()
- private var listener: OnItemClickListener? = null
-
- init {
- setContentView(R.layout.dialog_bottom_menu)
- }
-
- override fun onCreate(savedInstanceState: Bundle?) {
- super.onCreate(savedInstanceState)
- initRv()
- initView()
- }
-
- private fun initView() {
- titleTv.text = title
- titleTv.visibility = if (title.isEmpty()) View.GONE else View.VISIBLE
- }
-
- private fun initRv() {
- menuListRv.layoutManager = LinearLayoutManager(context)
- val decoration = SpaceItemDecoration(UiUtils.dp2px(context, 0F))
- decoration.setLastBottomPadding(UiUtils.dp2px(context, 10F))
- decoration.setFirstTopPadding(UiUtils.dp2px(context, 10F))
- menuListRv.addItemDecoration(decoration)
- val dividerItemDecoration = DividerItemDecoration(context, DividerItemDecoration.VERTICAL)
- dividerItemDecoration.setDrawable(ColorDrawable(context.resources.getColor(R.color.divider_color)))
- menuListRv.addItemDecoration(dividerItemDecoration)
- adapter.setOnItemClickListener(this)
- menuListRv.adapter = adapter
- adapter.list = items
- }
-
- override fun onClick(position: Int) {
- listener?.onClick(position)
- dismiss()
- }
-
- class Builder(val context: Context) {
- private var title = ""
- private var items: Array = arrayOf()
- private var listener: OnItemClickListener? = null
-
- fun title(id: Int): Builder {
- title = context.getString(id)
- return this
- }
-
- fun title(title: String): Builder {
- this.title = title
- return this
- }
-
- fun menuItems(items: Array): Builder {
- this.items = items
- return this
- }
-
- fun onItemClickListener(listener: OnItemClickListener?): Builder {
- this.listener = listener
- return this
- }
-
- fun build(): BottomMenu {
- val menu = BottomMenu(context)
- menu.items = items
- menu.title = title
- menu.listener = listener
- return menu
- }
- }
-
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/BottomMenuItemAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/BottomMenuItemAdapter.kt
deleted file mode 100644
index a8e5b9c..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/BottomMenuItemAdapter.kt
+++ /dev/null
@@ -1,51 +0,0 @@
-package com.wrbug.developerhelper.ui.widget.bottommenu
-
-import android.content.Context
-import android.view.Gravity
-import android.view.View
-import android.view.ViewGroup
-import android.widget.TextView
-import androidx.recyclerview.widget.RecyclerView
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.commonutil.UiUtils
-
-class BottomMenuItemAdapter(val context: Context) : RecyclerView.Adapter() {
- var list: Array = arrayOf()
- set(value) {
- field = value
- notifyDataSetChanged()
- }
- var listener: OnItemClickListener? = null
- override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder {
- val tv = TextView(context)
- tv.setTextColor(context.resources.getColor(R.color.text_color_666666))
- tv.gravity = Gravity.CENTER
- val dp10 = UiUtils.dp2px(context, 10F)
- tv.setPadding(dp10, dp10, dp10, dp10)
- val params = ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)
- tv.layoutParams = params
- return ViewHolder(tv)
- }
-
- override fun getItemCount(): Int = list.size
-
- override fun onBindViewHolder(holder: ViewHolder, position: Int) {
- holder.tv.text = list[position]
- holder.index = position
- }
-
- fun setOnItemClickListener(listener: OnItemClickListener) {
- this.listener = listener
- }
-
- inner class ViewHolder(val tv: TextView) : RecyclerView.ViewHolder(tv) {
- var index = 0
-
- init {
- tv.setOnClickListener {
- listener?.onClick(index)
- }
- }
-
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/OnItemClickListener.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/OnItemClickListener.kt
deleted file mode 100644
index 2a4f90b..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/bottommenu/OnItemClickListener.kt
+++ /dev/null
@@ -1,5 +0,0 @@
-package com.wrbug.developerhelper.ui.widget.bottommenu
-
-interface OnItemClickListener {
- fun onClick(position: Int)
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/boundsinfoview/BoundsInfoView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/boundsinfoview/BoundsInfoView.kt
index 3787c96..828526c 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/boundsinfoview/BoundsInfoView.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/boundsinfoview/BoundsInfoView.kt
@@ -41,7 +41,7 @@ class BoundsInfoView : View {
unit = if (unit == Unit.DP) Unit.PX else Unit.DP
}
- override fun onDraw(canvas: Canvas?) {
+ override fun onDraw(canvas: Canvas) {
val rect = drawRect(canvas)
val lineRect = drawEdge(rect, canvas)
drawAl(rect, lineRect, canvas)
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/emptyview/EmptyView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/emptyview/EmptyView.kt
index e5c3504..92af967 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/emptyview/EmptyView.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/emptyview/EmptyView.kt
@@ -3,48 +3,21 @@ package com.wrbug.developerhelper.ui.widget.emptyview
import android.content.Context
import android.util.AttributeSet
import android.view.Gravity
-import android.view.View
+import android.view.LayoutInflater
import android.widget.ImageView
import android.widget.LinearLayout
import android.widget.TextView
+import androidx.constraintlayout.widget.ConstraintLayout
import com.wrbug.developerhelper.R
import com.wrbug.developerhelper.commonutil.UiUtils
+import com.wrbug.developerhelper.databinding.ViewEmptyViewBinding
-class EmptyView : LinearLayout {
- val icoIv: ImageView by lazy {
- val icoIv = ImageView(context)
- icoIv.setImageResource(R.drawable.ic_ic_empty_666666)
- icoIv
- }
- val textTv: TextView by lazy {
- val tv = TextView(context)
- tv.setTextColor(resources.getColor(R.color.text_color_666666))
- tv.text = "无数据"
- tv.textSize = 20F
- tv.gravity = Gravity.CENTER
- val params = LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT)
- params.topMargin = UiUtils.dp2px(context, 20F)
- tv.layoutParams = params
- tv
- }
-
- constructor(context: Context) : super(context) {
- }
-
- constructor(context: Context, attributeSet: AttributeSet?) : super(context, attributeSet) {
- init()
- initView()
- }
-
- private fun init() {
- orientation = VERTICAL
- gravity = Gravity.CENTER
- }
+class EmptyView @JvmOverloads constructor(
+ context: Context, attrs: AttributeSet? = null
+) : ConstraintLayout(context, attrs) {
+ private val binding = ViewEmptyViewBinding.inflate(LayoutInflater.from(context), this)
- private fun initView() {
- addView(icoIv)
- addView(textTv)
+ fun setTitle(title: String) {
+ binding.tvTitle.text = title
}
-
-
}
\ No newline at end of file
diff --git a/commonwidget/src/main/java/com/wrbug/developerhelper/commonwidget/flexibletoast/FlexibleToast.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/flexibletoast/FlexibleToast.kt
similarity index 73%
rename from commonwidget/src/main/java/com/wrbug/developerhelper/commonwidget/flexibletoast/FlexibleToast.kt
rename to app/src/main/java/com/wrbug/developerhelper/ui/widget/flexibletoast/FlexibleToast.kt
index d1deff6..ddd5aa0 100644
--- a/commonwidget/src/main/java/com/wrbug/developerhelper/commonwidget/flexibletoast/FlexibleToast.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/flexibletoast/FlexibleToast.kt
@@ -1,4 +1,4 @@
-package com.wrbug.developerhelper.commonwidget.flexibletoast
+package com.wrbug.developerhelper.ui.widget.flexibletoast
import android.content.Context
import android.os.Handler
@@ -10,7 +10,7 @@ import android.widget.ImageView
import android.widget.TextView
import android.widget.Toast
import com.wrbug.developerhelper.commonutil.UiUtils
-import com.wrbug.developerhelper.commonwidget.R
+import com.wrbug.developerhelper.databinding.LayoutToastFlexibleBinding
class FlexibleToast private constructor(private val mContext: Context) {
@@ -52,11 +52,13 @@ class FlexibleToast private constructor(private val mContext: Context) {
0,
0
)
+
builder.mGravity == GRAVITY_TOP -> flexibleToast.setGravity(
Gravity.TOP or Gravity.CENTER_VERTICAL,
0,
UiUtils.dp2px(mContext, 20F)
)
+
else -> flexibleToast.setGravity(
Gravity.BOTTOM or Gravity.CENTER_VERTICAL,
0,
@@ -71,7 +73,7 @@ class FlexibleToast private constructor(private val mContext: Context) {
if (builder.hasCustomerView && builder.mCustomerView != null) {
flexibleToast.view = builder.mCustomerView
} else {
- flexibleToast.view = builder.mDefaultView
+ flexibleToast.view = builder.binding.root
}
flexibleToast.show()
}
@@ -80,46 +82,31 @@ class FlexibleToast private constructor(private val mContext: Context) {
* 控制Toast的显示样式
*/
class Builder(context: Context) {
- val mDefaultView: View = LayoutInflater.from(context)
- .inflate(R.layout.layout_toast_flexible, null)
+ val binding = LayoutToastFlexibleBinding.inflate(LayoutInflater.from(context))
var mCustomerView: View? = null
- private val mIvImage: ImageView
- private val mTvFirst: TextView
- private val mTvSecond: TextView
-
- private val dividerFirst: View
- private val dividerSecond: View
var mDuration = Toast.LENGTH_SHORT// 0 short, 1 long
var mGravity = 0
var hasCustomerView = false // 是否使用自定义layout
- init {
- mIvImage = mDefaultView.findViewById(R.id.imgIv)
- mTvFirst = mDefaultView.findViewById(R.id.firstTv)
- mTvSecond = mDefaultView.findViewById(R.id.secondTv)
- dividerFirst = mDefaultView.findViewById(R.id.firstDividerView)
- dividerSecond = mDefaultView.findViewById(R.id.secondDividerView)
- }
-
fun setImageResource(resId: Int): Builder {
- this.mIvImage.setImageResource(resId)
- this.mIvImage.visibility = View.VISIBLE
- this.dividerFirst.visibility = View.VISIBLE
+ binding.imgIv.setImageResource(resId)
+ binding.imgIv.visibility = View.VISIBLE
+ binding.firstDividerView.visibility = View.VISIBLE
return this
}
fun setFirstText(firstText: String): Builder {
- this.mTvFirst.text = firstText
- this.mTvFirst.visibility = View.VISIBLE
- this.dividerSecond.visibility = View.VISIBLE
+ binding.firstTv.text = firstText
+ binding.firstTv.visibility = View.VISIBLE
+ binding.secondDividerView.visibility = View.VISIBLE
return this
}
fun setSecondText(secondText: String): Builder {
- this.mTvSecond.text = secondText
- this.mTvSecond.visibility = View.VISIBLE
+ binding.secondTv.text = secondText
+ binding.secondTv.visibility = View.VISIBLE
return this
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyDetailView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyDetailView.kt
index 9b869db..fb54b6e 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyDetailView.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyDetailView.kt
@@ -6,15 +6,16 @@ import android.graphics.Color
import android.graphics.Paint
import android.util.AttributeSet
import android.widget.FrameLayout
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
+import com.wrbug.developerhelper.base.entry.HierarchyNode
import com.wrbug.developerhelper.R
import com.wrbug.developerhelper.ui.widget.helper.CanvasHelper
import com.wrbug.developerhelper.commonutil.UiUtils
class HierarchyDetailView : FrameLayout {
+
private val paint: Paint by lazy {
val paint = Paint()
- paint.color = context.resources.getColor(R.color.colorAccent)
+ paint.color = context.resources.getColor(R.color.material_color_blue_800)
paint.style = Paint.Style.STROKE
paint.strokeWidth = 3F
paint
@@ -34,7 +35,6 @@ class HierarchyDetailView : FrameLayout {
initView()
}
-
constructor(context: Context, attrs: AttributeSet) : super(context, attrs) {
initView()
}
@@ -50,14 +50,13 @@ class HierarchyDetailView : FrameLayout {
setWillNotDraw(false)
}
-
- override fun onDraw(canvas: Canvas?) {
+ override fun onDraw(canvas: Canvas) {
super.onDraw(canvas)
if (hierarchyNode != null) {
- drawNode(canvas)
if (hierarchyNode?.parentId!!> -1) {
drawParentNode(canvas)
}
+ drawNode(canvas)
}
}
@@ -65,7 +64,7 @@ class HierarchyDetailView : FrameLayout {
if (parentHierarchyNode == null) {
return
}
- val bounds = parentHierarchyNode?.screenBounds
+ val bounds = parentHierarchyNode?.screenBounds ?: return
canvas?.drawRect(bounds, parentpPaint)
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyView.kt
index 5ffd740..93d95f5 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyView.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/HierarchyView.kt
@@ -7,8 +7,9 @@ import android.graphics.Paint
import android.util.AttributeSet
import android.view.MotionEvent
import android.view.View
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
+import com.wrbug.developerhelper.base.entry.HierarchyNode
import com.wrbug.developerhelper.service.DeveloperHelperAccessibilityService
+import kotlin.math.pow
class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attrs) {
private val strokePaint = Paint()
@@ -18,6 +19,7 @@ class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attr
private var onHierarchyNodeClickListener: OnHierarchyNodeClickListener? = null
private var selectedNode: HierarchyNode? = null
private var selectedParentNode: HierarchyNode? = null
+ private var isFromLeftDown = false
constructor(context: Context) : this(context, null)
@@ -36,7 +38,7 @@ class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attr
mHierarchyNodes.addAll(it)
}
nodeMap.putAll(DeveloperHelperAccessibilityService.nodeMap)
- visibility=View.VISIBLE
+ visibility = View.VISIBLE
invalidate()
}
@@ -47,15 +49,25 @@ class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attr
override fun onTouchEvent(event: MotionEvent?): Boolean {
when (event?.action) {
MotionEvent.ACTION_DOWN -> {
+ isFromLeftDown = event.x <= 15 + if (isFromLeftDown) { + return super.onTouchEvent(event) + } getSelectedNode(event.x, event.y) return true } + MotionEvent.ACTION_MOVE -> {
- getSelectedNode(event.x, event.y)
+ if (!isFromLeftDown) {
+ getSelectedNode(event.x, event.y)
+ }
}
+
MotionEvent.ACTION_UP -> {
- selectedNode?.let {
- onHierarchyNodeClickListener?.onClick(it, selectedParentNode)
+ if (!isFromLeftDown) {
+ selectedNode?.let {
+ onHierarchyNodeClickListener?.onClick(it, selectedParentNode)
+ }
}
}
}
@@ -70,15 +82,17 @@ class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attr
}
selectedNode = hierarchyNode.selectedNode
selectedParentNode = hierarchyNode.parentNode
- onHierarchyNodeClickListener?.onSelectedNodeChanged(hierarchyNode.selectedNode, hierarchyNode.parentNode)
+ onHierarchyNodeClickListener?.onSelectedNodeChanged(
+ hierarchyNode.selectedNode, hierarchyNode.parentNode
+ )
}
}
- override fun onDraw(canvas: Canvas?) {
+ override fun onDraw(canvas: Canvas) {
drawRect(canvas, mHierarchyNodes)
}
- private fun drawRect(canvas: Canvas?, hierarchyNodes: List) {
+ private fun drawRect(canvas: Canvas, hierarchyNodes: List) {
for (hierarchyNode in hierarchyNodes) {
drawWidget(canvas, hierarchyNode)
drawRect(canvas, hierarchyNode.childId)
@@ -108,10 +122,10 @@ class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attr
} else {
var left = info.selectedNode.screenBounds?.left ?: 0
var top = info.selectedNode.screenBounds?.top ?: 0
- val len = Math.pow(x.toDouble() - left, 2.0) + Math.pow(y.toDouble() - top, 2.0)
+ val len = (x.toDouble() - left).pow(2.0) + (y.toDouble() - top).pow(2.0)
left = list[index].selectedNode.screenBounds?.left ?: 0
top = list[index].selectedNode.screenBounds?.top ?: 0
- val len1 = Math.pow(x.toDouble() - left, 2.0) + Math.pow(y.toDouble() - top, 2.0)
+ val len1 = (x.toDouble() - left).pow(2.0) + (y.toDouble() - top).pow(2.0)
if (len1 < len) { info = list[index] } @@ -120,10 +134,12 @@ class HierarchyView(context: Context, attrs: AttributeSet?) : View(context, attr return info } - private fun getNode(x: Float, y: Float, hierarchyNode: HierarchyNode, list: ArrayList) {
+ private fun getNode(
+ x: Float, y: Float, hierarchyNode: HierarchyNode, list: ArrayList
+ ) {
val rect = hierarchyNode.screenBounds ?: return
if (rect.contains(x.toInt(), y.toInt())) {
- if (!hierarchyNode.childId.isEmpty()) {
+ if (hierarchyNode.childId.isNotEmpty()) {
for (child in hierarchyNode.childId.reversed()) {
getNode(x, y, child, list)
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/SelectedNodeInfo.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/SelectedNodeInfo.kt
index b3a16b8..c960bfd 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/SelectedNodeInfo.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/hierarchyView/SelectedNodeInfo.kt
@@ -1,6 +1,6 @@
package com.wrbug.developerhelper.ui.widget.hierarchyView
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
+import com.wrbug.developerhelper.base.entry.HierarchyNode
class SelectedNodeInfo(var selectedNode: HierarchyNode, var parentNode: HierarchyNode?) {
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoDialog.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoDialog.kt
new file mode 100644
index 0000000..96bdc1f
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoDialog.kt
@@ -0,0 +1,100 @@
+package com.wrbug.developerhelper.ui.widget.layoutinfoview
+
+import android.app.Dialog
+import android.os.Bundle
+import android.view.Gravity
+import android.view.LayoutInflater
+import android.view.View
+import android.view.ViewGroup
+import androidx.fragment.app.DialogFragment
+import androidx.fragment.app.FragmentManager
+import com.google.android.material.bottomsheet.BottomSheetDialogFragment
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.base.ExtraKey
+import com.wrbug.developerhelper.base.entry.HierarchyNode
+import com.wrbug.developerhelper.commonutil.UiUtils
+import com.wrbug.developerhelper.commonutil.dp2px
+import com.wrbug.developerhelper.databinding.ViewLayoutInfoBinding
+import com.wrbug.developerhelper.util.isPortrait
+
+class LayoutInfoDialog : DialogFragment() {
+
+ companion object {
+ fun show(
+ fragmentManager: FragmentManager,
+ list: List?,
+ selectedNode: HierarchyNode,
+ listener: ((HierarchyNode, HierarchyNode?) -> Unit)
+ ) {
+ LayoutInfoDialog().apply {
+ arguments = Bundle().apply {
+ putParcelableArrayList(ExtraKey.DATA, list?.let { ArrayList(it) })
+ putParcelable(ExtraKey.SELECTED, selectedNode)
+ }
+ onNodeChangedListener = listener
+ }.show(fragmentManager, "LayoutInfoDialog")
+ }
+ }
+
+ private val nodeList: List? by lazy {
+ arguments?.getParcelableArrayList(ExtraKey.DATA)
+ }
+ private val hierarchyNode: HierarchyNode? by lazy {
+ arguments?.getParcelable(ExtraKey.SELECTED)
+ }
+ private var onNodeChangedListener: ((HierarchyNode, HierarchyNode?) -> Unit)? = null
+
+ val adapter by lazy {
+ LayoutInfoViewPagerAdapter(requireContext(), nodeList, hierarchyNode!!)
+ }
+
+ private lateinit var binding: ViewLayoutInfoBinding
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ setStyle(STYLE_NORMAL, R.style.FullScreenDialog)
+ }
+
+ override fun onCreateView(
+ inflater: LayoutInflater,
+ container: ViewGroup?,
+ savedInstanceState: Bundle?
+ ): View {
+ binding = ViewLayoutInfoBinding.inflate(layoutInflater)
+ return binding.root
+ }
+
+ override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
+ return super.onCreateDialog(savedInstanceState).apply {
+ setCanceledOnTouchOutside(true)
+ window?.run {
+ val layoutParams = attributes
+ if (isPortrait()) {
+ layoutParams.width = ViewGroup.LayoutParams.MATCH_PARENT
+ layoutParams.height = UiUtils.getDeviceHeight() / 2
+ setGravity(Gravity.BOTTOM)
+ } else {
+ layoutParams.width = UiUtils.getDeviceWidth() / 2
+ layoutParams.height = ViewGroup.LayoutParams.MATCH_PARENT
+ setGravity(Gravity.END)
+ }
+ attributes = layoutParams
+ }
+ }
+ }
+
+ override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
+ if (dialog?.window?.attributes?.gravity != Gravity.BOTTOM) {
+ binding.layoutInfoContainer.setPadding(0, UiUtils.getStatusHeight(), 0, 0)
+ }
+ onNodeChangedListener?.let {
+ adapter.setOnNodeChangedListener(it)
+ }
+ initViewpager()
+ }
+
+ private fun initViewpager() {
+ binding.viewPager.adapter = adapter
+ binding.tabLayout.setupWithViewPager(binding.viewPager)
+ }
+}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoView.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoView.kt
deleted file mode 100644
index 874b6fd..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoView.kt
+++ /dev/null
@@ -1,38 +0,0 @@
-package com.wrbug.developerhelper.ui.widget.layoutinfoview
-
-import android.content.Context
-import com.google.android.material.bottomsheet.BottomSheetDialog
-import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
-import com.wrbug.developerhelper.commonutil.UiUtils
-import kotlinx.android.synthetic.main.view_layout_info.*
-
-
-class LayoutInfoView(
- context: Context,
- private val nodeList: List?,
- private val hierarchyNode: HierarchyNode
-) : BottomSheetDialog(context) {
- val adapter = LayoutInfoViewPagerAdapter(context, nodeList, hierarchyNode)
-
- init {
- init()
- }
-
- fun setOnNodeChangedListener(listener: OnNodeChangedListener) {
- adapter.setOnNodeChangedListener(listener)
- }
-
- private fun init() {
- setContentView(R.layout.view_layout_info)
- val layoutParams = layoutInfoContainer.layoutParams
- layoutParams.height = UiUtils.getDeviceHeight(context) / 2
- layoutInfoContainer.layoutParams = layoutParams
- initViewpager()
- }
-
- private fun initViewpager() {
- viewPager.adapter = adapter
- tabLayout.setupWithViewPager(viewPager)
- }
-}
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoViewPagerAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoViewPagerAdapter.kt
index 051790d..bb9148e 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoViewPagerAdapter.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/LayoutInfoViewPagerAdapter.kt
@@ -9,7 +9,7 @@ import androidx.recyclerview.widget.RecyclerView
import androidx.viewpager.widget.PagerAdapter
import androidx.viewpager.widget.ViewPager
import com.wrbug.developerhelper.R
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
+import com.wrbug.developerhelper.base.entry.HierarchyNode
import com.wrbug.developerhelper.ui.widget.boundsinfoview.BoundsInfoView
import com.wrbug.developerhelper.ui.widget.layoutinfoview.infopage.InfoAdapter
import com.wrbug.developerhelper.ui.widget.layoutinfoview.infopage.ItemInfo
@@ -30,7 +30,7 @@ class LayoutInfoViewPagerAdapter(
private val infoAdapter: InfoAdapter = InfoAdapter(context)
private lateinit var graphView: GraphView
private val boundsInfoView = BoundsInfoView(context)
- private var onNodeChangedListener: OnNodeChangedListener? = null
+ private var onNodeChangedListener: ((HierarchyNode, HierarchyNode?) -> Unit)? = null
init {
initInfoTab()
@@ -38,14 +38,14 @@ class LayoutInfoViewPagerAdapter(
initViewTreeTab()
}
- fun setOnNodeChangedListener(listener: OnNodeChangedListener) {
+ fun setOnNodeChangedListener(listener: (HierarchyNode, HierarchyNode?) -> Unit) {
onNodeChangedListener = listener
}
private fun initViewTreeTab() {
tabList.add("ViewTree")
graphView =
- LayoutInflater.from(context).inflate(R.layout.layout_hierarchy_tree, null) as GraphView
+ LayoutInflater.from(context).inflate(R.layout.layout_hierarchy_tree, null) as GraphView
val adapter = ViewTreeGraphAdapter(context, R.layout.item_tree_node_view)
graphView.adapter = adapter
adapter.setOnItemClickListener(object : ViewTreeGraphAdapter.OnItemClickListener {
@@ -54,7 +54,7 @@ class LayoutInfoViewPagerAdapter(
resetInfoTab()
resetLayoutTable()
resetViewTreeTab(node)
- onNodeChangedListener?.onChanged(node.node, node.parent?.node)
+ onNodeChangedListener?.invoke(node.node, node.parent?.node)
}
})
val configuration = BuchheimWalkerConfiguration.Builder()
@@ -150,7 +150,7 @@ class LayoutInfoViewPagerAdapter(
"${resourceId.replace(packageName, "app")}[${idHex?.replace("#", "0x") ?: "NO_ID"}]"
)
)
- if (!text.isEmpty()) {
+ if (text.isNotEmpty()) {
list.add(ItemInfo("Text", text))
}
list.add(ItemInfo("Enable", enabled))
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/OnNodeChangedListener.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/OnNodeChangedListener.kt
deleted file mode 100644
index 70af503..0000000
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/OnNodeChangedListener.kt
+++ /dev/null
@@ -1,7 +0,0 @@
-package com.wrbug.developerhelper.ui.widget.layoutinfoview
-
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
-
-interface OnNodeChangedListener {
- fun onChanged(node: HierarchyNode, parentNode: HierarchyNode?)
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphAdapter.kt
index 8a3e153..1871acd 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphAdapter.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphAdapter.kt
@@ -9,7 +9,7 @@ import androidx.cardview.widget.CardView
import com.wrbug.developerhelper.R
import de.blox.graphview.BaseGraphAdapter
-class ViewTreeGraphAdapter(@NonNull val context: Context, @LayoutRes val layoutRes: Int) :
+class ViewTreeGraphAdapter(val context: Context, @LayoutRes val layoutRes: Int) :
BaseGraphAdapter(context, layoutRes) {
private var listener: OnItemClickListener? = null
override fun onCreateViewHolder(view: View?) = ViewHolder(view!!)
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphNode.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphNode.kt
index 866b0b2..21a3846 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphNode.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/ViewTreeGraphNode.kt
@@ -1,6 +1,6 @@
package com.wrbug.developerhelper.ui.widget.layoutinfoview
-import com.wrbug.developerhelper.basecommon.entry.HierarchyNode
+import com.wrbug.developerhelper.base.entry.HierarchyNode
class ViewTreeGraphNode(val node: HierarchyNode) {
var selected: Boolean = false
diff --git a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/infopage/InfoAdapter.kt b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/infopage/InfoAdapter.kt
index 043c10f..f37bfe6 100644
--- a/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/infopage/InfoAdapter.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/ui/widget/layoutinfoview/infopage/InfoAdapter.kt
@@ -2,25 +2,56 @@ package com.wrbug.developerhelper.ui.widget.layoutinfoview.infopage
import android.content.Context
import android.view.LayoutInflater
-import android.view.View
import android.view.ViewGroup
import androidx.recyclerview.widget.RecyclerView
-import com.wrbug.developerhelper.R
+import com.airbnb.lottie.LottieDrawable
+import com.wrbug.developerhelper.commonutil.ClipboardUtils
import com.wrbug.developerhelper.commonutil.print
-import kotlinx.android.synthetic.main.item_view_info.view.*
+import com.wrbug.developerhelper.util.setOnDoubleCheckClickListener
+import com.wrbug.developerhelper.util.visible
+import com.wrbug.developerhelper.databinding.ItemInfoLoadingBinding
+import com.wrbug.developerhelper.databinding.ItemViewInfoBinding
-class InfoAdapter(val context: Context) : RecyclerView.Adapter() {
- private val list = arrayListOf()
- override fun onCreateViewHolder(p0: ViewGroup, p1: Int): ViewHolder {
- return ViewHolder(LayoutInflater.from(context).inflate(R.layout.item_view_info, p0, false))
+class InfoAdapter(val context: Context, private val topItem: ItemInfo? = null) :
+ RecyclerView.Adapter() {
+ companion object {
+ private const val VIEW_TYPE_ITEM = 0
+ private const val VIEW_TYPE_LOADING = 1
+ }
+
+ private val list = arrayListOf()
+ override fun onCreateViewHolder(p0: ViewGroup, p1: Int): RecyclerView.ViewHolder {
+ if (p1 == VIEW_TYPE_LOADING) {
+ return LoadingViewHolder(
+ ItemInfoLoadingBinding.inflate(
+ LayoutInflater.from(p0.context), p0, false
+ )
+ )
+ }
+ return InfoViewHolder(
+ ItemViewInfoBinding.inflate(
+ LayoutInflater.from(p0.context), p0, false
+ )
+ )
+ }
+
+ override fun getItemViewType(position: Int): Int {
+ return if (list[position] is LoadingItem) {
+ VIEW_TYPE_LOADING
+ } else {
+ VIEW_TYPE_ITEM
+ }
}
override fun getItemCount(): Int {
return list.size
}
- fun setItems(list: ArrayList) {
+ fun setItems(list: List) {
this.list.clear()
+ topItem?.let {
+ this.list.add(it)
+ }
if (list.isEmpty().not()) {
this.list.addAll(list)
}
@@ -35,17 +66,27 @@ class InfoAdapter(val context: Context) : RecyclerView.Adapter Unit) {
switcherMaskViewClicked = true
- switcherMaskView.setOnClickListener(listener)
+ binding.switcherMaskView.setOnDoubleCheckClickListener(clickListener = listener)
}
- fun isChecked() = switcher.isChecked
+ fun isChecked() = binding.switcher.isChecked
private fun setSwitchCheckable() {
- if (checkable) {
-// switcher.setOnTouchListener(null)
- switcherMaskView.visibility = View.GONE
- } else {
-// switcher.setOnTouchListener { _, _ -> true }
- switcherMaskView.visibility = View.VISIBLE
- }
+ binding.switcherMaskView.isGone = checkable
}
fun setOnCheckedChangeListener(listener: CompoundButton.OnCheckedChangeListener) {
- switcher.setOnCheckedChangeListener(listener)
+ binding.switcher.setOnCheckedChangeListener(listener)
}
- fun setImage(drawable: Drawable) {
- icoIv.setImageDrawable(drawable)
- icoIv.visibility = View.VISIBLE
+ fun setImage(drawable: Drawable?) {
+ binding.icoIv.setImageDrawable(drawable)
+ binding.icoIv.isVisible = drawable != null
}
fun setSummary(summary: String?) {
- summary.takeIf {
- !TextUtils.isEmpty(it)
- }?.let {
- summaryTv.text = it
- summaryTv.visibility = View.VISIBLE
- }
+ binding.summaryTv.text = summary
+ binding.summaryTv.isVisible = !summary.isNullOrEmpty()
}
-
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/AppStatusRegister.kt b/app/src/main/java/com/wrbug/developerhelper/util/AppStatusRegister.kt
new file mode 100644
index 0000000..1d49973
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/util/AppStatusRegister.kt
@@ -0,0 +1,56 @@
+package com.wrbug.developerhelper.util
+
+import android.app.Activity
+import android.app.Application
+import android.os.Bundle
+
+object AppStatusRegister {
+ private var count = 0
+ private val backgroundMap = hashMapOf Unit>()
+ fun init(application: Application) {
+ application.registerActivityLifecycleCallbacks(object :
+ Application.ActivityLifecycleCallbacks {
+ override fun onActivityCreated(activity: Activity, savedInstanceState: Bundle?) {
+
+ }
+
+ override fun onActivityStarted(activity: Activity) {
+ count++
+ }
+
+ override fun onActivityResumed(activity: Activity) {
+ }
+
+ override fun onActivityPaused(activity: Activity) {
+ }
+
+ override fun onActivityStopped(activity: Activity) {
+ count--
+ backgroundCheck()
+ }
+
+ override fun onActivitySaveInstanceState(activity: Activity, outState: Bundle) {
+ }
+
+ override fun onActivityDestroyed(activity: Activity) {
+ }
+
+ })
+ }
+
+
+ fun registerBackgroundListener(key: String, listener: () -> Unit) {
+ backgroundMap[key] = listener
+ }
+
+ fun removeBackgroundListener(key: String) {
+ backgroundMap.remove(key)
+ }
+
+ private fun backgroundCheck() {
+ if (count == 0) {
+ backgroundMap.values.forEach { it() }
+ }
+ }
+
+}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/BackupUtils.kt b/app/src/main/java/com/wrbug/developerhelper/util/BackupUtils.kt
index d037393..e77a499 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/BackupUtils.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/BackupUtils.kt
@@ -1,36 +1,159 @@
package com.wrbug.developerhelper.util
-import android.net.Uri
import android.os.Environment
-import com.wrbug.developerhelper.basecommon.BaseApp
+import com.wrbug.developerhelper.commonutil.Constant
+import com.wrbug.developerhelper.commonutil.entity.ApkInfo
+import com.wrbug.developerhelper.commonutil.fromJson
+import com.wrbug.developerhelper.commonutil.safeCreateSingle
+import com.wrbug.developerhelper.commonutil.safeRead
import com.wrbug.developerhelper.commonutil.shell.ShellManager
-import com.wrbug.developerhelper.commonutil.toUri
-import com.wrbug.developerhelper.xposed.developerhelper.DeveloperHelper
+import com.wrbug.developerhelper.commonutil.toJson
+import com.wrbug.developerhelper.model.entity.BackupAppData
+import com.wrbug.developerhelper.model.entity.BackupAppInfo
+import com.wrbug.developerhelper.model.entity.BackupAppItemInfo
+import io.reactivex.rxjava3.core.Single
+import net.dongliu.apk.parser.ApkFile
import java.io.File
object BackupUtils {
- private val backupDir: File by lazy {
- val file = File(Environment.getExternalStorageDirectory(), "com.wrbug.developerHelper/backup")
+ private const val ANDROID_DATA_TAR = "android_data.tar"
+ private const val DATA_TAR = "data.tar"
+ private const val CONFIG_JSON = "config.json"
+ private const val ICON_PNG = "icon.png"
+ private val backupRootDir: File by lazy {
+ val file =
+ File(Environment.getExternalStorageDirectory(), "DeveloperHelper/backup")
if (file.exists().not()) {
file.mkdirs()
}
file
}
- fun backupApk(packageName: String, apkPath: String, fileName: String): Uri? {
- val apkDir = File(backupDir, "apks/$packageName/$fileName")
+ fun getCurrentAppBackupDir(packageName: String, dateDir: String): File {
+ return File(getAppBackupDir(packageName), dateDir)
+ }
+
+ fun getAppBackupDir(packageName: String): File {
+ return File(backupRootDir, packageName).apply {
+ if (!exists()) {
+ mkdirs()
+ }
+ }
+ }
+
+ fun backupApk(
+ packageName: String,
+ dateDir: String,
+ apkPath: String,
+ fileName: String
+ ): String? {
+ val apkDir = File(getCurrentAppBackupDir(packageName, dateDir), fileName)
if (ShellManager.cpFile(apkPath, apkDir.absolutePath)) {
- return apkDir.toUri(BaseApp.instance)
+ return apkDir.absolutePath
}
return null
}
- fun backupAppData(packageName: String, dataDir: String): File? {
- val backupDataDir =
- File(backupDir, "datas/$packageName/${System.currentTimeMillis().format("yyyy-MM-dd-HH_mm_ss")}")
- if (ShellManager.cpFile(dataDir, backupDataDir.absolutePath)) {
+ fun backupAppData(dateDir: String, packageName: String): File? {
+ val backupDataDir = File(getCurrentAppBackupDir(packageName, dateDir), DATA_TAR)
+ val dataDir = Constant.getDataDir(packageName)
+ if (ShellManager.tarCF(backupDataDir.absolutePath, dataDir)) {
return backupDataDir
}
return null
}
+
+ fun backupAppAndroidData(dateDir: String, packageName: String): String? {
+ val backupDataDir = File(
+ getCurrentAppBackupDir(packageName, dateDir), ANDROID_DATA_TAR
+ )
+ val dataDir =
+ Environment.getExternalStorageDirectory().absolutePath + "/Android/data/" + packageName
+ if (!File(dataDir).exists()) {
+ return ""
+ }
+ if (ShellManager.tarCF(backupDataDir.absolutePath, dataDir)) {
+ return backupDataDir.absolutePath
+ }
+ return null
+ }
+
+ fun saveBackupInfo(
+ apkInfo: ApkInfo,
+ backupAppItemInfo: BackupAppItemInfo,
+ tarFile: String
+ ): Boolean {
+ val apkFile = apkInfo.applicationInfo.publicSourceDir
+ val tmpApkFile = File(getAppBackupDir(apkInfo.applicationInfo.packageName), "tmp.apk")
+ ShellManager.cpFile(apkFile, tmpApkFile.absolutePath)
+ runCatching {
+ ApkFile(tmpApkFile).allIcons.find { it.isFile }?.data?.let {
+ File(getAppBackupDir(apkInfo.applicationInfo.packageName), ICON_PNG).writeBytes(it)
+ }
+ }
+ tmpApkFile.delete()
+ backupAppItemInfo.androidDataFile = ANDROID_DATA_TAR
+ backupAppItemInfo.dataFile = DATA_TAR
+ val configFile = File(getAppBackupDir(apkInfo.applicationInfo.packageName), CONFIG_JSON)
+ val info = configFile.safeRead().fromJson() ?: BackupAppInfo()
+ info.appName = apkInfo.getAppName()
+ info.packageName = apkInfo.applicationInfo.packageName
+ info.backupMap[tarFile] = backupAppItemInfo
+ configFile.writeText(info.toJson().orEmpty())
+ return true
+ }
+
+
+ fun zipBackupFile(packageName: String, dateDir: String): File? {
+ val target = File(getAppBackupDir(packageName), "$dateDir.tar")
+ val src = getCurrentAppBackupDir(packageName, dateDir).absolutePath
+ if (ShellManager.tarCF(target.absolutePath, src)) {
+ ShellManager.rmFile(src)
+ return target
+ }
+ return null
+ }
+
+ fun getAllBackupInfo(): Single> {
+ return safeCreateSingle {
+ val list = arrayListOf()
+ backupRootDir.listFiles()?.forEach { root ->
+ val configJson = File(root, CONFIG_JSON)
+ if (!configJson.exists()) {
+ return@forEach
+ }
+ val info = configJson.safeRead().fromJson() ?: return@forEach
+ val map = info.backupMap.filter { File(root, it.key).exists() }
+ val icoFile = File(root, ICON_PNG).takeIf { it.exists() }
+ list.add(BackupAppData(info.appName, info.packageName, root, HashMap(map), icoFile))
+ }
+ it.onSuccess(list)
+ }
+ }
+
+ fun deleteBackupItem(packageName: String, tarFile: String): Single {
+ return safeCreateSingle {
+ val dir = backupRootDir.listFiles { _, name -> name == packageName }?.getOrNull(0)
+ val backupArch = dir?.let { File(it, tarFile) }
+ if (dir == null || !dir.exists() || backupArch?.exists() != true) {
+ it.onError(Exception())
+ return@safeCreateSingle
+ }
+ backupArch.delete()
+ val configJson = File(dir, CONFIG_JSON)
+ val info = configJson.safeRead().fromJson()
+ if (info == null) {
+ it.onError(Exception())
+ return@safeCreateSingle
+ }
+ val newInfo = info.copy(
+ backupMap = info.backupMap.apply { remove(tarFile) }
+ )
+ configJson.writeText(newInfo.toJson().orEmpty())
+ it.onSuccess(newInfo)
+ }
+
+ }
+
+
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/DateUtilsExt.kt b/app/src/main/java/com/wrbug/developerhelper/util/DateUtilsExt.kt
index 3e6e0cd..6fc9513 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/DateUtilsExt.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/DateUtilsExt.kt
@@ -1,9 +1,11 @@
package com.wrbug.developerhelper.util
+import android.annotation.SuppressLint
import java.text.SimpleDateFormat
import java.util.*
+@SuppressLint("SimpleDateFormat")
fun Date.format(format: String="yyyy-MM-dd HH:mm:ss"): String {
val simpleDateFormat = SimpleDateFormat(format)
return simpleDateFormat.format(this)
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/DeviceUtils.kt b/app/src/main/java/com/wrbug/developerhelper/util/DeviceUtils.kt
index b44b4c9..59742b1 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/DeviceUtils.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/DeviceUtils.kt
@@ -3,9 +3,10 @@ package com.wrbug.developerhelper.util
import android.content.Context
import android.os.Build
import android.provider.Settings
-import android.text.TextUtils
-import com.wrbug.developerhelper.basecommon.BaseApp
-import com.wrbug.developerhelper.commonutil.ShellUtils
+import android.util.DisplayMetrics
+import android.view.WindowManager
+import com.wrbug.developerhelper.base.BaseApp
+import com.wrbug.developerhelper.commonutil.shell.ShellUtils
object DeviceUtils {
@@ -14,10 +15,27 @@ object DeviceUtils {
}
fun isFloatWindowOpened(): Boolean {
- return isFloatWindowOpened(BaseApp.instance!!)
+ return isFloatWindowOpened(BaseApp.instance)
}
fun isFloatWindowOpened(context: Context): Boolean {
return Build.VERSION.SDK_INT < Build.VERSION_CODES.M || Settings.canDrawOverlays(context) } + + + fun getScreenWidth(): Int { + val manager: WindowManager = + BaseApp.instance.getSystemService(Context.WINDOW_SERVICE) as WindowManager + val outMetrics = DisplayMetrics() + manager.defaultDisplay.getMetrics(outMetrics) + return outMetrics.widthPixels + } + + fun getScreenHeight(): Int { + val manager: WindowManager = + BaseApp.instance.getSystemService(Context.WINDOW_SERVICE) as WindowManager + val outMetrics = DisplayMetrics() + manager.defaultDisplay.getMetrics(outMetrics) + return outMetrics.heightPixels + } } \ No newline at end of file diff --git a/app/src/main/java/com/wrbug/developerhelper/util/EnforceUtils.kt b/app/src/main/java/com/wrbug/developerhelper/util/EnforceUtils.kt deleted file mode 100644 index c4d389e..0000000 --- a/app/src/main/java/com/wrbug/developerhelper/util/EnforceUtils.kt +++ /dev/null @@ -1,42 +0,0 @@ -package com.wrbug.developerhelper.util -import com.wrbug.developerhelper.commonutil.shell.ShellManager - -object EnforceUtils { - fun getEnforceType(packageName: String): EnforceType { - val files = ShellManager.getZipFileList(ShellManager.findApkDir(packageName)).toString().trim() - return when { - isIjiaMi(files) -> EnforceType.I_JIA_MI
- is360(files) -> EnforceType.QI_HOO
- isLeGu(files) -> EnforceType.LE_GU
- isBangcle(files) -> EnforceType.BANGCLE
- else -> EnforceType.UN_KNOWN
- }
- }
-
- private fun isIjiaMi(files: String): Boolean {
- return files.contains("ijm") or files.contains("ijiami")
-
- }
-
- private fun is360(files: String): Boolean {
- return files.contains("libjiagu.so")
- }
-
- private fun isLeGu(files: String): Boolean {
- return files.contains("tencent_stub")
- }
-
- private fun isBangcle(files: String): Boolean {
- return files.contains("bangcle") || files.contains("SecShell")
- }
-
-
- enum class EnforceType(val type: String) {
- QI_HOO("360"),
- BAI_DU("百度"),
- I_JIA_MI("爱加密"),
- LE_GU("乐固"),
- BANGCLE("梆梆"),
- UN_KNOWN("未知/无加固"),
- }
-}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/OutSharedPreferenceManager.kt b/app/src/main/java/com/wrbug/developerhelper/util/OutSharedPreferenceManager.kt
index cc64336..5a6c852 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/OutSharedPreferenceManager.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/OutSharedPreferenceManager.kt
@@ -1,37 +1,95 @@
package com.wrbug.developerhelper.util
+import android.annotation.SuppressLint
import android.content.Context
+import com.wrbug.developerhelper.commonutil.Constant
+import com.wrbug.developerhelper.commonutil.shell.ShellManager
import com.wrbug.developerhelper.commonutil.toBoolean
import com.wrbug.developerhelper.commonutil.toInt
import com.wrbug.developerhelper.commonutil.toLong
import com.wrbug.developerhelper.model.entity.SharedPreferenceItemInfo
+import io.reactivex.rxjava3.core.Single
import java.io.File
-object OutSharedPreferenceManager {
- fun saveToFile(context: Context, list: Array): File {
- val name = System.currentTimeMillis().toString()
- val sp = context.applicationContext.getSharedPreferences(name, Context.MODE_PRIVATE)
- val edit = sp.edit()
- for (sharedPreferenceItemInfo in list) {
- when (sharedPreferenceItemInfo.type.toLowerCase()) {
- "string" -> {
- edit.putString(sharedPreferenceItemInfo.key, sharedPreferenceItemInfo.getValidValue())
- }
- "int" -> {
- edit.putInt(sharedPreferenceItemInfo.key, sharedPreferenceItemInfo.getValidValue().toInt())
- }
- "long" -> {
- edit.putLong(sharedPreferenceItemInfo.key, sharedPreferenceItemInfo.getValidValue().toLong())
- }
- "boolean" -> {
- edit.putBoolean(sharedPreferenceItemInfo.key, sharedPreferenceItemInfo.getValidValue().toBoolean())
- }
- "float" -> {
- edit.putFloat(sharedPreferenceItemInfo.key, sharedPreferenceItemInfo.getValidValue().toFloat())
- }
+class OutSharedPreference(private val context: Context, private val filePath: String) {
+ private val fileName by lazy {
+ System.currentTimeMillis().toString()
+ }
+ private val tmpSpFile by lazy {
+ File("${Constant.getDataDir(context.packageName)}/shared_prefs/${fileName}.xml")
+ }
+ private val tmpFile by lazy {
+ File(context.cacheDir, "$fileName.xml")
+ }
+
+ fun parse(): Single> {
+ return Single.just(filePath).map {
+ if (!ShellManager.catFile(it, tmpFile.absolutePath, "777")) {
+ return@map emptyArray()
}
+ val xml = tmpFile.readText()
+ XmlUtil.parseSharedPreference(xml)
}
- edit.commit()
- return File("/data/data/${context.packageName}/shared_prefs/$name.xml")
+ }
+
+ @SuppressLint("ApplySharedPref")
+ fun saveToFile(context: Context, list: Array): Single {
+ return Single.just(list).map {
+ val sp = context.applicationContext.getSharedPreferences(fileName, Context.MODE_PRIVATE)
+ val edit = sp.edit()
+ for (sharedPreferenceItemInfo in list) {
+ when (sharedPreferenceItemInfo.type.lowercase()) {
+ "string" -> {
+ edit.putString(
+ sharedPreferenceItemInfo.key,
+ sharedPreferenceItemInfo.getValidValue()
+ )
+ }
+
+ "int" -> {
+ edit.putInt(
+ sharedPreferenceItemInfo.key,
+ sharedPreferenceItemInfo.getValidValue().toInt()
+ )
+ }
+
+ "long" -> {
+ edit.putLong(
+ sharedPreferenceItemInfo.key,
+ sharedPreferenceItemInfo.getValidValue().toLong()
+ )
+ }
+
+ "boolean" -> {
+ edit.putBoolean(
+ sharedPreferenceItemInfo.key,
+ sharedPreferenceItemInfo.getValidValue().toBoolean()
+ )
+ }
+
+ "float" -> {
+ edit.putFloat(
+ sharedPreferenceItemInfo.key,
+ sharedPreferenceItemInfo.getValidValue().toFloat()
+ )
+ }
+
+ "set" -> {
+ edit.putStringSet(
+ sharedPreferenceItemInfo.key,
+ sharedPreferenceItemInfo.getValidValue().split(",").toSet()
+ )
+ }
+ }
+ }
+ edit.commit()
+ ShellManager.catFile(tmpSpFile.absolutePath, filePath, "666")
+ }.onErrorReturn { false }
+ }
+
+
+ fun deleteTmpFile() {
+ tmpSpFile.delete()
+ tmpFile.delete()
}
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/ResourceUtilsExt.kt b/app/src/main/java/com/wrbug/developerhelper/util/ResourceUtilsExt.kt
index 4effcd8..a5083dc 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/ResourceUtilsExt.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/ResourceUtilsExt.kt
@@ -1,12 +1,27 @@
package com.wrbug.developerhelper.util
import android.content.Context
-import com.wrbug.developerhelper.basecommon.BaseApp
+import android.content.res.Configuration
+import androidx.core.content.ContextCompat
+import com.wrbug.developerhelper.base.BaseApp
-fun Int.toResString(context: Context = BaseApp.instance): String {
+fun Int.getString(context: Context = BaseApp.instance): String {
return context.getString(this)
}
+fun Int.getColor(context: Context = BaseApp.instance): Int {
+ return ContextCompat.getColor(context, this)
+}
+
+fun Int.getString(vararg formatArgs: Any): String {
+ return BaseApp.instance.getString(this, *formatArgs)
+}
+
fun getString(resId: Int): String {
return BaseApp.instance.getString(resId)
+}
+
+
+fun isPortrait(): Boolean {
+ return BaseApp.instance.resources.configuration.orientation == Configuration.ORIENTATION_PORTRAIT
}
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/UpdateUtils.kt b/app/src/main/java/com/wrbug/developerhelper/util/UpdateUtils.kt
index 3010f62..c991959 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/UpdateUtils.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/UpdateUtils.kt
@@ -1,42 +1,11 @@
package com.wrbug.developerhelper.util
-import com.wrbug.developerhelper.commonutil.HttpUtil
-import com.wrbug.developerhelper.commonutil.shell.Callback
import org.jetbrains.anko.doAsync
-import org.jetbrains.anko.uiThread
-import com.wrbug.developerhelper.commonutil.OkhttpUtils
-import com.wrbug.developerhelper.model.entity.VersionInfo
-import org.jsoup.Jsoup
-import java.lang.Exception
object UpdateUtils {
- private const val URL = "https://www.coolapk.com/apk/com.wrbug.developerhelper"
- fun checkUpdate(callback: Callback) {
+ fun checkUpdate() {
doAsync {
- try {
- val document = Jsoup.connect(URL)
- .sslSocketFactory(OkhttpUtils.createSSLSocketFactory()).get()
- val versionName = document.getElementsByClass("list_app_info").text() ?: ""
- val feature = document.getElementsByClass("apk_left_title_info").first().html().replace("
", "\n")
- val size = document.getElementsByClass("apk_topba_message").html().split("/")[0].trim()
- val updateTime =
- document.getElementsByClass("apk_left_title_info")[2].html().split("
")[1].replace(
- "更新时间:",
- ""
- )
- val info = VersionInfo()
- info.versionName = versionName
- info.feature = feature
- info.size = size
- info.updateDate = updateTime
- info.downloadUrl = URL
- uiThread {
- callback.onSuccess(info)
- }
- } catch (e: Exception) {
- uiThread { callback.onFailed() }
- }
}
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/ViewExt.kt b/app/src/main/java/com/wrbug/developerhelper/util/ViewExt.kt
new file mode 100644
index 0000000..4a39531
--- /dev/null
+++ b/app/src/main/java/com/wrbug/developerhelper/util/ViewExt.kt
@@ -0,0 +1,94 @@
+package com.wrbug.developerhelper.util
+
+import android.app.ActionBar.LayoutParams
+import android.os.SystemClock
+import android.view.View
+import android.view.ViewGroup
+import android.widget.FrameLayout
+import android.widget.ImageView
+import android.widget.Toast
+import androidx.core.view.isVisible
+import com.airbnb.lottie.LottieAnimationView
+import com.airbnb.lottie.LottieDrawable
+import com.bumptech.glide.Glide
+import com.wrbug.developerhelper.R
+import com.wrbug.developerhelper.mmkv.ConfigKv
+import com.wrbug.developerhelper.mmkv.manager.MMKVManager
+
+private val configKv by lazy {
+ MMKVManager.get(ConfigKv::class.java)
+}
+
+fun View?.setOnDoubleCheckClickListener(duration: Long = 800, clickListener: (View) -> Unit) {
+ this?.setOnClickListener {
+ val time = SystemClock.elapsedRealtime()
+ val lastTime = (it.getTag(R.id.double_check_click) as? Long) ?: 0
+ if (time - lastTime> duration) {
+ it.setTag(R.id.double_check_click, time)
+ clickListener(it)
+ }
+ }
+}
+
+
+fun View?.setOnRootCheckClickListener(clickListener: (View) -> Unit) {
+ setOnDoubleCheckClickListener {
+ if (!configKv.isOpenRoot()) {
+ Toast.makeText(it.context, R.string.open_root_notice, Toast.LENGTH_SHORT).show()
+ return@setOnDoubleCheckClickListener
+ }
+ clickListener(it)
+ }
+}
+
+fun FrameLayout.startPageLoading() {
+ val lottieView = LottieAnimationView(context)
+ lottieView.id = R.id.lottie_loading
+ lottieView.setAnimation(R.raw.lottie_page_loading)
+ lottieView.playAnimation()
+ lottieView.repeatCount = LottieDrawable.INFINITE
+ removeAllViews()
+ isVisible = true
+ addView(
+ lottieView,
+ FrameLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT)
+ )
+}
+
+fun FrameLayout.stopPageLoading() {
+ val lottieView: LottieAnimationView? = findViewById(R.id.lottie_loading)
+ lottieView?.cancelAnimation()
+ removeAllViews()
+ isVisible = false
+}
+
+
+fun ImageView.loadImage(url: Any?, default: Int? = null) {
+ Glide.with(this).load(url).apply {
+ if (default != null) {
+ error(default)
+ }
+ }.into(this)
+}
+
+
+inline var View.visible: Boolean
+ set(value) {
+ visibility = if (value) {
+ View.VISIBLE
+ } else {
+ View.GONE
+ }
+ }
+ get() = visibility == View.VISIBLE
+
+
+inline var View.inVisible: Boolean
+ set(value) {
+ visibility = if (value) {
+ View.INVISIBLE
+ } else {
+ View.VISIBLE
+ }
+ }
+ get() = visibility == View.INVISIBLE
\ No newline at end of file
diff --git a/app/src/main/java/com/wrbug/developerhelper/util/XmlUtil.kt b/app/src/main/java/com/wrbug/developerhelper/util/XmlUtil.kt
index 1347a91..4b89580 100644
--- a/app/src/main/java/com/wrbug/developerhelper/util/XmlUtil.kt
+++ b/app/src/main/java/com/wrbug/developerhelper/util/XmlUtil.kt
@@ -3,6 +3,7 @@ package com.wrbug.developerhelper.util
import com.wrbug.developerhelper.model.entity.SharedPreferenceItemInfo
import org.dom4j.DocumentHelper
import org.dom4j.Element
+import org.dom4j.tree.DefaultElement
/**
* xml相关的工具类
@@ -28,6 +29,11 @@ object XmlUtil {
if (type == "string") {
info.value = child.text
info.newValue = child.text
+ } else if (type == "set") {
+ info.value = child.content().filterIsInstance(DefaultElement::class.java)
+ .joinToString(",") { it.text }
+ info.newValue = child.content().filterIsInstance(DefaultElement::class.java)
+ .joinToString(",") { it.text }
} else {
info.value = child.attributeValue("value")
info.newValue = child.attributeValue("value")
diff --git a/app/src/main/res/drawable/bg_round_8dp_white.xml b/app/src/main/res/drawable/bg_round_8dp_white.xml
new file mode 100644
index 0000000..3edf0a7
--- /dev/null
+++ b/app/src/main/res/drawable/bg_round_8dp_white.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_accessibility.xml b/app/src/main/res/drawable/ic_accessibility.xml
new file mode 100644
index 0000000..c872987
--- /dev/null
+++ b/app/src/main/res/drawable/ic_accessibility.xml
@@ -0,0 +1,12 @@
+
+
+
+
diff --git a/app/src/main/res/drawable/ic_accessibility_666666.xml b/app/src/main/res/drawable/ic_accessibility_666666.xml
deleted file mode 100644
index 0204f1a..0000000
--- a/app/src/main/res/drawable/ic_accessibility_666666.xml
+++ /dev/null
@@ -1,13 +0,0 @@
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_accessibility_black.xml b/app/src/main/res/drawable/ic_accessibility_black.xml
deleted file mode 100644
index 3239379..0000000
--- a/app/src/main/res/drawable/ic_accessibility_black.xml
+++ /dev/null
@@ -1,13 +0,0 @@
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_application.xml b/app/src/main/res/drawable/ic_application.xml
new file mode 100644
index 0000000..630ea85
--- /dev/null
+++ b/app/src/main/res/drawable/ic_application.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_canceled.xml b/app/src/main/res/drawable/ic_canceled.xml
new file mode 100644
index 0000000..99b4c15
--- /dev/null
+++ b/app/src/main/res/drawable/ic_canceled.xml
@@ -0,0 +1,21 @@
+
+
+
+
+
+
+
diff --git a/app/src/main/res/drawable/ic_capture_666666.xml b/app/src/main/res/drawable/ic_capture_666666.xml
deleted file mode 100644
index 06e142e..0000000
--- a/app/src/main/res/drawable/ic_capture_666666.xml
+++ /dev/null
@@ -1,6 +0,0 @@
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_chevron_left.xml b/app/src/main/res/drawable/ic_chevron_left.xml
new file mode 100644
index 0000000..2fd7c1c
--- /dev/null
+++ b/app/src/main/res/drawable/ic_chevron_left.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_chevron_left_white_24dp.xml b/app/src/main/res/drawable/ic_chevron_left_24dp.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_chevron_left_white_24dp.xml
rename to app/src/main/res/drawable/ic_chevron_left_24dp.xml
diff --git a/app/src/main/res/drawable/ic_chevron_right.xml b/app/src/main/res/drawable/ic_chevron_right.xml
new file mode 100644
index 0000000..ce2744c
--- /dev/null
+++ b/app/src/main/res/drawable/ic_chevron_right.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_chevron_right_white_24dp.xml b/app/src/main/res/drawable/ic_chevron_right_24dp.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_chevron_right_white_24dp.xml
rename to app/src/main/res/drawable/ic_chevron_right_24dp.xml
diff --git a/app/src/main/res/drawable/ic_copy.xml b/app/src/main/res/drawable/ic_copy.xml
new file mode 100644
index 0000000..5b54247
--- /dev/null
+++ b/app/src/main/res/drawable/ic_copy.xml
@@ -0,0 +1,12 @@
+
+
+
+
diff --git a/app/src/main/res/drawable/ic_data_finder_666666.xml b/app/src/main/res/drawable/ic_data_finder_666666.xml
deleted file mode 100644
index dc5e784..0000000
--- a/app/src/main/res/drawable/ic_data_finder_666666.xml
+++ /dev/null
@@ -1,19 +0,0 @@
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_default_app_ico_place_holder.xml b/app/src/main/res/drawable/ic_default_app_ico_place_holder.xml
new file mode 100644
index 0000000..0eaded5
--- /dev/null
+++ b/app/src/main/res/drawable/ic_default_app_ico_place_holder.xml
@@ -0,0 +1,21 @@
+
+
+
+
+
+
+
diff --git a/app/src/main/res/drawable/ic_fail.xml b/app/src/main/res/drawable/ic_fail.xml
new file mode 100644
index 0000000..65ffd33
--- /dev/null
+++ b/app/src/main/res/drawable/ic_fail.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_float_air_bubble_666666.xml b/app/src/main/res/drawable/ic_float_air_bubble_666666.xml
deleted file mode 100644
index 31149a0..0000000
--- a/app/src/main/res/drawable/ic_float_air_bubble_666666.xml
+++ /dev/null
@@ -1,10 +0,0 @@
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_ic_empty_666666.xml b/app/src/main/res/drawable/ic_ic_empty.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_ic_empty_666666.xml
rename to app/src/main/res/drawable/ic_ic_empty.xml
diff --git a/app/src/main/res/drawable/ic_ignore.xml b/app/src/main/res/drawable/ic_ignore.xml
new file mode 100644
index 0000000..04dbaad
--- /dev/null
+++ b/app/src/main/res/drawable/ic_ignore.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_notification.xml b/app/src/main/res/drawable/ic_notification.xml
new file mode 100644
index 0000000..b60ed3d
--- /dev/null
+++ b/app/src/main/res/drawable/ic_notification.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_root_666666.xml b/app/src/main/res/drawable/ic_root.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_root_666666.xml
rename to app/src/main/res/drawable/ic_root.xml
diff --git a/app/src/main/res/drawable/ic_search_white_24dp.xml b/app/src/main/res/drawable/ic_search_24.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_search_white_24dp.xml
rename to app/src/main/res/drawable/ic_search_24.xml
diff --git a/app/src/main/res/drawable/ic_sheldon_root.xml b/app/src/main/res/drawable/ic_sheldon_root.xml
new file mode 100644
index 0000000..ef66512
--- /dev/null
+++ b/app/src/main/res/drawable/ic_sheldon_root.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_shield_666666.xml b/app/src/main/res/drawable/ic_shield.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_shield_666666.xml
rename to app/src/main/res/drawable/ic_shield.xml
diff --git a/app/src/main/res/drawable/ic_success.xml b/app/src/main/res/drawable/ic_success.xml
new file mode 100644
index 0000000..309b83d
--- /dev/null
+++ b/app/src/main/res/drawable/ic_success.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_switch_666666.xml b/app/src/main/res/drawable/ic_switch.xml
similarity index 100%
rename from app/src/main/res/drawable/ic_switch_666666.xml
rename to app/src/main/res/drawable/ic_switch.xml
diff --git a/app/src/main/res/drawable/ic_waiting.xml b/app/src/main/res/drawable/ic_waiting.xml
new file mode 100644
index 0000000..b090eb1
--- /dev/null
+++ b/app/src/main/res/drawable/ic_waiting.xml
@@ -0,0 +1,9 @@
+
+
+
diff --git a/app/src/main/res/drawable/ic_xposed_666666.xml b/app/src/main/res/drawable/ic_xposed_666666.xml
deleted file mode 100644
index 247f7a8..0000000
--- a/app/src/main/res/drawable/ic_xposed_666666.xml
+++ /dev/null
@@ -1,23 +0,0 @@
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_xposed_custom_666666.xml b/app/src/main/res/drawable/ic_xposed_custom_666666.xml
deleted file mode 100644
index 80f08b0..0000000
--- a/app/src/main/res/drawable/ic_xposed_custom_666666.xml
+++ /dev/null
@@ -1,6 +0,0 @@
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/onboarding_indicator_selected.xml b/app/src/main/res/drawable/onboarding_indicator_selected.xml
deleted file mode 100644
index 3d6e9fa..0000000
--- a/app/src/main/res/drawable/onboarding_indicator_selected.xml
+++ /dev/null
@@ -1,6 +0,0 @@
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/drawable/onboarding_indicator_unselected.xml b/app/src/main/res/drawable/onboarding_indicator_unselected.xml
deleted file mode 100644
index 7461fa9..0000000
--- a/app/src/main/res/drawable/onboarding_indicator_unselected.xml
+++ /dev/null
@@ -1,22 +0,0 @@
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/commonwidget/src/main/res/drawable/toast_bg_circle_cornor_rect.xml b/app/src/main/res/drawable/toast_bg_circle_cornor_rect.xml
similarity index 100%
rename from commonwidget/src/main/res/drawable/toast_bg_circle_cornor_rect.xml
rename to app/src/main/res/drawable/toast_bg_circle_cornor_rect.xml
diff --git a/app/src/main/res/drawable/widget_ic_no_data.xml b/app/src/main/res/drawable/widget_ic_no_data.xml
new file mode 100644
index 0000000..35be377
--- /dev/null
+++ b/app/src/main/res/drawable/widget_ic_no_data.xml
@@ -0,0 +1,66 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/app/src/main/res/layout/activity_app_backup_detail.xml b/app/src/main/res/layout/activity_app_backup_detail.xml
new file mode 100644
index 0000000..6960008
--- /dev/null
+++ b/app/src/main/res/layout/activity_app_backup_detail.xml
@@ -0,0 +1,30 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/app/src/main/res/layout/activity_app_xposed_module_manager.xml b/app/src/main/res/layout/activity_app_xposed_module_manager.xml
deleted file mode 100644
index f8d26b1..0000000
--- a/app/src/main/res/layout/activity_app_xposed_module_manager.xml
+++ /dev/null
@@ -1,9 +0,0 @@
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/layout/activity_app_xposed_setting.xml b/app/src/main/res/layout/activity_app_xposed_setting.xml
deleted file mode 100644
index 2947c32..0000000
--- a/app/src/main/res/layout/activity_app_xposed_setting.xml
+++ /dev/null
@@ -1,33 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/layout/activity_backup_app.xml b/app/src/main/res/layout/activity_backup_app.xml
new file mode 100644
index 0000000..bc844e7
--- /dev/null
+++ b/app/src/main/res/layout/activity_backup_app.xml
@@ -0,0 +1,27 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/app/src/main/res/layout/activity_base_xposed_app_manager.xml b/app/src/main/res/layout/activity_base_xposed_app_manager.xml
deleted file mode 100644
index 0ad8ed7..0000000
--- a/app/src/main/res/layout/activity_base_xposed_app_manager.xml
+++ /dev/null
@@ -1,34 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/layout/activity_database_edit.xml b/app/src/main/res/layout/activity_database_edit.xml
index bfacaa4..4492843 100644
--- a/app/src/main/res/layout/activity_database_edit.xml
+++ b/app/src/main/res/layout/activity_database_edit.xml
@@ -25,7 +25,8 @@
android:padding="20dp"
android:text="@string/table_data_is_empty"
android:textSize="18sp"
- android:visibility="gone" />
+ android:visibility="gone"
+ app:layout_constraintTop_toTopOf="parent" />
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml
index 64a84cb..7325ea0 100644
--- a/app/src/main/res/layout/activity_main.xml
+++ b/app/src/main/res/layout/activity_main.xml
@@ -1,100 +1,114 @@
-
+ xmlns:tools="http://schemas.android.com/tools"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent"
+ tools:context=".ui.activity.main.MainActivity">
-
-
+
-
-
+
+
-
+ app:layout_behavior="@string/appbar_scrolling_view_behavior">
-
+ android:layout_height="wrap_content"
+ android:orientation="vertical"
+ android:paddingHorizontal="24dp">
-
-
+
-
+
-
+ app:checkable="false"
+ app:icoTint="@color/iconPrimary"
+ app:src="@drawable/ic_float_air_bubble"
+ app:title="@string/open_float_window" />
-
+
-
+
-
+
-
+
-
+
-
+
-
-
-
-
-
\ No newline at end of file
+
+
+
+
diff --git a/app/src/main/res/layout/activity_shared_preference_edit.xml b/app/src/main/res/layout/activity_shared_preference_edit.xml
index 3bec598..8f95392 100644
--- a/app/src/main/res/layout/activity_shared_preference_edit.xml
+++ b/app/src/main/res/layout/activity_shared_preference_edit.xml
@@ -17,5 +17,19 @@
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
- app:layout_behavior="@string/appbar_scrolling_view_behavior" />
+ app:layout_behavior="@string/appbar_scrolling_view_behavior"
+ tools:listitem="@layout/item_shared_preference_info" />
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/activity_xposed_setting.xml b/app/src/main/res/layout/activity_xposed_setting.xml
deleted file mode 100644
index 3a36541..0000000
--- a/app/src/main/res/layout/activity_xposed_setting.xml
+++ /dev/null
@@ -1,68 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/app/src/main/res/layout/dialog_apk_info.xml b/app/src/main/res/layout/dialog_apk_info.xml
index 7d8a32a..e3d19ea 100644
--- a/app/src/main/res/layout/dialog_apk_info.xml
+++ b/app/src/main/res/layout/dialog_apk_info.xml
@@ -1,4 +1,5 @@
+
-
\ No newline at end of file
+
+
diff --git a/app/src/main/res/layout/dialog_backup_app.xml b/app/src/main/res/layout/dialog_backup_app.xml
new file mode 100644
index 0000000..9211711
--- /dev/null
+++ b/app/src/main/res/layout/dialog_backup_app.xml
@@ -0,0 +1,71 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/dialog_backup_app_select.xml b/app/src/main/res/layout/dialog_backup_app_select.xml
new file mode 100644
index 0000000..21d3dc6
--- /dev/null
+++ b/app/src/main/res/layout/dialog_backup_app_select.xml
@@ -0,0 +1,56 @@
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/fragment_guide.xml b/app/src/main/res/layout/fragment_guide.xml
index e67d07d..4e15b75 100644
--- a/app/src/main/res/layout/fragment_guide.xml
+++ b/app/src/main/res/layout/fragment_guide.xml
@@ -1,4 +1,5 @@
@@ -48,5 +49,4 @@
tools:text="Your awesome description about the app" />
-
-
\ No newline at end of file
+
diff --git a/app/src/main/res/layout/item_backup_app_info.xml b/app/src/main/res/layout/item_backup_app_info.xml
new file mode 100644
index 0000000..9fdd236
--- /dev/null
+++ b/app/src/main/res/layout/item_backup_app_info.xml
@@ -0,0 +1,64 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_backup_detail_info.xml b/app/src/main/res/layout/item_backup_detail_info.xml
new file mode 100644
index 0000000..b35cff6
--- /dev/null
+++ b/app/src/main/res/layout/item_backup_detail_info.xml
@@ -0,0 +1,82 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_empty_data.xml b/app/src/main/res/layout/item_empty_data.xml
new file mode 100644
index 0000000..c90c36a
--- /dev/null
+++ b/app/src/main/res/layout/item_empty_data.xml
@@ -0,0 +1,12 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_info_loading.xml b/app/src/main/res/layout/item_info_loading.xml
new file mode 100644
index 0000000..89661e5
--- /dev/null
+++ b/app/src/main/res/layout/item_info_loading.xml
@@ -0,0 +1,11 @@
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_loading_data.xml b/app/src/main/res/layout/item_loading_data.xml
new file mode 100644
index 0000000..54f2089
--- /dev/null
+++ b/app/src/main/res/layout/item_loading_data.xml
@@ -0,0 +1,8 @@
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_shared_preference_info.xml b/app/src/main/res/layout/item_shared_preference_info.xml
index 44c755d..6b6eb94 100644
--- a/app/src/main/res/layout/item_shared_preference_info.xml
+++ b/app/src/main/res/layout/item_shared_preference_info.xml
@@ -16,8 +16,7 @@
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
- android:textColor="@color/item_title_text"
- android:textSize="16sp"
+ android:textAppearance="@style/TextAppearance.ItemTitle"
tools:text="标题" />
+ android:textColor="@color/material_color_light_blue_500"
+ android:visibility="gone" />
diff --git a/app/src/main/res/layout/item_view_info.xml b/app/src/main/res/layout/item_view_info.xml
index a92cd12..57ce62b 100644
--- a/app/src/main/res/layout/item_view_info.xml
+++ b/app/src/main/res/layout/item_view_info.xml
@@ -1,28 +1,45 @@
-
+ android:paddingHorizontal="24dp">
-
\ No newline at end of file
+
+
+
\ No newline at end of file
diff --git a/commonwidget/src/main/res/layout/layout_toast_flexible.xml b/app/src/main/res/layout/layout_toast_flexible.xml
similarity index 100%
rename from commonwidget/src/main/res/layout/layout_toast_flexible.xml
rename to app/src/main/res/layout/layout_toast_flexible.xml
diff --git a/app/src/main/res/layout/layout_toolbar.xml b/app/src/main/res/layout/layout_toolbar.xml
index 5532922..9b16398 100644
--- a/app/src/main/res/layout/layout_toolbar.xml
+++ b/app/src/main/res/layout/layout_toolbar.xml
@@ -1,13 +1,13 @@
-
\ No newline at end of file
diff --git a/app/src/main/res/layout/view_app_bar.xml b/app/src/main/res/layout/view_app_bar.xml
new file mode 100644
index 0000000..ba968b3
--- /dev/null
+++ b/app/src/main/res/layout/view_app_bar.xml
@@ -0,0 +1,63 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/view_app_data_info.xml b/app/src/main/res/layout/view_app_data_info.xml
index bca1bbc..104f635 100644
--- a/app/src/main/res/layout/view_app_data_info.xml
+++ b/app/src/main/res/layout/view_app_data_info.xml
@@ -17,8 +17,7 @@
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Apk目录"
- android:textColor="@color/item_title_text"
- android:textSize="16sp" />
+ android:textAppearance="@style/TextAppearance.ItemTitle" />
+ android:textAppearance="@style/TextAppearance.ItemTitle" />
+ android:textAppearance="@style/TextAppearance.ItemTitle" />
+ android:textAppearance="@style/TextAppearance.ItemTitle" />
+ android:textAppearance="@style/TextAppearance.ItemTitle" />
-
+ android:paddingTop="24dp"
+ app:constraint_referenced_ids="backupAppBtn,restoreAppBtn,restartAppBtn,stopAppBtn,deleteAppDataBtn,uninstallAppBtn"
+ app:flow_verticalGap="16dp"
+ app:flow_wrapMode="aligned"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintTop_toTopOf="parent" />
+
+ android:text="@string/backup_app_file"
+ app:layout_constraintWidth_percent="0.45" />
+ android:text="@string/restore_app_file"
+ app:layout_constraintWidth_percent="0.45" />
+ app:layout_constraintWidth_percent="0.45" />
+ app:layout_constraintWidth_percent="0.45" />
+ app:layout_constraintWidth_percent="0.45" />
-
-
+ app:layout_constraintWidth_percent="0.45" />
\ No newline at end of file
diff --git a/app/src/main/res/layout/view_backup_progress.xml b/app/src/main/res/layout/view_backup_progress.xml
new file mode 100644
index 0000000..102ea64
--- /dev/null
+++ b/app/src/main/res/layout/view_backup_progress.xml
@@ -0,0 +1,49 @@
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/view_empty_view.xml b/app/src/main/res/layout/view_empty_view.xml
new file mode 100644
index 0000000..6b19298
--- /dev/null
+++ b/app/src/main/res/layout/view_empty_view.xml
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/view_layout_info.xml b/app/src/main/res/layout/view_layout_info.xml
index 381235f..531797c 100644
--- a/app/src/main/res/layout/view_layout_info.xml
+++ b/app/src/main/res/layout/view_layout_info.xml
@@ -3,12 +3,17 @@
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/layoutInfoContainer"
android:layout_width="match_parent"
- android:layout_height="300dp">
+ android:layout_height="match_parent"
+ android:background="@android:color/white">
-
+ android:paddingVertical="8dp"
+ tools:parentTag="androidx.constraintlayout.widget.ConstraintLayout">
+ tools:src="@drawable/ic_accessibility" />
-
+ app:layout_constraintTop_toTopOf="@+id/titleTv" />
+
+
-
\ No newline at end of file
+
\ No newline at end of file
diff --git a/app/src/main/res/raw/lottie_list_loading.json b/app/src/main/res/raw/lottie_list_loading.json
new file mode 100644
index 0000000..8fa429d
--- /dev/null
+++ b/app/src/main/res/raw/lottie_list_loading.json
@@ -0,0 +1 @@
+{"v":"5.7.4","fr":29.9700012207031,"ip":0,"op":48.0000019550801,"w":400,"h":400,"nm":"Comp 1","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1L 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,289,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.928477328431,0.15922648112,0.723155182483,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":21,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":29,"s":[50]},{"t":36.0000014663101,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":21,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":29,"s":[25]},{"t":36.0000014663101,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":21.0000008553475,"op":55.0000022401959,"st":21.0000008553475,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 1L 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,289,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.928477328431,0.15922648112,0.723155182483,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[50]},{"t":32.0000013033867,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[25]},{"t":32.0000013033867,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":17.0000006924242,"op":51.0000020772726,"st":17.0000006924242,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shape Layer 1L","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,289,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.928477328431,0.15922648112,0.723155182483,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[50]},{"t":20.0000008146167,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[25]},{"t":20.0000008146167,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":5.00000020365417,"op":39.0000015885026,"st":5.00000020365417,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Shape Layer 2R 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,339,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.482111194087,0.119379574645,0.946568627451,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[50]},{"t":32.0000013033867,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[25]},{"t":32.0000013033867,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":17.0000006924242,"op":52.0000021180034,"st":17.0000006924242,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Shape Layer 2R 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,339,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.482111194087,0.119379574645,0.946568627451,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[50]},{"t":37.0000015070409,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[25]},{"t":37.0000015070409,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":22.0000008960784,"op":57.0000023216576,"st":22.0000008960784,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Shape Layer 2R","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,339,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.482111194087,0.119379574645,0.946568627451,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[50]},{"t":24.00000097754,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[25]},{"t":24.00000097754,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":9.00000036657752,"op":44.0000017921567,"st":9.00000036657752,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"Shape Layer 3L 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,389,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.096529941933,0.306382272758,0.935631127451,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[50]},{"t":37.0000015070409,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[25]},{"t":37.0000015070409,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":22.0000008960784,"op":56.0000022809268,"st":22.0000008960784,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Shape Layer 3L 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,389,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.096529941933,0.306382272758,0.935631127451,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":26,"s":[50]},{"t":33.0000013441176,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":26,"s":[25]},{"t":33.0000013441176,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":18.000000733155,"op":52.0000021180034,"st":18.000000733155,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"Shape Layer 3L","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,389,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.096529941933,0.306382272758,0.935631127451,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[50]},{"t":20.0000008146167,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[25]},{"t":20.0000008146167,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":5.00000020365417,"op":39.0000015885026,"st":5.00000020365417,"bm":0},{"ddd":0,"ind":10,"ty":4,"nm":"Shape Layer 4R 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,439,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.069122576246,0.9015625,0.230790366378,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":21,"s":[50]},{"t":28.0000011404634,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":21,"s":[25]},{"t":28.0000011404634,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":13.0000005295009,"op":48.0000019550801,"st":13.0000005295009,"bm":0},{"ddd":0,"ind":11,"ty":4,"nm":"Shape Layer 4R 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,439,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.069122576246,0.9015625,0.230790366378,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[50]},{"t":37.0000015070409,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[25]},{"t":37.0000015070409,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":22.0000008960784,"op":57.0000023216576,"st":22.0000008960784,"bm":0},{"ddd":0,"ind":12,"ty":4,"nm":"Shape Layer 4R","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,439,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.069122576246,0.9015625,0.230790366378,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[50]},{"t":18.000000733155,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[25]},{"t":18.000000733155,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3.00000012219251,"op":38.0000015477717,"st":3.00000012219251,"bm":0},{"ddd":0,"ind":13,"ty":4,"nm":"Shape Layer 5L 4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,179,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.078822124706,0.739201085708,0.917202818627,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[50]},{"t":27.0000010997325,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[25]},{"t":27.0000010997325,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":12.00000048877,"op":46.0000018736184,"st":12.00000048877,"bm":0},{"ddd":0,"ind":14,"ty":4,"nm":"Shape Layer 5L 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,179,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.078822124706,0.739201085708,0.917202818627,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":19,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":27,"s":[50]},{"t":34.0000013848484,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":19,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":27,"s":[25]},{"t":34.0000013848484,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":19.0000007738859,"op":53.0000021587343,"st":19.0000007738859,"bm":0},{"ddd":0,"ind":15,"ty":4,"nm":"Shape Layer 5L 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,179,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.078822124706,0.739201085708,0.917202818627,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":1,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[50]},{"t":16.0000006516934,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":1,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[25]},{"t":16.0000006516934,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":1.00000004073083,"op":35.0000014255792,"st":1.00000004073083,"bm":0},{"ddd":0,"ind":16,"ty":4,"nm":"Shape Layer 5L","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,179,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.078822124706,0.739201085708,0.917202818627,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[50]},{"t":23.0000009368092,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[25]},{"t":23.0000009368092,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":8.00000032584668,"op":42.0000017106951,"st":8.00000032584668,"bm":0},{"ddd":0,"ind":17,"ty":4,"nm":"Shape Layer 6R 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,239,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.524494904163,0.884911151961,0.087406652114,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[50]},{"t":30.0000012219251,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[25]},{"t":30.0000012219251,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":15.0000006109625,"op":50.0000020365418,"st":15.0000006109625,"bm":0},{"ddd":0,"ind":18,"ty":4,"nm":"Shape Layer 6R 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,239,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.524494904163,0.884911151961,0.087406652114,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[50]},{"t":37.0000015070409,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[25]},{"t":37.0000015070409,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":22.0000008960784,"op":57.0000023216576,"st":22.0000008960784,"bm":0},{"ddd":0,"ind":19,"ty":4,"nm":"Shape Layer 6R","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,239,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.524494904163,0.884911151961,0.087406652114,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[50]},{"t":20.0000008146167,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[25]},{"t":20.0000008146167,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":5.00000020365417,"op":40.0000016292334,"st":5.00000020365417,"bm":0},{"ddd":0,"ind":20,"ty":4,"nm":"Shape Layer 7L 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,489,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.611406034582,0.891253063725,0.090217530494,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[1],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0],"y":[0]},"t":21,"s":[50]},{"t":28.0000011404634,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.333],"y":[0]},"t":21,"s":[25]},{"t":28.0000011404634,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":13.0000005295009,"op":47.0000019143492,"st":13.0000005295009,"bm":0},{"ddd":0,"ind":21,"ty":4,"nm":"Shape Layer 7L 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,489,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.611406034582,0.891253063725,0.090217530494,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[1],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0],"y":[0]},"t":25,"s":[50]},{"t":32.0000013033867,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.333],"y":[0]},"t":25,"s":[25]},{"t":32.0000013033867,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":17.0000006924242,"op":51.0000020772726,"st":17.0000006924242,"bm":0},{"ddd":0,"ind":22,"ty":4,"nm":"Shape Layer 7L","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,489,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.611406034582,0.891253063725,0.090217530494,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[1],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0],"y":[0]},"t":12,"s":[50]},{"t":19.0000007738859,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.333],"y":[0]},"t":12,"s":[25]},{"t":19.0000007738859,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":4.00000016292334,"op":38.0000015477717,"st":4.00000016292334,"bm":0},{"ddd":0,"ind":23,"ty":4,"nm":"Shape Layer 8R 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,532,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.954534313725,0.324577391381,0.102267328898,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[50]},{"t":27.0000010997325,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[25]},{"t":27.0000010997325,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":12.00000048877,"op":47.0000019143492,"st":12.00000048877,"bm":0},{"ddd":0,"ind":24,"ty":4,"nm":"Shape Layer 8R 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,532,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.954534313725,0.324577391381,0.102267328898,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[50]},{"t":37.0000015070409,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[25]},{"t":37.0000015070409,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":22.0000008960784,"op":57.0000023216576,"st":22.0000008960784,"bm":0},{"ddd":0,"ind":25,"ty":4,"nm":"Shape Layer 8R","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[197.852,532,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[-100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-198,-154],[200,-154]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.954534313725,0.324577391381,0.102267328898,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[114.784,101.093],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[50]},{"t":21.0000008553475,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[25]},{"t":21.0000008553475,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6.00000024438501,"op":41.0000016699642,"st":6.00000024438501,"bm":0}],"markers":[]}
\ No newline at end of file
diff --git a/app/src/main/res/raw/lottie_loading.json b/app/src/main/res/raw/lottie_loading.json
new file mode 100644
index 0000000..6347c41
--- /dev/null
+++ b/app/src/main/res/raw/lottie_loading.json
@@ -0,0 +1 @@
+{"nm":"Main Scene","ddd":0,"h":100,"w":180,"meta":{"g":"@lottiefiles/creator 1.25.0"},"layers":[{"ty":0,"nm":" Comp 1","sr":1,"st":0,"op":31.00000126265584,"ip":4.00000016292334,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[63,24]},"s":{"a":0,"k":[250,250]},"sk":{"a":0,"k":0},"p":{"a":0,"k":[0,0]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":200,"h":100,"refId":"precomp_Comp 1_ff1747d4-671c-4717-aeaa-23476b2a6335","ind":1}],"v":"5.7.0","fr":29.9700012207031,"op":31,"ip":0,"assets":[{"nm":"","id":"comp_0_85618a3e-3dd7-43d3-a8a7-28422c05ed7d","layers":[{"ty":4,"nm":"Shape Layer 2","sr":1,"st":-4.00000016292334,"op":90.00000366577514,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-32.5,5,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[20,20,100],"t":9.000000162923339},{"s":[100,100,100],"t":18.00000073315504}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[72,74.8,0],"t":10.000000162923339},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[72,62.8,0],"t":16.00000016292334},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[72,66.1,0],"t":22.00000016292334},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[72,64.5,0],"t":28.00000016292334},{"s":[72,64.8,0],"t":34.00000138484844}],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":9.000000162923339},{"s":[100],"t":18.00000073315504}],"ix":11}},"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Ellipse 1","ix":1,"cix":2,"np":3,"it":[{"ty":"el","bm":0,"hd":false,"mn":"ADBE Vector Shape - Ellipse","nm":"Ellipse Path 1","d":1,"p":{"a":0,"k":[0,0],"ix":3},"s":{"a":0,"k":[15,15],"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"c":{"a":0,"k":[1,1,1],"ix":3}},{"ty":"fl","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Fill","nm":"Fill 1","c":{"a":0,"k":[0.498,0.4902,0.5176],"ix":4},"r":1,"o":{"a":0,"k":100,"ix":5}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[-32.618,4.916],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]}],"ind":1},{"ty":4,"nm":"Shape Layer 1","sr":1,"st":0,"op":94.00000382869844,"ip":4.00000016292334,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-32.5,5,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":6.00000016292334},{"s":[120,120,100],"t":15.000000610962541}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[72,64.75,0],"t":4.00000016292334},{"s":[72,44.8,0],"t":13.00000052950086}],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":4.00000016292334},{"s":[0],"t":13.00000052950086}],"ix":11}},"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Ellipse 1","ix":1,"cix":2,"np":3,"it":[{"ty":"el","bm":0,"hd":false,"mn":"ADBE Vector Shape - Ellipse","nm":"Ellipse Path 1","d":1,"p":{"a":0,"k":[0,0],"ix":3},"s":{"a":0,"k":[15,15],"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"c":{"a":0,"k":[1,1,1],"ix":3}},{"ty":"fl","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Fill","nm":"Fill 1","c":{"a":0,"k":[0.498,0.4902,0.5176],"ix":4},"r":1,"o":{"a":0,"k":100,"ix":5}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[-32.618,4.916],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]}],"ind":2}]},{"nm":"Comp 1","id":"precomp_Comp 1_ff1747d4-671c-4717-aeaa-23476b2a6335","fr":29.9700012207031,"layers":[{"ty":0,"nm":"åä ̧aç1","sr":1,"st":0,"op":94.00000382869844,"ip":4.00000016292334,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[50,50,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[49.75,38.5,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":0,"k":100,"ix":11}},"w":100,"h":100,"refId":"comp_0_85618a3e-3dd7-43d3-a8a7-28422c05ed7d","ind":1},{"ty":0,"nm":"åä ̧aç1","sr":1,"st":2.00000008146167,"op":96.00000391016015,"ip":6.00000024438501,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[50,50,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[74.75,38.5,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":0,"k":100,"ix":11}},"w":100,"h":100,"refId":"comp_0_85618a3e-3dd7-43d3-a8a7-28422c05ed7d","ind":2},{"ty":4,"nm":"Shape Layer 3","sr":1,"st":0,"op":7.000000285115849,"ip":4.00000016292334,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-32.5,5,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[96.75,53.25,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":0,"k":100,"ix":11}},"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Ellipse 1","ix":1,"cix":2,"np":3,"it":[{"ty":"el","bm":0,"hd":false,"mn":"ADBE Vector Shape - Ellipse","nm":"Ellipse Path 1","d":1,"p":{"a":0,"k":[0,0],"ix":3},"s":{"a":0,"k":[15,15],"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"c":{"a":0,"k":[1,1,1],"ix":3}},{"ty":"fl","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Fill","nm":"Fill 1","c":{"a":0,"k":[0.498,0.4902,0.5176],"ix":4},"r":1,"o":{"a":0,"k":100,"ix":5}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[-32.618,4.916],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]}],"ind":3},{"ty":4,"nm":"Shape Layer 2","sr":1,"st":0,"op":10.000000407308349,"ip":4.00000016292334,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-32.5,5,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[121.75,53.25,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":0,"k":100,"ix":11}},"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Ellipse 1","ix":1,"cix":2,"np":3,"it":[{"ty":"el","bm":0,"hd":false,"mn":"ADBE Vector Shape - Ellipse","nm":"Ellipse Path 1","d":1,"p":{"a":0,"k":[0,0],"ix":3},"s":{"a":0,"k":[15,15],"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"c":{"a":0,"k":[1,1,1],"ix":3}},{"ty":"fl","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Fill","nm":"Fill 1","c":{"a":0,"k":[0.498,0.4902,0.5176],"ix":4},"r":1,"o":{"a":0,"k":100,"ix":5}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[-32.618,4.916],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]}],"ind":4},{"ty":0,"nm":"åä ̧aç1","sr":1,"st":4.00000016292334,"op":98.00000399162184,"ip":8.00000032584668,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[50,50,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[99.75,38.5,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":0,"k":100,"ix":11}},"w":100,"h":100,"refId":"comp_0_85618a3e-3dd7-43d3-a8a7-28422c05ed7d","ind":5}]}]}
\ No newline at end of file
diff --git a/app/src/main/res/raw/lottie_page_loading.json b/app/src/main/res/raw/lottie_page_loading.json
new file mode 100644
index 0000000..016f5fd
--- /dev/null
+++ b/app/src/main/res/raw/lottie_page_loading.json
@@ -0,0 +1 @@
+{"v":"5.5.7","meta":{"g":"LottieFiles AE 0.1.21","a":"Syed Mubarak","k":"Loading","d":"Loading Animation","tc":"#FFFFFF"},"fr":29.9700012207031,"ip":0,"op":71.0000028918893,"w":1080,"h":1080,"nm":"Loading2","ddd":0,"assets":[{"id":"image_0","w":1080,"h":1080,"u":"","p":"","e":1},{"id":"image_1","w":1080,"h":1080,"u":"","p":"","e":1},{"id":"image_2","w":1080,"h":1080,"u":"","p":"","e":1},{"id":"image_3","w":1080,"h":1080,"u":"","p":"","e":1},{"id":"image_4","w":1080,"h":1080,"u":"","p":"","e":1},{"id":"image_5","w":1080,"h":1080,"u":"","p":"","e":1},{"id":"image_6","w":1080,"h":1080,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"Cup/loading animation.ai","cl":"ai","refId":"image_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[540,540,0],"ix":2},"a":{"a":0,"k":[540,540,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"cup shape/loading animation.ai","cl":"ai","td":1,"refId":"image_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[540,540,0],"ix":2},"a":{"a":0,"k":[540,540,0],"ix":1},"s":{"a":0,"k":[94.556,94.556,100],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"wave/loading animation.ai","cl":"ai","tt":1,"refId":"image_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.607,"y":0.595},"o":{"x":0.529,"y":0},"t":0,"s":[660.805,540,0],"to":[-30.035,0,0],"ti":[68.54,0,0]},{"i":{"x":0.507,"y":1},"o":{"x":0.21,"y":0.779},"t":30,"s":[469.965,532,0],"to":[-30.703,0,0],"ti":[13.455,0,0]},{"t":70.0000028511585,"s":[399.866,540,0]}],"ix":2},"a":{"a":0,"k":[540,540,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":2,"nm":"dot 1/loading animation.ai","cl":"ai","refId":"image_3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[608,678,0],"ix":2},"a":{"a":0,"k":[608,678,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":0,"s":[100,100,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":10,"s":[131.78,131.78,100]},{"t":20.0000008146167,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":2,"nm":"dot 2/loading animation.ai","cl":"ai","refId":"image_4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[646,676,0],"ix":2},"a":{"a":0,"k":[646,676,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":25,"s":[100,100,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":35,"s":[131.78,131.78,100]},{"t":45.0000018328876,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":2,"nm":"Dot 3/loading animation.ai","cl":"ai","refId":"image_5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[684.966,675.302,0],"ix":2},"a":{"a":0,"k":[684.966,675.302,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":50,"s":[100,100,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":60,"s":[131.78,131.78,100]},{"t":70.0000028511585,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":2,"nm":"text/loading animation.ai","cl":"ai","refId":"image_6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[540,540,0],"ix":2},"a":{"a":0,"k":[540,540,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":71.0000028918893,"st":0,"bm":0}],"markers":[]}
\ No newline at end of file
diff --git a/app/src/main/res/values/attrs.xml b/app/src/main/res/values/attrs.xml
index a942b60..ca76bf8 100644
--- a/app/src/main/res/values/attrs.xml
+++ b/app/src/main/res/values/attrs.xml
@@ -1,11 +1,18 @@
+
-
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml
index 449df2b..a0dbd7e 100644
--- a/app/src/main/res/values/colors.xml
+++ b/app/src/main/res/values/colors.xml
@@ -2,7 +2,9 @@
#03a9f4
#0288d1
+ #f6f6f6
#ffab40
+ #2c2c2c
#ffc77f
#30000000
@android:color/white
diff --git a/app/src/main/res/values/ids.xml b/app/src/main/res/values/ids.xml
new file mode 100644
index 0000000..2291a30
--- /dev/null
+++ b/app/src/main/res/values/ids.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index 997224b..7f7821a 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -1,5 +1,6 @@
易开发
+ 该功能需要读写内部存储权限,点击前往设置
GuideActivity
Settings
上一步
@@ -11,18 +12,19 @@
无障碍功能已开启
该设备未root,无法开启
基本设置
- 高级功能设置
+ 高级功能
开启无障碍辅助
开启悬浮窗权限
- 推荐开启,用于获取应用数据库等信息,不开启部分信息无法显示,Xposed无法使用
+ 推荐开启,用于获取应用SharedPreference、数据库等数据,关闭后部分信息无法显示
开启root权限
启用Xposed特性
支持脱壳,更多功能持续开发中,点击进入管理
辅助功能用于界面布局分析等
关于
软件开发中,交流群:627962572
- 界面分析
- 点击分析
+ 界面布局
+ 功能需要root权限,请开启后使用!
+ 点击查看
首次安装时间
最后更新时间
守护进程
@@ -47,24 +49,23 @@
基本信息
加固类型
正在分析
- 请稍后...
+ 请稍后...
正在获取应用信息
请开启无障碍功能
表
该表数据为空
获取数据库信息失败
- 获取中...
+ 获取中...
数据库列表
无
应用设置
- 备份Apk文件
- 备份data目录
+ 备份应用
+ 还原应用
重启应用
停止应用
清理应用数据
卸载应用
请开启root权限
- 部分功能需要root权限,请打开后使用!
是否清除数据和缓存?
清理完成
是否停止该应用?
@@ -74,23 +75,40 @@
备份失败
备份成功,文件已备份到内部存储下com.wrbug.developerHelper目录中,是否立即分享给好友?
分享失败
- 导出脱壳数据
正在打包文件
导出失败!
- 无脱壳文件
- Xposed设置
- 管理、移除需要脱壳的应用
- 应用脱壳
移除该项
请先开启Xposed设置
检查更新
- 正在检查新版本...
+ 正在检查新版本...
暂无新版本
- 检查失败...
+ 检查失败...
发现新版本
下载
- 应用Xposed功能管理
- 应用Xposed总开关,开启后其他xposed模块才会生效
- DataFinder
- DataFinder服务,开启后可以在浏览器访问数据库等信息,需要wifi环境
+ 通知栏常驻
+ 用于进程保活,当应用经常被杀死时,请打开该功能
+ 请选择需要备份的内容,备份所有数据支持无损恢复
+ 正在备份数据
+ 备份失败
+ 跳过该备份
+ 备份正在进行中,请勿关闭该窗口
+ 应用安装包
+ Data目录
+ Android/Data目录
+ 备份已完成,您可以前往 %1$s 目录查看备份文件
+ 等待中
+ 打包备份文件
+ 暂无数据
+ %1$d个备份
+ 上一次:%1$s
+ 其他功能
+ 无需备份
+ 备份管理
+ 管理/恢复已备份的应用
+ 已取消
+ %1$s 的备份
+ 备注
+ 删除
+ 是否删除该备份?
+ 删除备份失败,请重试
diff --git a/app/src/main/res/values/styles.xml b/app/src/main/res/values/styles.xml
index ddc3fd5..906b089 100644
--- a/app/src/main/res/values/styles.xml
+++ b/app/src/main/res/values/styles.xml
@@ -7,6 +7,7 @@
- @color/colorPrimaryDark
- @color/colorAccent
- @android:color/white
+ - @color/colorWindowBackground
@@ -36,6 +37,25 @@
- @android:color/white
+
+
+
+
+
+
+
+
+
diff --git a/basecommon/.gitignore b/basecommon/.gitignore
deleted file mode 100644
index 796b96d..0000000
--- a/basecommon/.gitignore
+++ /dev/null
@@ -1 +0,0 @@
-/build
diff --git a/basecommon/build.gradle b/basecommon/build.gradle
deleted file mode 100644
index 922788e..0000000
--- a/basecommon/build.gradle
+++ /dev/null
@@ -1,41 +0,0 @@
-apply plugin: 'com.android.library'
-apply plugin: 'kotlin-android'
-apply plugin: 'kotlin-kapt'//kapt 插件
-
-android {
- compileSdkVersion 28
-
-
-
- defaultConfig {
- minSdkVersion 21
- targetSdkVersion 28
- versionCode 1
- versionName "1.0"
- testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
- }
-
- buildTypes {
- release {
- minifyEnabled false
- proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- }
- }
-}
-def lifecycle_version = "2.0.0"
-dependencies {
- implementation fileTree(include: ['*.jar'], dir: 'libs')
- implementation 'androidx.appcompat:appcompat:1.0.2'
- testImplementation 'junit:junit:4.12'
- androidTestImplementation 'com.android.support.test:runner:1.0.2'
- androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
- implementation 'com.google.android.material:material:1.1.0-alpha01'
- implementation 'android.arch.lifecycle:extensions:1.1.1'
- annotationProcessor "android.arch.lifecycle:compiler:1.1.1"
- implementation project(':commonutil')
- implementation project(':commonwidget')
-}
-repositories {
- mavenCentral()
-}
diff --git a/basecommon/proguard-rules.pro b/basecommon/proguard-rules.pro
deleted file mode 100644
index f1b4245..0000000
--- a/basecommon/proguard-rules.pro
+++ /dev/null
@@ -1,21 +0,0 @@
-# Add project specific ProGuard rules here.
-# You can control the set of applied configuration files using the
-# proguardFiles setting in build.gradle.
-#
-# For more details, see
-# http://developer.android.com/guide/developing/tools/proguard.html
-
-# If your project uses WebView with JS, uncomment the following
-# and specify the fully qualified class name to the JavaScript interface
-# class:
-#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
-# public *;
-#}
-
-# Uncomment this to preserve the line number information for
-# debugging stack traces.
-#-keepattributes SourceFile,LineNumberTable
-
-# If you keep the line number information, uncomment this to
-# hide the original source file name.
-#-renamesourcefileattribute SourceFile
diff --git a/basecommon/src/androidTest/java/com/wrbug/developerhelper/basecommon/ExampleInstrumentedTest.java b/basecommon/src/androidTest/java/com/wrbug/developerhelper/basecommon/ExampleInstrumentedTest.java
deleted file mode 100644
index 7e11c97..0000000
--- a/basecommon/src/androidTest/java/com/wrbug/developerhelper/basecommon/ExampleInstrumentedTest.java
+++ /dev/null
@@ -1,26 +0,0 @@
-package com.wrbug.developerhelper.basecommon;
-
-import android.content.Context;
-import android.support.test.InstrumentationRegistry;
-import android.support.test.runner.AndroidJUnit4;
-
-import org.junit.Test;
-import org.junit.runner.RunWith;
-
-import static org.junit.Assert.*;
-
-/**
- * Instrumented test, which will execute on an Android device.
- *
- * @see Testing documentation
- */
-@RunWith(AndroidJUnit4.class)
-public class ExampleInstrumentedTest {
- @Test
- public void useAppContext() {
- // Context of the app under test.
- Context appContext = InstrumentationRegistry.getTargetContext();
-
- assertEquals("com.wrbug.developerhelper.basecommon.test", appContext.getPackageName());
- }
-}
diff --git a/basecommon/src/main/AndroidManifest.xml b/basecommon/src/main/AndroidManifest.xml
deleted file mode 100644
index 443e74f..0000000
--- a/basecommon/src/main/AndroidManifest.xml
+++ /dev/null
@@ -1,2 +0,0 @@
-
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/AppCompatActivityExt.kt b/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/AppCompatActivityExt.kt
deleted file mode 100644
index bd807e8..0000000
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/AppCompatActivityExt.kt
+++ /dev/null
@@ -1,67 +0,0 @@
-/*
- * Copyright (C) 2017 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-package com.wrbug.developerhelper.basecommon
-
-import android.content.Intent
-import androidx.annotation.IdRes
-import androidx.appcompat.app.ActionBar
-import androidx.appcompat.app.AppCompatActivity
-import androidx.fragment.app.FragmentManager
-import androidx.fragment.app.FragmentTransaction
-import androidx.lifecycle.ViewModel
-import androidx.lifecycle.ViewModelProviders
-import com.wrbug.developerhelper.basecommon.activityresultcallback.ActResultRequest
-import com.wrbug.developerhelper.basecommon.activityresultcallback.ActivityResultCallback
-
-/**
- * Various extension functions for AppCompatActivity.
- */
-
-
-fun AppCompatActivity.setupActionBar(@IdRes toolbarId: Int, action: ActionBar.() -> Unit) {
- setSupportActionBar(findViewById(toolbarId))
- supportActionBar?.run {
- setDisplayHomeAsUpEnabled(false)
- action()
- }
-}
-
-fun AppCompatActivity.obtainViewModel(viewModelClass: Class) =
- ViewModelProviders.of(this).get(viewModelClass)
-
-/**
- * Runs a FragmentTransaction, then calls commit().
- */
-private inline fun FragmentManager.transact(action: FragmentTransaction.() -> Unit) {
- beginTransaction().apply {
- action()
- }.commit()
-}
-
-fun AppCompatActivity.startActivityForResult(intent: Intent, callback: ActivityResultCallback) {
- ActResultRequest(this).startForResult(intent, callback)
-}
-
-fun AppCompatActivity.startActivityForResultOk(
- intent: Intent,
- action: Intent.() -> Unit
-) {
- ActResultRequest(this).startForResult(intent, object : ActivityResultCallback() {
- override fun onActivityResultOk(data: Intent) {
- action(data)
- }
- })
-}
diff --git a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/entry/HierarchyNode.kt b/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/entry/HierarchyNode.kt
deleted file mode 100644
index 795b145..0000000
--- a/basecommon/src/main/java/com/wrbug/developerhelper/basecommon/entry/HierarchyNode.kt
+++ /dev/null
@@ -1,100 +0,0 @@
-package com.wrbug.developerhelper.basecommon.entry
-
-import android.graphics.Rect
-import android.os.Parcel
-import android.os.Parcelable
-import com.google.gson.reflect.TypeToken
-import com.wrbug.developerhelper.commonutil.JsonHelper
-import java.io.Serializable
-
-@Suppress("NULLABILITY_MISMATCH_BASED_ON_JAVA_ANNOTATIONS")
-class HierarchyNode() : Parcelable, Serializable {
- var id: Long = -1L
- var screenBounds: Rect? = null
- var parentBounds: Rect? = null
- var checkable: Boolean = false
- var checked: Boolean = false
- var widget: String = ""
- var clickable: Boolean = false
- var contentDesc: String = ""
- var enabled: Boolean = false
- var focusable: Boolean = false
- var focused: Boolean = false
- var index: String = ""
- var longClickable: Boolean = false
- var packageName: String = ""
- var password: Boolean = false
- var scrollable: Boolean = false
- var selected: Boolean = false
- var text: String = ""
- var resourceId: String = ""
- var idHex: String? = null
- var parentId: Long = -1L
- var childId: ArrayList = arrayListOf()
-
- constructor(parcel: Parcel) : this() {
- id = parcel.readLong()
- screenBounds = parcel.readParcelable(Rect::class.java.classLoader)
- parentBounds = parcel.readParcelable(Rect::class.java.classLoader)
- checkable = parcel.readByte() != 0.toByte()
- checked = parcel.readByte() != 0.toByte()
- widget = parcel.readString()
- clickable = parcel.readByte() != 0.toByte()
- contentDesc = parcel.readString()
- enabled = parcel.readByte() != 0.toByte()
- focusable = parcel.readByte() != 0.toByte()
- focused = parcel.readByte() != 0.toByte()
- index = parcel.readString()
- longClickable = parcel.readByte() != 0.toByte()
- packageName = parcel.readString()
- password = parcel.readByte() != 0.toByte()
- scrollable = parcel.readByte() != 0.toByte()
- selected = parcel.readByte() != 0.toByte()
- text = parcel.readString()
- resourceId = parcel.readString()
- idHex = parcel.readString()
- parentId = parcel.readLong()
- childId = JsonHelper.fromJson(parcel.readString(), object : TypeToken>() {}.type) ?:
- arrayListOf()
- }
-
- override fun writeToParcel(parcel: Parcel, flags: Int) {
- parcel.writeLong(id)
- parcel.writeParcelable(screenBounds, flags)
- parcel.writeParcelable(parentBounds, flags)
- parcel.writeByte(if (checkable) 1 else 0)
- parcel.writeByte(if (checked) 1 else 0)
- parcel.writeString(widget)
- parcel.writeByte(if (clickable) 1 else 0)
- parcel.writeString(contentDesc)
- parcel.writeByte(if (enabled) 1 else 0)
- parcel.writeByte(if (focusable) 1 else 0)
- parcel.writeByte(if (focused) 1 else 0)
- parcel.writeString(index)
- parcel.writeByte(if (longClickable) 1 else 0)
- parcel.writeString(packageName)
- parcel.writeByte(if (password) 1 else 0)
- parcel.writeByte(if (scrollable) 1 else 0)
- parcel.writeByte(if (selected) 1 else 0)
- parcel.writeString(text)
- parcel.writeString(resourceId)
- parcel.writeString(idHex)
- parcel.writeLong(parentId)
- parcel.writeString(JsonHelper.toJson(childId))
- }
-
- override fun describeContents(): Int {
- return 0
- }
-
- companion object CREATOR : Parcelable.Creator {
- override fun createFromParcel(parcel: Parcel): HierarchyNode {
- return HierarchyNode(parcel)
- }
-
- override fun newArray(size: Int): Array {
- return arrayOfNulls(size)
- }
- }
-
-}
\ No newline at end of file
diff --git a/basecommon/src/main/res/values/strings.xml b/basecommon/src/main/res/values/strings.xml
deleted file mode 100644
index 717ea7a..0000000
--- a/basecommon/src/main/res/values/strings.xml
+++ /dev/null
@@ -1,3 +0,0 @@
-
- baseCommon
-
diff --git a/basemoduleimport/.gitignore b/basemoduleimport/.gitignore
deleted file mode 100644
index 796b96d..0000000
--- a/basemoduleimport/.gitignore
+++ /dev/null
@@ -1 +0,0 @@
-/build
diff --git a/basemoduleimport/build.gradle b/basemoduleimport/build.gradle
deleted file mode 100644
index 7ae3275..0000000
--- a/basemoduleimport/build.gradle
+++ /dev/null
@@ -1,43 +0,0 @@
-apply plugin: 'com.android.library'
-apply plugin: 'kotlin-android'
-
-android {
- compileSdkVersion 28
-
-
-
- defaultConfig {
- minSdkVersion 21
- targetSdkVersion 28
- versionCode 1
- versionName "1.0"
-
- testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
-
- }
-
- buildTypes {
- release {
- minifyEnabled false
- proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- }
- }
-
-}
-
-dependencies {
- api fileTree(include: ['*.jar'], dir: 'libs')
- api 'androidx.appcompat:appcompat:1.0.2'
- api 'androidx.constraintlayout:constraintlayout:1.1.3'
- api 'androidx.legacy:legacy-support-v4:1.0.0'
- api 'androidx.recyclerview:recyclerview:1.0.0'
- api 'com.google.android.material:material:1.1.0-alpha02'
- api project(':commonutil')
- api project(':commonwidget')
- api project(':mmkv')
- api project(':basecommon')
- api "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
-}
-repositories {
- mavenCentral()
-}
diff --git a/basemoduleimport/proguard-rules.pro b/basemoduleimport/proguard-rules.pro
deleted file mode 100644
index f1b4245..0000000
--- a/basemoduleimport/proguard-rules.pro
+++ /dev/null
@@ -1,21 +0,0 @@
-# Add project specific ProGuard rules here.
-# You can control the set of applied configuration files using the
-# proguardFiles setting in build.gradle.
-#
-# For more details, see
-# http://developer.android.com/guide/developing/tools/proguard.html
-
-# If your project uses WebView with JS, uncomment the following
-# and specify the fully qualified class name to the JavaScript interface
-# class:
-#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
-# public *;
-#}
-
-# Uncomment this to preserve the line number information for
-# debugging stack traces.
-#-keepattributes SourceFile,LineNumberTable
-
-# If you keep the line number information, uncomment this to
-# hide the original source file name.
-#-renamesourcefileattribute SourceFile
diff --git a/basemoduleimport/src/androidTest/java/com/wrbug/developerhelper/basewidgetimport/ExampleInstrumentedTest.java b/basemoduleimport/src/androidTest/java/com/wrbug/developerhelper/basewidgetimport/ExampleInstrumentedTest.java
deleted file mode 100644
index 38b9eb4..0000000
--- a/basemoduleimport/src/androidTest/java/com/wrbug/developerhelper/basewidgetimport/ExampleInstrumentedTest.java
+++ /dev/null
@@ -1,26 +0,0 @@
-package com.wrbug.developerhelper.basewidgetimport;
-
-import android.content.Context;
-import android.support.test.InstrumentationRegistry;
-import android.support.test.runner.AndroidJUnit4;
-
-import org.junit.Test;
-import org.junit.runner.RunWith;
-
-import static org.junit.Assert.*;
-
-/**
- * Instrumented test, which will execute on an Android device.
- *
- * @see Testing documentation
- */
-@RunWith(AndroidJUnit4.class)
-public class ExampleInstrumentedTest {
- @Test
- public void useAppContext() {
- // Context of the app under test.
- Context appContext = InstrumentationRegistry.getTargetContext();
-
- assertEquals("com.wrbug.developerhelper.basewidgetimport.test", appContext.getPackageName());
- }
-}
diff --git a/basemoduleimport/src/main/AndroidManifest.xml b/basemoduleimport/src/main/AndroidManifest.xml
deleted file mode 100644
index 27a1cb8..0000000
--- a/basemoduleimport/src/main/AndroidManifest.xml
+++ /dev/null
@@ -1,2 +0,0 @@
-
diff --git a/basemoduleimport/src/main/res/values/strings.xml b/basemoduleimport/src/main/res/values/strings.xml
deleted file mode 100644
index 2e0519d..0000000
--- a/basemoduleimport/src/main/res/values/strings.xml
+++ /dev/null
@@ -1,3 +0,0 @@
-
- basewidgetimport
-
diff --git a/build.gradle b/build.gradle
index 9bf0edc..183b457 100644
--- a/build.gradle
+++ b/build.gradle
@@ -1,35 +1,11 @@
-// Top-level build file where you can add configuration options common to all sub-projects/modules.
-
-buildscript {
- ext.kotlin_version = '1.3.21'
- ext.gradle_version = '3.2.0'
- ext.enableNdk=false
- repositories {
- google()
- jcenter()
- maven {
- url "https://jitpack.io"
- }
- }
- dependencies {
- classpath "com.android.tools.build:gradle:$gradle_version"
- classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
-
- // NOTE: Do not place your application dependencies here; they belong
- // in the individual module build.gradle files
- }
-}
-allprojects {
- repositories {
- google()
- jcenter()
- maven {
- url "https://jitpack.io"
- }
- }
+// Top-level build file where you can add configuration options common to all sub-projects/modules.
+plugins {
+ id 'com.android.application' version '8.2.2' apply false
+ id 'com.android.library' version '8.2.2' apply false
+ id 'org.jetbrains.kotlin.android' version '1.9.22' apply false
}
task clean(type: Delete) {
delete rootProject.buildDir
-}
+}
\ No newline at end of file
diff --git a/common.gradle b/common.gradle
new file mode 100644
index 0000000..a84c948
--- /dev/null
+++ b/common.gradle
@@ -0,0 +1,19 @@
+apply plugin: 'kotlin-parcelize'
+
+android {
+ buildFeatures {
+ viewBinding true
+ }
+ kotlinOptions {
+ jvmTarget = '1.8'
+ }
+ lintOptions {
+ abortOnError true
+ disable 'UnusedResources'
+ ignoreWarnings true
+ }
+ compileOptions {
+ sourceCompatibility JavaVersion.VERSION_1_8
+ targetCompatibility JavaVersion.VERSION_1_8
+ }
+}
\ No newline at end of file
diff --git a/commonutil/build.gradle b/commonutil/build.gradle
index 1f7cdb6..b39d840 100644
--- a/commonutil/build.gradle
+++ b/commonutil/build.gradle
@@ -1,14 +1,13 @@
apply plugin: 'com.android.library'
apply plugin: 'kotlin-android'
+apply from: "${rootDir}/common.gradle"
android {
- compileSdkVersion 28
-
-
-
+ namespace "com.wrbug.developerhelper.commonutil"
+ compileSdk 34
defaultConfig {
- minSdkVersion 21
- targetSdkVersion 28
+ minSdkVersion 23
+ targetSdkVersion 34
versionCode 1
versionName "1.0"
@@ -27,12 +26,10 @@ android {
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
- api 'com.jaredrummler:android-shell:1.0.0'
- implementation 'androidx.appcompat:appcompat:1.0.2'
+ implementation 'androidx.appcompat:appcompat:1.7.0'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test:runner:1.1.0-alpha3'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.0-alpha3'
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
implementation project(':mmkv')
api "org.jetbrains.anko:anko-commons:0.10.8"
api 'com.google.code.gson:gson:2.8.5'
@@ -42,6 +39,3 @@ dependencies {
api "io.reactivex.rxjava3:rxjava:3.0.3"
api 'io.reactivex.rxjava3:rxandroid:3.0.0'
}
-repositories {
- mavenCentral()
-}
diff --git a/commonutil/src/androidTest/java/com/wrbug/developerhelper/commonutil/ExampleInstrumentedTest.java b/commonutil/src/androidTest/java/com/wrbug/developerhelper/commonutil/ExampleInstrumentedTest.java
deleted file mode 100644
index 87b387b..0000000
--- a/commonutil/src/androidTest/java/com/wrbug/developerhelper/commonutil/ExampleInstrumentedTest.java
+++ /dev/null
@@ -1,26 +0,0 @@
-package com.wrbug.developerhelper.commonutil;
-
-import android.content.Context;
-import android.support.test.InstrumentationRegistry;
-import android.support.test.runner.AndroidJUnit4;
-
-import org.junit.Test;
-import org.junit.runner.RunWith;
-
-import static org.junit.Assert.*;
-
-/**
- * Instrumented test, which will execute on an Android device.
- *
- * @see Testing documentation
- */
-@RunWith(AndroidJUnit4.class)
-public class ExampleInstrumentedTest {
- @Test
- public void useAppContext() {
- // Context of the app under test.
- Context appContext = InstrumentationRegistry.getTargetContext();
-
- assertEquals("com.wrbug.developerhelper.commonutil.test", appContext.getPackageName());
- }
-}
diff --git a/commonutil/src/main/AndroidManifest.xml b/commonutil/src/main/AndroidManifest.xml
index 68282c7..cf5f98d 100644
--- a/commonutil/src/main/AndroidManifest.xml
+++ b/commonutil/src/main/AndroidManifest.xml
@@ -1,2 +1,8 @@
+ xmlns:tools="http://schemas.android.com/tools"
+ package="com.wrbug.developerhelper.commonutil">
+
+
+
diff --git a/commonutil/src/main/assets/busybox_arm64 b/commonutil/src/main/assets/busybox_arm64
new file mode 100644
index 0000000..cf9e5be
Binary files /dev/null and b/commonutil/src/main/assets/busybox_arm64 differ
diff --git a/commonutil/src/main/assets/busybox_x86_64 b/commonutil/src/main/assets/busybox_x86_64
new file mode 100755
index 0000000..77523e2
Binary files /dev/null and b/commonutil/src/main/assets/busybox_x86_64 differ
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppInfoManager.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppInfoManager.kt
index ab0b1d1..7f6e08e 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppInfoManager.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppInfoManager.kt
@@ -1,9 +1,12 @@
package com.wrbug.developerhelper.commonutil
+import android.app.ActivityManager
+import android.content.Context
import com.wrbug.developerhelper.commonutil.entity.ApkInfo
import com.wrbug.developerhelper.commonutil.shell.ShellManager
import java.io.File
+
object AppInfoManager {
private val appMap = HashMap()
@@ -31,8 +34,27 @@ object AppInfoManager {
}
+ fun getTopActivityClassName(context: Context): String? {
+ var topActivityClass: String? = null
+ val activityManager = context.getSystemService(Context.ACTIVITY_SERVICE) as ActivityManager
+ try {
+ val runningTaskInfos = activityManager.getRunningTasks(1)
+ if (runningTaskInfos != null && runningTaskInfos.size> 0) {
+ val f = runningTaskInfos[0].topActivity
+ topActivityClass = f!!.className
+ }
+ } catch (e: Exception) {
+ }
+ return topActivityClass
+ }
+
+
fun getSharedPreferencesFiles(packageName: String): Array {
- val path = "/data/data/$packageName/shared_prefs"
+ return getSharedPreferencesFiles(Constant.dataDir, packageName)
+ }
+
+ private fun getSharedPreferencesFiles(dir: String, packageName: String): Array {
+ val path = "$dir/$packageName/shared_prefs"
val list = ShellManager.lsDir(path)
val files = ArrayList()
for (file in list) {
@@ -46,5 +68,4 @@ object AppInfoManager {
return files.toTypedArray()
}
-
}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppManagerUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppManagerUtils.kt
index bfb2c8c..5d7fb28 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppManagerUtils.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/AppManagerUtils.kt
@@ -23,7 +23,7 @@ object AppManagerUtils {
}
fun restartApp(context: Context, packageName: String) {
- if (!AppManagerUtils.forceStopApp(packageName)) {
+ if (!forceStopApp(packageName)) {
Toast.makeText(context, "重启失败", Toast.LENGTH_SHORT).show()
return
}
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ClipboardUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ClipboardUtils.kt
index 79e719d..4f54f22 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ClipboardUtils.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ClipboardUtils.kt
@@ -31,7 +31,7 @@ object ClipboardUtils {
try {
val cm = context.getSystemService(Context.CLIPBOARD_SERVICE) as ClipboardManager
// 将文本内容放到系统剪贴板里。
- cm.primaryClip = ClipData.newPlainText(null, text)
+ cm.setPrimaryClip(ClipData.newPlainText(null, text))
} catch (t: Throwable) {
}
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/CommonUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/CommonUtils.kt
index f112bd2..d7430fa 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/CommonUtils.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/CommonUtils.kt
@@ -2,10 +2,41 @@ package com.wrbug.developerhelper.commonutil
import android.app.Application
import android.content.Context
+import com.wrbug.developerhelper.commonutil.entity.CpuABI
+import com.wrbug.developerhelper.commonutil.shell.ShellUtils
+import org.jetbrains.anko.doAsync
+
object CommonUtils {
lateinit var application: Application
fun register(ctx: Context) {
application = ctx.applicationContext as Application
+ releaseBusyBox(ctx)
+ }
+
+ private fun releaseBusyBox(ctx: Context) {
+ doAsync {
+ val name = when (getCPUABI()) {
+ CpuABI.ARM -> "busybox_arm64"
+ CpuABI.X86 -> "busybox_x86_64"
+ }
+ val data = ctx.resources.assets.open(name).readBytes()
+ val file = ShellUtils.busyBoxFile
+ file.writeBytes(data)
+ ShellUtils.run("chmod +x " + file.absolutePath)
+ }
+ }
+
+
+ private fun getCPUABI(): CpuABI {
+ val result = ShellUtils.run("getprop ro.product.cpu.abi")
+ if (result.isSuccessful.not()) {
+ return CpuABI.ARM
+ }
+ return if (result.getStdout().contains("x86")) {
+ CpuABI.X86
+ } else {
+ CpuABI.ARM
+ }
}
}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/Constant.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/Constant.kt
new file mode 100644
index 0000000..373c805
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/Constant.kt
@@ -0,0 +1,20 @@
+package com.wrbug.developerhelper.commonutil
+
+import android.os.Build
+
+object Constant {
+ private const val DATA_MIRROR_DIR = "/data_mirror/data_ce/null/0"
+ private const val DATA_DIR = "/data/data"
+
+ fun getDataDir(packageName: String): String {
+ return "$dataDir/$packageName"
+ }
+
+ val dataDir by lazy {
+ if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.R) {
+ DATA_MIRROR_DIR
+ } else {
+ DATA_DIR
+ }
+ }
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/FileExt.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/FileExt.kt
new file mode 100644
index 0000000..ec660ce
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/FileExt.kt
@@ -0,0 +1,9 @@
+package com.wrbug.developerhelper.commonutil
+
+import java.io.File
+
+fun File.safeRead(): String {
+ return runCatching {
+ readText()
+ }.getOrDefault("")
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/FileUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/FileUtils.kt
deleted file mode 100644
index 33c6241..0000000
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/FileUtils.kt
+++ /dev/null
@@ -1,94 +0,0 @@
-package com.wrbug.developerhelper.commonutil
-
-import android.content.Context
-import android.media.MediaMetadataRetriever
-import android.net.Uri
-import android.os.Build
-import androidx.core.content.FileProvider
-import java.io.*
-
-
-/**
- * Created by wrbug on 2017年8月23日.
- */
-object FileUtils {
-
-
- fun writeByteToFile(data: ByteArray, path: String) {
- try {
- val localFileOutputStream = FileOutputStream(path)
- localFileOutputStream.write(data)
- localFileOutputStream.close()
- } catch (e: Exception) {
- }
- }
-
-
- fun inputstreamtofile(ins: InputStream, file: File) {
- try {
- if (file.exists()) {
- file.createNewFile()
- }
- val os = FileOutputStream(file)
- os.write(ins.readBytes())
- os.close()
- ins.close()
- } catch (t: Throwable) {
-
- }
-
- }
-
-
- fun inputStream2String(ins: InputStream): String {
- val out = StringBuffer()
- val b = ByteArray(4096)
- try {
- var n: Int = ins.read(b)
- while (n != -1) {
- out.append(String(b, 0, n))
- n = ins.read(b)
- }
- } catch (e: IOException) {
- return ""
- }
-
- return out.toString()
- }
-
-
- fun readFile(file: File): String {
- var inputStream: FileInputStream? = null
- try {
- inputStream = FileInputStream(file)
- val bytes = ByteArray(inputStream.available())
- inputStream.read(bytes)
- return String(bytes)
- } catch (e: IOException) {
- } finally {
- inputStream?.close()
- }
- return ""
- }
-
- fun whiteFile(file: File, data: String) {
- var outputStream: FileOutputStream? = null
- try {
- outputStream = FileOutputStream(file)
- outputStream.write(data.toByteArray())
- } catch (e: IOException) {
- }finally {
- outputStream?.close()
- }
-
- }
-}
-
-
-fun File.toUri(context: Context): Uri? {
- return if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.N) {
- FileProvider.getUriForFile(context, "com.wrbug.developerhelper.fileprovider", this)
- } else {
- Uri.fromFile(this)
- }
-}
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ReflectExts.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ReflectExts.kt
new file mode 100644
index 0000000..5066229
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ReflectExts.kt
@@ -0,0 +1,115 @@
+package com.wrbug.developerhelper.commonutil
+import java.lang.reflect.Field
+import java.lang.reflect.Method
+
+fun Class<*>.setValue(obj: Any?, fieldName: String, value: Any?) {
+ this.matchField(fieldName)?.set(obj, value)
+}
+
+fun Class.getValue(obj: T?, fieldName: String): Any? {
+ return this.matchField(fieldName)?.get(obj)
+}
+
+fun Any.getFieldValue(fieldName: String): Any? {
+ return javaClass.getValue(this, fieldName)
+}
+
+fun Any.setFieldValue(fieldName: String, value: Any?) {
+ javaClass.setValue(this, fieldName, value)
+}
+
+fun Any.callMethod(methodName: String, vararg args: Any?): Any? {
+ return this.javaClass.matchMethod(methodName, *args)?.invoke(this, *args)
+}
+
+fun Class.callMethod(obj: T?, methodName: String, vararg args: Any?): Any? {
+ return this.matchMethod(methodName, *args)?.invoke(obj, *args)
+}
+
+fun Class<*>.matchMethod(methodName: String, vararg args: Any?): Method? {
+ return findDeclaredMethod(methodName, *args) ?: findMethod(methodName, *args)
+}
+
+fun Class<*>.matchField(fieldName: String): Field? {
+ return findDeclaredField(fieldName) ?: findField(fieldName)
+}
+
+fun Class<*>.findDeclaredField(fieldName: String): Field? {
+ return runCatching {
+ this.getDeclaredField(fieldName).apply {
+ if (!isAccessible) {
+ isAccessible = true
+ }
+ }
+ }.getOrElse {
+ if (this == Any::class.java) {
+ return null
+ }
+ this.superclass.findDeclaredField(fieldName)
+ }
+}
+
+fun Class<*>.findDeclaredMethod(methodName: String, vararg args: Any?): Method? {
+ val method =
+ this.declaredMethods.asSequence()
+ .filter { it.name == methodName && it.parameterCount == args.size }
+ .find {
+ val size = args.size
+ for (i in 0 until size) {
+ val arg = args[i] ?: continue
+ if (!it.parameterTypes[i].isAssignableFrom(arg.javaClass)) {
+ return@find false
+ }
+ }
+ return@find true
+ }?.apply {
+ if (!isAccessible) {
+ isAccessible = true
+ }
+ }
+ return method ?: if (this == Any::class.java) {
+ null
+ } else {
+ superclass.findDeclaredMethod(methodName, *args)
+ }
+}
+
+fun Class<*>.findField(fieldName: String): Field? {
+ return runCatching {
+ this.getField(fieldName).apply {
+ if (!isAccessible) {
+ isAccessible = true
+ }
+ }
+ }.getOrElse {
+ if (this == Any::class.java) {
+ return null
+ }
+ this.superclass.findField(fieldName)
+ }
+}
+
+fun Class<*>.findMethod(methodName: String, vararg args: Any?): Method? {
+ val method =
+ this.methods.asSequence()
+ .filter { it.name == methodName && it.parameterCount == args.size }
+ .find {
+ val size = args.size
+ for (i in 0..size) {
+ val arg = args[i] ?: continue
+ if (!it.parameterTypes[i].isAssignableFrom(arg.javaClass)) {
+ return@find false
+ }
+ }
+ return@find true
+ }?.apply {
+ if (!isAccessible) {
+ isAccessible = true
+ }
+ }
+ return method ?: if (this == Any::class.java) {
+ null
+ } else {
+ superclass.findDeclaredMethod(methodName, *args)
+ }
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/RxJavaExt.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/RxJavaExt.kt
new file mode 100644
index 0000000..4bfefc3
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/RxJavaExt.kt
@@ -0,0 +1,31 @@
+package com.wrbug.developerhelper.commonutil
+
+import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers
+import io.reactivex.rxjava3.core.Single
+import io.reactivex.rxjava3.core.SingleEmitter
+import io.reactivex.rxjava3.core.SingleOnSubscribe
+import io.reactivex.rxjava3.disposables.CompositeDisposable
+import io.reactivex.rxjava3.disposables.Disposable
+import io.reactivex.rxjava3.schedulers.Schedulers
+
+fun Single.runOnIO(): Single {
+ return this.subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread())
+}
+
+fun Single.observeOnMain(): Single {
+ return this.observeOn(AndroidSchedulers.mainThread())
+}
+
+fun safeCreateSingle(source: (SingleEmitter) -> Unit): Single {
+ return Single.create { emit ->
+ runCatching {
+ source(emit)
+ }.getOrElse {
+ emit.onError(it)
+ }
+ }
+}
+
+fun Disposable.addTo(compositeDisposable: CompositeDisposable) {
+ compositeDisposable.add(this)
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ShellUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ShellUtils.kt
deleted file mode 100644
index aaba361..0000000
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/ShellUtils.kt
+++ /dev/null
@@ -1,57 +0,0 @@
-package com.wrbug.developerhelper.commonutil
-
-import com.jaredrummler.android.shell.CommandResult
-import com.jaredrummler.android.shell.Shell
-import org.jetbrains.anko.doAsync
-
-object ShellUtils {
- fun run(cmds: Array, callback: ShellResultCallback) {
- doAsync {
- val run = Shell.SH.run(*cmds)
- callback.onComplete(run)
- }
- }
-
- fun run(vararg cmds: String): CommandResult {
- return Shell.SH.run(*cmds)
- }
-
- fun runWithSu(cmds: Array, callback: ShellResultCallback) {
- if (!RootUtils.isRoot()) {
- callback.onError("未开启root权限")
- return
- }
- doAsync {
- val run = Shell.SU.run(*cmds)
- callback.onComplete(run)
- }
- }
-
- fun runWithSuIgnoreSetting(vararg cmds: String): CommandResult {
- return Shell.SU.run(*cmds)
- }
-
-
- fun runWithSu(vararg cmds: String): CommandResult {
- if (RootUtils.isRoot().not()) {
- return CommandResult(arrayListOf("未开启root权限"), arrayListOf("未开启root权限"), 1)
- }
- return Shell.SU.run(*cmds)
- }
-
- fun isRoot(): Boolean {
- return Shell.SU.available()
- }
-
- abstract class ShellResultCallback(vararg args: Any) {
- protected var args = args
- open fun onComplete(result: CommandResult) {
-
- }
-
- open fun onError(msg: String) {
-
- }
- }
-
-}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/SpannableBuilder.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/SpannableBuilder.kt
new file mode 100644
index 0000000..acd44ff
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/SpannableBuilder.kt
@@ -0,0 +1,125 @@
+package com.wrbug.developerhelper.commonutil
+
+import android.content.Context
+import android.graphics.Typeface
+import android.text.Spannable
+import android.text.SpannableStringBuilder
+import android.text.Spanned
+import android.text.TextPaint
+import android.text.style.ClickableSpan
+import android.text.style.ForegroundColorSpan
+import android.text.style.StrikethroughSpan
+import android.text.style.StyleSpan
+import android.text.style.TextAppearanceSpan
+import android.view.View
+
+class SpannableBuilder private constructor(private val context: Context, private val text: String) {
+
+ private val spanMap = hashMapOf>>()
+
+ companion object {
+ fun with(context: Context, strRes: Int): SpannableBuilder {
+ return SpannableBuilder(context, strRes)
+ }
+
+ fun with(context: Context, text: String): SpannableBuilder {
+ return SpannableBuilder(context, text)
+ }
+ }
+
+ private var strRes: Int = -1
+
+ private constructor(context: Context, strRes: Int) : this(context, "") {
+ this.strRes = strRes
+ }
+
+ fun addSpanWithTextAppearance(
+ value: String,
+ textAppearance: Int,
+ color: Int? = null,
+ index: Int = 0
+ ): SpannableBuilder {
+ addSpan(value, TextAppearanceSpan(context, textAppearance), index)
+ if (color != null) {
+ addSpan(value, ForegroundColorSpan(color), index)
+ }
+ return this
+ }
+
+ fun addSpanWithClickListener(
+ key: String,
+ linkColor: Int,
+ index: Int = 0,
+ listener: () -> Unit
+ ): SpannableBuilder {
+ addSpan(key, object : ClickableSpan() {
+ override fun onClick(widget: View) {
+ listener()
+ }
+
+ override fun updateDrawState(ds: TextPaint) {
+ ds.isUnderlineText = true
+ ds.color = linkColor
+ }
+ }, index)
+ return this
+ }
+
+ fun addSpanWithColor(value: String, color: Int, index: Int = 0): SpannableBuilder {
+ addSpan(value, ForegroundColorSpan(color), index)
+ return this
+ }
+
+
+ fun addSpanWithDeleteLine(value: String, index: Int = 0): SpannableBuilder {
+ addSpan(value, StrikethroughSpan(), index)
+ return this
+ }
+
+ fun addSpanWithBold(value: String, index: Int = 0): SpannableBuilder {
+ addSpan(value, StyleSpan(Typeface.BOLD), index)
+ return this
+ }
+
+ fun addCustomSpan(value: String, what: Any, index: Int = 0): SpannableBuilder {
+ addSpan(value, what, index)
+ return this
+ }
+
+ private fun addSpan(key: String, what: Any, index: Int) {
+ spanMap[key] = (spanMap[key] ?: arrayListOf()).apply { add(index to what) }
+ }
+
+ fun build(): Spannable {
+ val originStr = if (strRes == -1) {
+ text
+ } else {
+ context.getString(strRes)
+ }
+ val spannableBuilder = SpannableStringBuilder(originStr)
+ spanMap.forEach { (key, pair) ->
+ pair.forEach { (index, what) ->
+ val strIndex = index.let {
+ var i = it
+ var currentIndex = 0
+ while (i != 0) {
+ i--
+ currentIndex = originStr.indexOf(key, currentIndex) + key.length
+ }
+ originStr.indexOf(key, currentIndex)
+ }
+ if (strIndex == -1) {
+ return@forEach
+ }
+ spannableBuilder.setSpan(
+ what,
+ strIndex,
+ strIndex + key.length,
+ Spanned.SPAN_EXCLUSIVE_EXCLUSIVE
+ )
+ }
+
+ }
+ return spannableBuilder
+ }
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/StringConvertUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/StringConvertUtils.kt
index 4aca828..a3cb476 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/StringConvertUtils.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/StringConvertUtils.kt
@@ -4,7 +4,6 @@ import java.math.BigDecimal
import java.math.BigInteger
import java.util.regex.Pattern
-
fun String.isInt(): Boolean {
if (isNumber()) {
return BigInteger(this).toLong() < Integer.MAX_VALUE && BigInteger(this).toLong()> Integer.MIN_VALUE
@@ -39,30 +38,32 @@ fun String.toBoolean(): Boolean {
return false
}
-fun String.toInt(): Int {
+fun String?.toInt(): Int {
+ this ?: return 0
if (isInt()) {
return BigInteger(this).toInt()
}
return 0
}
-
-fun String.toLong(): Long {
+fun String?.toLong(): Long {
+ this ?: return 0
if (isNumber()) {
return BigInteger(this).toLong()
}
return 0L
}
-
-fun String.toFloat(): Float {
+fun String?.toFloat(): Float {
+ this ?: return 0F
if (isDecimal()) {
return BigDecimal(this).toFloat()
}
return 0F
}
-fun String.toDouble(): Double {
+fun String?.toDouble(): Double {
+ this ?: return 0.0
if (isDecimal()) {
return BigDecimal(this).toDouble()
}
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/UiUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/UiUtils.kt
index 99fb62f..fe73ddd 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/UiUtils.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/UiUtils.kt
@@ -10,21 +10,17 @@ fun Context.dp2px(dpVal: Float): Int = UiUtils.dp2px(this, dpVal)
fun Fragment.dp2px(dpVal: Float): Int = UiUtils.dp2px(activity!!, dpVal)
fun Dialog.dp2px(dpVal: Float): Int = UiUtils.dp2px(context, dpVal)
fun View.dp2px(dpVal: Float): Int = UiUtils.dp2px(context, dpVal)
+fun Float.dpInt(context: Context) = UiUtils.dp2px(context, this)
+fun Int.dpInt(context: Context = CommonUtils.application) = UiUtils.dp2px(context, toFloat())
object UiUtils {
private var statusBarHeight: Int = -1
fun dp2px(context: Context = CommonUtils.application, dpVal: Float): Int {
return TypedValue.applyDimension(
- TypedValue.COMPLEX_UNIT_DIP,
- dpVal,
- context.resources?.displayMetrics
+ TypedValue.COMPLEX_UNIT_DIP, dpVal, context.resources?.displayMetrics
).toInt()
}
- fun sp2px(context: Context = CommonUtils.application, spVal: Float): Int {
- return TypedValue.applyDimension(2, spVal, context.resources?.displayMetrics).toInt()
- }
-
fun px2dp(context: Context = CommonUtils.application, pxVal: Float): Float {
val scale = context.resources?.displayMetrics?.density!!
return pxVal / scale
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/ApkInfo.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/ApkInfo.kt
index ea30de8..f0ca8de 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/ApkInfo.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/ApkInfo.kt
@@ -6,13 +6,14 @@ import android.graphics.drawable.Drawable
import android.os.Parcel
import android.os.Parcelable
import com.wrbug.developerhelper.commonutil.CommonUtils
+import kotlinx.parcelize.Parcelize
-class ApkInfo(val packageInfo: PackageInfo, val applicationInfo: ApplicationInfo) : Parcelable {
-
- constructor(parcel: Parcel) : this(
- parcel.readParcelable(PackageInfo::class.java.classLoader),
- parcel.readParcelable(ApplicationInfo::class.java.classLoader)
- )
+@Parcelize
+class ApkInfo(
+ val packageInfo: PackageInfo,
+ val applicationInfo: ApplicationInfo,
+ var topActivity: String = ""
+) : Parcelable {
fun getIco(): Drawable {
return applicationInfo.loadIcon(CommonUtils.application.packageManager)
@@ -20,26 +21,11 @@ class ApkInfo(val packageInfo: PackageInfo, val applicationInfo: ApplicationInfo
fun getAppName(): String {
val label = CommonUtils.application.packageManager.getApplicationLabel(applicationInfo)
- return label?.toString() ?: ""
- }
-
- override fun writeToParcel(parcel: Parcel, flags: Int) {
- parcel.writeParcelable(packageInfo, flags)
- parcel.writeParcelable(applicationInfo, flags)
+ return label.toString()
}
- override fun describeContents(): Int {
- return 0
- }
-
- companion object CREATOR : Parcelable.Creator {
- override fun createFromParcel(parcel: Parcel): ApkInfo {
- return ApkInfo(parcel)
- }
-
- override fun newArray(size: Int): Array {
- return arrayOfNulls(size)
- }
+ fun generateBackupApkFileName(): String {
+ return packageInfo.versionName + ".apk"
}
}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/CpuABI.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/CpuABI.kt
new file mode 100644
index 0000000..952e1b8
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/CpuABI.kt
@@ -0,0 +1,5 @@
+package com.wrbug.developerhelper.commonutil.entity
+
+enum class CpuABI {
+ ARM,X86
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/FragmentInfo.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/FragmentInfo.kt
index ab23081..cea0527 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/FragmentInfo.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/FragmentInfo.kt
@@ -1,71 +1,23 @@
package com.wrbug.developerhelper.commonutil.entity
-import android.os.Parcel
import android.os.Parcelable
-
-class FragmentInfo() : Parcelable {
- var name = ""
- var fragmentId = ""
- var containerId = ""
- var tag = ""
- var state = 0
- var index = 0
- var who = ""
- var backStackNesting = 0
- var added = false
- var removing = false
- var fromLayout = false
- var inLayout = false
- var hidden = true
- var detached = false
-
- constructor(parcel: Parcel) : this() {
- name = parcel.readString()
- fragmentId = parcel.readString()
- containerId = parcel.readString()
- tag = parcel.readString()
- state = parcel.readInt()
- index = parcel.readInt()
- who = parcel.readString()
- backStackNesting = parcel.readInt()
- added = parcel.readByte() != 0.toByte()
- removing = parcel.readByte() != 0.toByte()
- fromLayout = parcel.readByte() != 0.toByte()
- inLayout = parcel.readByte() != 0.toByte()
- hidden = parcel.readByte() != 0.toByte()
- detached = parcel.readByte() != 0.toByte()
- }
-
- override fun writeToParcel(parcel: Parcel, flags: Int) {
- parcel.writeString(name)
- parcel.writeString(fragmentId)
- parcel.writeString(containerId)
- parcel.writeString(tag)
- parcel.writeInt(state)
- parcel.writeInt(index)
- parcel.writeString(who)
- parcel.writeInt(backStackNesting)
- parcel.writeByte(if (added) 1 else 0)
- parcel.writeByte(if (removing) 1 else 0)
- parcel.writeByte(if (fromLayout) 1 else 0)
- parcel.writeByte(if (inLayout) 1 else 0)
- parcel.writeByte(if (hidden) 1 else 0)
- parcel.writeByte(if (detached) 1 else 0)
- }
-
- override fun describeContents(): Int {
- return 0
- }
-
- companion object CREATOR : Parcelable.Creator {
- override fun createFromParcel(parcel: Parcel): FragmentInfo {
- return FragmentInfo(parcel)
- }
-
- override fun newArray(size: Int): Array {
- return arrayOfNulls(size)
- }
- }
-
+import kotlinx.parcelize.Parcelize
+
+@Parcelize
+data class FragmentInfo(
+ var name: String = "",
+ var containerId: String = "",
+ var tag: String = "",
+ var state: Int = 0,
+ var index: Int = 0,
+ var who: String = "",
+ var backStackNesting: Int = 0,
+ var added: Boolean = false,
+ var removing: Boolean = false,
+ var fromLayout: Boolean = false,
+ var inLayout: Boolean = false,
+ var hidden: Boolean = true,
+ var detached: Boolean = false
+): Parcelable {
}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/TopActivityInfo.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/TopActivityInfo.kt
index d5e415a..2dd2b58 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/TopActivityInfo.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/entity/TopActivityInfo.kt
@@ -1,36 +1,22 @@
package com.wrbug.developerhelper.commonutil.entity
-import android.os.Parcel
import android.os.Parcelable
+import kotlinx.parcelize.Parcelize
-class TopActivityInfo() : Parcelable {
- var activity = ""
- var viewIdHex = HashMap()
+@Parcelize
+class TopActivityInfo(
+ var packageName: String = "",
+ var activity: String = "",
var fragments: Array? = null
+): Parcelable {
- constructor(parcel: Parcel) : this() {
- activity = parcel.readString()
- fragments = parcel.createTypedArray(FragmentInfo)
- }
-
- override fun writeToParcel(parcel: Parcel, flags: Int) {
- parcel.writeString(activity)
- parcel.writeTypedArray(fragments, flags)
- }
-
- override fun describeContents(): Int {
- return 0
- }
-
- companion object CREATOR : Parcelable.Creator {
- override fun createFromParcel(parcel: Parcel): TopActivityInfo {
- return TopActivityInfo(parcel)
- }
-
- override fun newArray(size: Int): Array {
- return arrayOfNulls(size)
+ fun setFullActivity(ac: String) {
+ kotlin.runCatching {
+ val arr = ac.split("/")
+ packageName = arr[0]
+ activity = arr[0] + arr[1]
}
}
-
+ var viewIdHex = HashMap()
}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/Callback.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/Callback.kt
deleted file mode 100644
index 4bc41c9..0000000
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/Callback.kt
+++ /dev/null
@@ -1,6 +0,0 @@
-package com.wrbug.developerhelper.commonutil.shell
-
-interface Callback {
- fun onSuccess(data: T)
- fun onFailed(msg: String = "") {}
-}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/CommandResult.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/CommandResult.kt
new file mode 100644
index 0000000..4894b56
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/CommandResult.kt
@@ -0,0 +1,28 @@
+package com.wrbug.developerhelper.commonutil.shell
+
+data class CommandResult(
+ val stdout: List,
+ val stderr: List,
+ val exitCode: Int,
+ val details: Shell.Command.Result.Details?
+) {
+ val isSuccessful: Boolean
+ get() = exitCode == 0
+
+ fun getStdout(): String {
+ return toString(stdout)
+ }
+
+ /**
+ * Get the standard error.
+ *
+ * @return The standard error as a string.
+ */
+ fun getStderr(): String {
+ return toString(stderr)
+ }
+
+ private fun toString(lines: List?): String {
+ return lines?.joinToString("\n").orEmpty()
+ }
+}
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/Shell.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/Shell.kt
new file mode 100644
index 0000000..4841d47
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/Shell.kt
@@ -0,0 +1,826 @@
+package com.wrbug.developerhelper.commonutil.shell
+
+import com.wrbug.developerhelper.commonutil.toInt
+import java.io.BufferedReader
+import java.io.DataOutputStream
+import java.io.IOException
+import java.io.InputStream
+import java.io.InputStreamReader
+import java.io.OutputStream
+import java.util.Collections
+import java.util.UUID
+import java.util.concurrent.CountDownLatch
+import java.util.concurrent.TimeUnit
+import java.util.concurrent.locks.ReentrantLock
+import java.util.regex.Pattern
+
+/** Environment variable. */
+typealias Variable = String
+/** Environment variable value. */
+typealias Value = String
+/** A [Map] for the environment variables used in the shell. */
+typealias EnvironmentMap = Map
+
+
+/**
+ * A shell starts a [Process] with the provided shell and additional/optional environment variables.
+ * The shell handles maintaining the [Process] and reads standard output and standard error streams,
+ * returning stdout, stderr, and the last exit code as a [Command.Result] when a command is complete.
+ *
+ * Example usage:
+ *
+ * val sh = Shell("sh")
+ * val result = sh.run("echo 'Hello, World!'")
+ * assert(result.isSuccess)
+ * assert(result.stdout() == "Hello, World")
+ *
+ * @property path The path to the shell to start.
+ * @property environment Map of all environment variables to include with the system environment.
+ * Default value is an empty map.
+ * @throws Shell.NotFoundException If the shell cannot be opened this runtime exception is thrown.
+ * @author Jared Rummler (jaredrummler@gmail.com)
+ * @since 05-05-2021
+ */
+class Shell @Throws(NotFoundException::class) @JvmOverloads constructor(
+ val path: String,
+ val environment: EnvironmentMap = emptyMap()
+) {
+
+
+ /**
+ * Construct a new [Shell] with optional environment variable arguments as a [Pair].
+ *
+ * @param shell The path to the shell to start.
+ * @param environment varargs of all environment variables as a [Pair] which are included
+ * with the system environment.
+ */
+ constructor(shell: String, vararg environment: Pair) :
+ this(shell, environment.toEnvironmentMap())
+
+ /**
+ * Construct a new [Shell] with optional environment variable arguments as an array.
+ *
+ * @param shell The path to the shell to start.
+ * @param environment varargs of all environment variables as a [Pair] which are included
+ * with the system environment.
+ */
+ constructor(shell: String, environment: Array) :
+ this(shell, environment.toEnvironmentMap())
+
+ /**
+ * Get the current state of the shell
+ */
+ var state: State = State.Idle
+ private set
+
+ private val onResultListeners = mutableSetOf()
+ private val onStdOutListeners = mutableSetOf()
+
+ private val onStdErrListeners = mutableSetOf()
+ private val stdin: StandardInputStream
+ private val stdoutReader: StreamReader
+ private val stderrReader: StreamReader
+ private var watchdog: Watchdog? = null
+
+ private val process: Process
+
+ init {
+ try {
+ process = runWithEnv(path, environment)
+ stdin = StandardInputStream(process.outputStream)
+ stdoutReader = StreamReader.createAndStart(THREAD_NAME_STDOUT, process.inputStream)
+ stderrReader = StreamReader.createAndStart(THREAD_NAME_STDERR, process.errorStream)
+ } catch (cause: Exception) {
+ throw NotFoundException(String.format(EXCEPTION_SHELL_CANNOT_OPEN, path), cause)
+ }
+ }
+
+ /**
+ * Add a listener that will be invoked each time a command finishes.
+ *
+ * @param listener The listener to receive callbacks when commands finish executing.
+ * @return This shell instance for chaining calls.
+ */
+ fun addOnCommandResultListener(listener: OnCommandResultListener) = apply {
+ onResultListeners.add(listener)
+ }
+
+ /**
+ * Remove a listener previously added to stop receiving callbacks when commands finish.
+ *
+ * @param listener The listener registered via [addOnCommandResultListener].
+ * @return This shell instance for chaining calls.
+ */
+ fun removeOnCommandResultListener(listener: OnCommandResultListener) = apply {
+ onResultListeners.remove(listener)
+ }
+
+ /**
+ * Add a listener that will be invoked each time the STDOUT stream reads a new line.
+ *
+ * @param listener The listener to receive callbacks when the STDOUT stream reads a line.
+ * @return This shell instance for chaining calls.
+ */
+ fun addOnStdoutLineListener(listener: OnLineListener) = apply {
+ onStdOutListeners.add(listener)
+ }
+
+ /**
+ * Remove a listener previously added to stop receiving callbacks for STDOUT read lines.
+ *
+ * @param listener The listener registered via [addOnStdoutLineListener].
+ * @return This shell instance for chaining calls.
+ */
+ fun removeOnStdoutLineListener(listener: OnLineListener) = apply {
+ onStdOutListeners.remove(listener)
+ }
+
+ /**
+ * Add a listener that will be invoked each time the STDERR stream reads a new line.
+ *
+ * @param listener The listener to receive callbacks when the STDERR stream reads a line.
+ * @return This shell instance for chaining calls.
+ */
+ fun addOnStderrLineListener(listener: OnLineListener) = apply {
+ onStdErrListeners.add(listener)
+ }
+
+ /**
+ * Remove a listener previously added to stop receiving callbacks for STDERR read lines.
+ *
+ * @param listener The listener registered via [addOnStderrLineListener].
+ * @return This shell instance for chaining calls.
+ */
+ fun removeOnStderrLineListener(listener: OnLineListener) = apply {
+ onStdErrListeners.remove(listener)
+ }
+
+ /**
+ * Run a command in the current shell and return its [result][Command.Result].
+ *
+ * @param command The command to execute.
+ * @param config The [options][Command.Config] to set when running the command.
+ * @return The [result][Command.Result] containing stdout, stderr, status of running the command.
+ * @throws ClosedException if the shell was closed prior to running the command.
+ * @see shutdown
+ * @see run
+ */
+ @Throws(ClosedException::class)
+ @Synchronized
+ fun run(
+ command: String,
+ config: Command.Config.Builder.() -> Unit,
+ ) = run(command, Command.Config.Builder().apply(config).create())
+
+ /**
+ * Run a command in the current shell and return its [result][Command.Result].
+ *
+ * @param command The command to execute.
+ * @param config The [options][Command.Config] to set when running the command.
+ * @return The [result][Command.Result] containing stdout, stderr, status of running the command.
+ * @throws ClosedException if the shell was closed prior to running the command.
+ * @see shutdown
+ * @see run
+ */
+ @Throws(ClosedException::class)
+ @Synchronized
+ @JvmOverloads
+ fun run(
+ command: String,
+ config: Command.Config = Command.Config.default(),
+ ): Command.Result {
+ // If the shell is shutdown, throw a ShellClosedException.
+ if (state == State.Shutdown) throw ClosedException(EXCEPTION_SHELL_SHUTDOWN)
+
+ val stdout = Collections.synchronizedList(mutableListOf())
+ val stderr = Collections.synchronizedList(mutableListOf())
+
+ val watchdog = Watchdog().also { watchdog = it }
+ var exitCode = Command.Status.INVALID
+ val uuid = config.uuid
+
+ val onComplete = { marker: Command.Marker ->
+ when (marker.uuid) {
+ uuid ->
+ try { // Reached the end of reading the stream for the command.
+ if (marker.status != Command.Status.INVALID) {
+ exitCode = marker.status
+ }
+ } finally {
+ watchdog.signal()
+ }
+ }
+ }
+
+ val lock = ReentrantLock()
+ val output = Collections.synchronizedList(mutableListOf())
+
+ // Function to process stderr and stdout streams.
+ fun onLine(
+ buffer: MutableList,
+ listeners: Set,
+ onLine: (line: String) -> Unit,
+ ) = { line: String ->
+ try {
+ lock.lock()
+ if (config.notify) {
+ listeners.forEach { listener -> listener.onLine(line) }
+ }
+ buffer.add(line)
+ output.add(line)
+ onLine(line)
+ } finally {
+ lock.unlock()
+ }
+ }
+
+ stdoutReader.onComplete = onComplete
+ stderrReader.onComplete = onComplete
+ stdoutReader.onReadLine = onLine(stdout, onStdOutListeners, config.onStdOut)
+ stderrReader.onReadLine = when (config.redirectStdErr) {
+ true -> onLine(stdout, onStdOutListeners, config.onStdOut)
+ else -> onLine(stderr, onStdErrListeners, config.onStdErr)
+ }
+
+ val startTime = System.currentTimeMillis()
+ try {
+ state = State.Running
+ // Write the command and command end marker to stdin.
+ write(command, "echo '$uuid' $?", "echo '$uuid'>&2")
+ // Wait for the result with a timeout, if provided.
+ if (!watchdog.await(config.timeout)) {
+ exitCode = Command.Status.TIMEOUT
+ config.onTimeout()
+ }
+ } catch (e: InterruptedException) {
+ exitCode = Command.Status.TERMINATED
+ config.onCancelled()
+ } finally {
+ this.watchdog = null
+ state = State.Idle
+ }
+
+ if (exitCode != Command.Status.SUCCESS) {
+ // Exit with the error code in a subshell
+ // This is necessary because we send commands to signal a command was completed
+ write("$(exit $exitCode)")
+ }
+
+ // Create the result from running the command.
+ val result = Command.Result.create(
+ uuid,
+ command,
+ stdout,
+ stderr,
+ output,
+ exitCode,
+ startTime
+ )
+
+ if (config.notify) {
+ onResultListeners.forEach { listener ->
+ listener.onResult(result)
+ }
+ }
+
+ return result
+ }
+
+ /**
+ * Check if the shell is idle.
+ *
+ * @return True if the shell is open but not running any commands.
+ */
+ fun isIdle() = state is State.Idle
+
+ /**
+ * Check if the shell is running a command.
+ *
+ * @return True if the shell is executing a command.
+ */
+ fun isRunning() = state is State.Running
+
+ /**
+ * Check if the shell is shutdown.
+ *
+ * @return True if the shell is closed.
+ * @see shutdown
+ */
+ fun isShutdown() = state is State.Shutdown
+
+ /**
+ * Check if the shell is alive and able to execute commands.
+ *
+ * @return True if the shell is running or idle.
+ */
+ fun isAlive() = try {
+ process.exitValue(); false
+ } catch (e: IllegalThreadStateException) {
+ true
+ }
+
+ /**
+ * Interrupt waiting for a command to complete.
+ */
+ fun interrupt() {
+ watchdog?.abort()
+ }
+
+ /**
+ * Shutdown the shell instance. After a shell is shutdown it can no longer execute commands
+ * and should be garbage collected.
+ */
+ @Synchronized
+ fun shutdown() {
+ try {
+ write("exit")
+ process.waitFor()
+ stdin.closeQuietly()
+ onStdOutListeners.clear()
+ onStdErrListeners.clear()
+ stdoutReader.join()
+ stderrReader.join()
+ process.destroy()
+ } catch (ignored: IOException) {
+ } finally {
+ state = State.Shutdown
+ }
+ }
+
+ private fun write(vararg commands: String) = try {
+ commands.forEach { command -> stdin.write(command) }
+ stdin.flush()
+ } catch (ignored: IOException) {
+ }
+
+ private fun DataOutputStream.closeQuietly() = try {
+ close()
+ } catch (ignored: IOException) {
+ }
+
+ /**
+ * Contains data classes used for running commands in a [Shell].
+ *
+ * @see Command.Result
+ * @see Command.Config
+ * @see Command.Status
+ */
+ object Command {
+
+ /**
+ * The result of running a command in a shell.
+ *
+ * @property stdout A list of lines read from the standard input stream.
+ * @property stderr A list of lines read from the standard error stream.
+ * @property exitCode The status code of running the command.
+ * @property details Additional command result details.
+ */
+ data class Result(
+ val stdout: List,
+ val stderr: List,
+ val output: List,
+ val exitCode: Int,
+ val details: Details?
+ ) {
+
+ /**
+ * True when the exit code is equal to 0.
+ */
+ val isSuccess: Boolean get() = exitCode == Status.SUCCESS
+
+ /**
+ * Get [stdout] and [stderr] as a string, separated by new lines.
+ *
+ * @return The output of running the command in a shell.
+ */
+ fun output(): String = output.joinToString("\n")
+
+ /**
+ * Get [stdout] as a string, separated by new lines.
+ *
+ * @return The standard ouput string.
+ */
+ fun stdout(): String = stdout.joinToString("\n")
+
+ /**
+ * Get [stdout] as a string, separated by new lines.
+ *
+ * @return The standard ouput string.
+ */
+ fun stderr(): String = stderr.joinToString("\n")
+
+ /**
+ * Additional details pertaining to running a command in a shell.
+ *
+ * @property uuid The unique identifier associated with the command.
+ * @property command The command sent to the shell to execute.
+ * @property startTime The time—in milliseconds since January 1, 1970, 00:00:00 GMT—when
+ * the command started execution.
+ * @property endTime The time—in milliseconds since January 1, 1970, 00:00:00 GMT—when
+ * the command completed execution.
+ * @property elapsed The number of milliseconds it took to execute the command.
+ */
+ data class Details internal constructor(
+ val uuid: UUID,
+ val command: String,
+ val startTime: Long,
+ val endTime: Long,
+ val elapsed: Long = endTime - startTime
+ )
+
+ companion object {
+ internal fun create(
+ uuid: UUID,
+ command: String,
+ stdout: List,
+ stderr: List,
+ output: List,
+ exitCode: Int,
+ startTime: Long,
+ endTime: Long = System.currentTimeMillis(),
+ ) = Result(
+ stdout,
+ stderr,
+ output,
+ exitCode,
+ Details(uuid, command, startTime, endTime)
+ )
+ }
+ }
+
+ /**
+ * Optional configuration settings when running a command in a [shell][Shell].
+ *
+ * @property uuid The unique identifier associated with the command.
+ * @property redirectStdErr True to redirect STDERR to STDOUT.
+ * @property onStdOut Callback that is invoked when reading a line from stdout.
+ * @property onStdErr Callback that is invoked when reading a line from stderr.
+ * @property onCancelled Callback that is invoked when the command is interrupted.
+ * @property onTimeout Callback that is invoked when the command timed-out.
+ * @property timeout The time to wait before killing the command.
+ * @property notify True to notify any [OnLineListener] and [OnCommandResultListener] of the command.
+ */
+ class Config private constructor(
+ val uuid: UUID = UUID.randomUUID(),
+ val redirectStdErr: Boolean = false,
+ val onStdOut: (line: String) -> Unit = {},
+ val onStdErr: (line: String) -> Unit = {},
+ val onCancelled: () -> Unit = {},
+ val onTimeout: () -> Unit = {},
+ val timeout: Timeout? = null,
+ val notify: Boolean = true
+ ) {
+
+ /**
+ * Optional configuration settings when running a command in a [shell][Shell].
+ *
+ * @property uuid The unique identifier associated with the command.
+ * @property redirectErrorStream True to redirect STDERR to STDOUT.
+ * @property onStdOut Callback that is invoked when reading a line from stdout.
+ * @property onStdErr Callback that is invoked when reading a line from stderr.
+ * @property onCancelled Callback that is invoked when the command is interrupted.
+ * @property onTimeout Callback that is invoked when the command timed-out.
+ * @property timeout The time to wait before killing the command.
+ * @property notify True to notify any [OnLineListener] and [OnCommandResultListener] of the command.
+ */
+ class Builder {
+ var uuid: UUID = UUID.randomUUID()
+ var redirectErrorStream = false
+ var onStdOut: (line: String) -> Unit = {}
+ var onStdErr: (line: String) -> Unit = {}
+ var onCancelled: () -> Unit = {}
+ var onTimeout: () -> Unit = {}
+ var timeout: Timeout? = null
+ var notify = true
+
+ /**
+ * Create the [Config] from this builder.
+ *
+ * @return A new [Config] for a command.
+ */
+ fun create() =
+ Config(
+ uuid,
+ redirectErrorStream,
+ onStdOut,
+ onStdErr,
+ onCancelled,
+ onTimeout,
+ timeout,
+ notify
+ )
+ }
+
+ companion object {
+
+ /**
+ * The default configuration for running a command in a shell.
+ *
+ * @return The default config.
+ */
+ fun default(): Config = Builder().create()
+
+ /**
+ * Config that doesn't invoke callbacks for line and command complete listeners.
+ */
+ fun silent(): Config = Builder().apply { notify = false }.create()
+ }
+ }
+
+ /**
+ * The command marker to process standard input/error streams.
+ *
+ * @property uuid The unique ID for a command.
+ * @property status the exit code for the last run command.
+ */
+ internal data class Marker(val uuid: UUID, val status: Int)
+
+ /** Exit codes */
+ object Status {
+ /** OK exit code value */
+ const val SUCCESS = 0
+
+ /** Command timeout exit status */
+ const val TIMEOUT = 124
+
+ /** Command failed exit status */
+ const val COMMAND_FAILED = 125
+
+ /** Command not executable exit status */
+ const val NOT_EXECUTABLE = 126
+
+ /** Command not found exit status */
+ const val NOT_FOUND = 127
+
+ /** Command terminated exit status. */
+ const val TERMINATED = 128 + 30
+ internal const val INVALID = 0x100
+ }
+ }
+
+ /**
+ * Interface to receive a callback when reading a line from standard output/error streams.
+ */
+ interface OnLineListener {
+
+ /**
+ * Called when a line was read from standard output/error streams
+ *
+ * @param line The string that was read.
+ */
+ fun onLine(line: String)
+ }
+
+ /**
+ * Interface to receive a callback when a command completes.
+ */
+ interface OnCommandResultListener {
+
+ /**
+ * Called when a command finishes running.
+ *
+ * @param result The result of running the command in a shell.
+ */
+ fun onResult(result: Command.Result)
+ }
+
+ /**
+ * A timeout used when running a command in a shell.
+ *
+ * @property value The value of the time based on the [unit].
+ * @property unit The time unit for the [value].
+ */
+ data class Timeout(val value: Long, val unit: TimeUnit)
+
+ /**
+ * The exception thrown when a command is passed to a closed shell.
+ */
+ class ClosedException(message: String) : IOException(message)
+
+ /**
+ * The exception thrown when the shell could not be opened.
+ */
+ class NotFoundException(message: String, cause: Throwable) : RuntimeException(message, cause)
+
+ /**
+ * Represents the possible states of the shell.
+ */
+ sealed class State {
+ /** The shell is idle; no commands are in progress. */
+ object Idle : State()
+
+ /** The shell is currently running a command. */
+ object Running : State()
+
+ /** The shell has been shutdown. */
+ object Shutdown : State()
+ }
+
+ /**
+ * A class to cause the current thread to wait until a command completes or is aborted.
+ */
+ private class Watchdog : CountDownLatch(STREAM_READER_COUNT) {
+
+ private var aborted = false
+
+ /**
+ * Releases the thread immediately instead of waiting for [signal] to be invoked twice.
+ */
+ fun abort() {
+ if (count == 0L) return
+ aborted = true
+ while (count> 0) countDown()
+ }
+
+ /**
+ * Signal that either standard output or standard input streams are finished processing.
+ */
+ fun signal() = countDown()
+
+ /**
+ * Causes the current thread to wait until [signal] is called twice.
+ *
+ * @param timeout The maximum time to wait before [AbortedException] is thrown.
+ * @throws AbortedException if the timeout completes before [signal] is called twice
+ * or if the thread is interrupted.
+ */
+ @Throws(AbortedException::class)
+ fun await(timeout: Timeout?): Boolean {
+ return when (timeout) {
+ null -> {
+ await(); true
+ }
+
+ else -> await(timeout.value, timeout.unit)
+ }
+ }
+
+ override fun await() = super.await().also {
+ if (aborted) throw AbortedException()
+ }
+
+ override fun await(timeout: Long, unit: TimeUnit) = super.await(timeout, unit).also {
+ if (aborted) throw AbortedException()
+ }
+
+ companion object {
+ /**
+ * The number of times [signal] should be called to release the latch.
+ */
+ private const val STREAM_READER_COUNT = 2
+
+ /**
+ * The exception thrown when [abort] is called and the [CountDownLatch] has not finished
+ */
+ class AbortedException : InterruptedException()
+ }
+ }
+
+ /**
+ * The [OutputStream] for writing commands to the shell.
+ */
+ private class StandardInputStream(stream: OutputStream) : DataOutputStream(stream) {
+
+ /**
+ * The helper function to write commands to the stream with an appended new line character.
+ *
+ * @param command The command to write.
+ */
+ fun write(command: String) = write("$command\n".toByteArray(Charsets.UTF_8))
+ }
+
+ /**
+ * A thread that parses the standard/error streams for the shell.
+ *
+ * @param name The name of the stream. One of: [THREAD_NAME_STDOUT], [THREAD_NAME_STDERR]
+ * @param stream Either the [Process.getInputStream] or [Process.getErrorStream]
+ */
+ private class StreamReader private constructor(
+ name: String,
+ private val stream: InputStream
+ ) : Thread(name) {
+
+ /**
+ * The lambda that is invoked when a line is read from the stream.
+ */
+ var onReadLine: (line: String) -> Unit = {}
+
+ /**
+ * The lambda that is invoked when a command completes.
+ */
+ var onComplete: (marker: Command.Marker) -> Unit = {}
+
+ override fun run() = BufferedReader(InputStreamReader(stream)).forEachLine { line ->
+ pattern.matcher(line).let { matcher ->
+ if (matcher.matches()) {
+ val uuid = UUID.fromString(matcher.group(GROUP_UUID))
+ onComplete(
+ when (val exitCode = matcher.group(GROUP_CODE)) {
+ null -> Command.Marker(uuid, Command.Status.INVALID)
+ else -> Command.Marker(uuid, exitCode.toInt())
+ }
+ )
+ } else {
+ onReadLine(line)
+ }
+ }
+ }
+
+ companion object {
+
+ private const val GROUP_UUID = 1
+
+ private const val GROUP_CODE = 2
+
+ //
+ private val pattern: Pattern = Pattern.compile(
+ "^([a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12})\\s?([0-9]{1,3})?$",
+ Pattern.CASE_INSENSITIVE
+ )
+
+ internal fun createAndStart(name: String, stream: InputStream) =
+ StreamReader(name, stream).also { reader -> reader.start() }
+ }
+ }
+
+ companion object {
+
+ private const val THREAD_NAME_STDOUT = "STDOUT"
+ private const val THREAD_NAME_STDERR = "STDERR"
+
+ private const val EXCEPTION_SHELL_CANNOT_OPEN = "Error opening shell: '%s'"
+ private const val EXCEPTION_SHELL_SHUTDOWN = "The shell is shutdown"
+ private val instances by lazy { mutableMapOf() }
+
+ /**
+ * Returns a [Shell] instance using the [path] as the path to the shell/executable.\
+ */
+ operator fun get(path: String): Shell = instances[path]?.takeIf { shell ->
+ shell.isAlive()
+ } ?: Shell(path).also { shell ->
+ instances[path] = shell
+ }
+
+ /** The Bourne shell (sh) */
+ val SH: Shell get() = this["sh"]
+
+ /** Switch to root, and run it as a shell */
+ val SU: Shell get() = this["su"]
+
+ /**
+ * Execute a command with the provided environment.
+ *
+ * @param command
+ * The name of the program to execute. E.g. "su" or "sh".
+ * @param environment
+ * Map of all environment variables to include with the system environment.
+ * @return The new [Process] instance.
+ * @throws IOException
+ * If the requested program could not be executed.
+ */
+ @Throws(IOException::class)
+ private fun runWithEnv(command: String, environment: EnvironmentMap): Process =
+ Runtime.getRuntime().exec(command, (System.getenv() + environment).toArray())
+
+ /**
+ * Convert an array to an [EnvironmentMap] with each variable/value separated by '='.
+ *
+ * @return The array converted to an [EnvironmentMap].
+ */
+ private fun Array.toEnvironmentMap(): EnvironmentMap =
+ mutableMapOf().also { map ->
+ forEach { str ->
+ str.split("=").takeIf { arr ->
+ arr.size == 2
+ }?.let { (variable, value) ->
+ map[variable] = value
+ }
+ }
+ }.toMap()
+
+ /**
+ * Convert an array of [Pair] to an [EnvironmentMap].
+ *
+ * @return The array of variable/value pairs as a new [EnvironmentMap].
+ */
+ private fun Array>.toEnvironmentMap(): EnvironmentMap =
+ mutableMapOf().also { map ->
+ forEach { (variable, value) ->
+ map[variable] = value
+ }
+ }
+
+ /**
+ * Converts an [EnvironmentMap] to an array of strings with the variable/value
+ * separated by an '=' character.
+ *
+ * @return An array of environment variables.
+ */
+ private fun EnvironmentMap.toArray(): Array =
+ mutableListOf().also { list ->
+ forEach { (variable, value) ->
+ list.add("$variable=$value")
+ }
+ }.toTypedArray()
+ }
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellException.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellException.kt
new file mode 100644
index 0000000..7c6f82b
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellException.kt
@@ -0,0 +1,4 @@
+package com.wrbug.developerhelper.commonutil.shell
+
+class ShellException(message: String) : Exception(message) {
+}
\ No newline at end of file
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellManager.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellManager.kt
index 801eb3c..61ddb69 100644
--- a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellManager.kt
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellManager.kt
@@ -1,18 +1,19 @@
package com.wrbug.developerhelper.commonutil.shell
-import com.jaredrummler.android.shell.CommandResult
-import com.wrbug.developerhelper.commonutil.CommonUtils
-import com.wrbug.developerhelper.commonutil.ShellUtils
-import com.wrbug.developerhelper.commonutil.entity.FragmentInfo
+import com.wrbug.developerhelper.commonutil.Constant
import com.wrbug.developerhelper.commonutil.entity.LsFileInfo
import com.wrbug.developerhelper.commonutil.entity.TopActivityInfo
+import com.wrbug.developerhelper.commonutil.runOnIO
+import io.reactivex.rxjava3.core.Single
import java.io.File
import java.util.regex.Pattern
-
object ShellManager {
+
private const val SHELL_TOP_ACTIVITY = "dumpsys activity top"
- private const val SHELL_PROCESS_PID_1 = "ps -ef | grep \"%1\$s\" | grep -v %1\$s:| grep -v grep | awk '{print \2ドル}'"
+ private const val SHELL_APP_ACTIVITY = "dumpsys activity %1\$s"
+ private const val SHELL_PROCESS_PID_1 =
+ "ps -ef | grep \"%1\$s\" | grep -v %1\$s:| grep -v grep | awk '{print \2ドル}'"
private const val SHELL_PROCESS_PID_2 = "top -b -n 1 |grep %1\$s |grep -v grep|grep -v %1\$s:"
private const val SHELL_PROCESS_PID_3 = "top -n 1 |grep %1\$s |grep -v grep|grep -v %1\$s:"
private var SHELL_OPEN_ACCESSiBILITY_SERVICE = arrayOf(
@@ -20,27 +21,24 @@ object ShellManager {
"settings put secure accessibility_enabled 1"
)
private const val SHELL_LS_FILE = "ls -l %1\$s"
- private const val SHELL_GET_ZIP_FILE_LIST =
- "app_process -Djava.class.path=/data/local/tmp/zip.dex /data/local/tmp Zip %s"
private const val SHELL_CHECK_IS_SQLITE = "od -An -tx %1\$s |grep '694c5153'"
private const val SHELL_UNINSTALL_APP = "pm uninstall %1\$s"
private const val SHELL_CLEAR_APP_DATA = "pm clear %1\$s"
private const val SHELL_FORCE_STOP_APP = "am force-stop %1\$s"
- private val SHELL_OPEN_ADB_WIFI = arrayOf("setprop service.adb.tcp.port 5555", "stop adbd", "start adbd")
- fun getTopActivity(callback: Callback) {
- ShellUtils.runWithSu(arrayOf(SHELL_TOP_ACTIVITY), object : ShellUtils.ShellResultCallback() {
- override fun onComplete(result: CommandResult) {
- callback.onSuccess(getTopActivity(result))
- }
+ private val SHELL_OPEN_ADB_WIFI =
+ arrayOf("setprop service.adb.tcp.port 5555", "stop adbd", "start adbd")
- override fun onError(msg: String) {
- callback.onSuccess(null)
- }
- })
+ fun getTopActivity(): Single {
+ return ShellUtils.runWithSuAsync(SHELL_TOP_ACTIVITY, useBusyBox = false).map {
+ getTopActivity(it)
+ }.runOnIO()
+ }
+ private fun tabCount(str: String): Int {
+ return (str.length - str.trimStart().length) / 2
}
- fun getTopActivity(result: CommandResult): TopActivityInfo {
+ private fun getTopActivity(result: CommandResult): TopActivityInfo {
val stdout = result.getStdout()
val topActivityInfo = TopActivityInfo()
val task_s = stdout.split("TASK ")
@@ -50,10 +48,9 @@ object ShellManager {
val pattern = Pattern.compile(regex)
val matcher = pattern.matcher(task_)
if (matcher.find()) {
- topActivityInfo.activity = matcher.group().split(" ")[1]
+ topActivityInfo.setFullActivity(matcher.group().split(" ")[1])
}
- val split =
- task_.split("\n[ ]{4}[A-Z]".toRegex()).dropLastWhile { it.isEmpty() }
+ val split = task_.split("\n[ ]{4}[A-Z]".toRegex()).dropLastWhile { it.isEmpty() }
for (s in split) {
if (s.contains("iew Hierarchy")) {
for (s1 in s.split("\n".toRegex()).dropLastWhile { it.isEmpty() }) {
@@ -65,67 +62,9 @@ object ShellManager {
continue
}
topActivityInfo.viewIdHex[split1[5].substring(split1[5].indexOf("id/"))] =
- split1[4]
- }
- }
- } else if (s.contains("ocal Activity")) {
- } else if (s.contains("ctive Fragments")) {
- val list = ArrayList()
- val split2 =
- s.split("\n {6}#[0-9]+:".toRegex()).filterNot { it.isBlank() }
- for (s2 in split2) {
- if (s2.contains("mFragmentId=")) {
- val name = s2.trim { it <= ' ' }.substring(0, s2.indexOf("{") - 1) - val fragmentInfo = FragmentInfo() - list.add(fragmentInfo) - fragmentInfo.name = name - val split3 = - s2.replace("Child FragmentManager", "Child_FragmentManager").split(" ".toRegex()) - .dropLastWhile { it.isEmpty() } - for (s31 in split3) { - if (s31.contains("Child_FragmentManager")) { - break - } - val s3 = s31.replace("\n", "").replace(" ", "") - when { - s3.startsWith("mFragmentId=") -> fragmentInfo.fragmentId =
- s3.replace("mFragmentId=", "")
- s3.startsWith("mContainerId=") -> fragmentInfo.containerId =
- s3.replace("mContainerId=", "")
- s3.startsWith("mTag=") -> fragmentInfo.tag = s3.replace("mTag=", "")
- s3.startsWith("mState=") -> fragmentInfo.state =
- s3.replace("mState=", "").toInt()
- s3.startsWith("mIndex=") -> fragmentInfo.index =
- s3.replace("mIndex=", "").toInt()
- s3.startsWith("mWho=") -> fragmentInfo.who = s3.replace("mWho=", "")
- s3.startsWith("mBackStackNesting=") -> fragmentInfo.backStackNesting =
- s3.replace("mBackStackNesting=", "").toInt()
- s3.startsWith("mAdded=") -> fragmentInfo.added = s3.replace("mAdded=", "") ==
- "true"
- s3.startsWith("mRemoving=") -> fragmentInfo.removing = s3.replace(
- "mRemoving=",
- ""
- ) == "true"
- s3.startsWith("mFromLayout=") -> fragmentInfo.fromLayout = s3.replace(
- "mFromLayout=",
- ""
- ) ==
- "true"
- s3.startsWith("mInLayout=") -> fragmentInfo.inLayout = s3.replace(
- "mInLayout=",
- ""
- ) == "true"
- s3.startsWith("mHidden=") -> fragmentInfo.hidden = s3.replace("mHidden=", "") ==
- "true"
- s3.startsWith("mDetached=") -> fragmentInfo.detached = s3.replace(
- "mDetached=",
- ""
- ) == "true"
- }
- }
+ split1[4]
}
}
- topActivityInfo.fragments = list.toTypedArray()
}
}
break
@@ -151,13 +90,23 @@ object ShellManager {
}
fun getPid(packageName: String): String {
- var result: CommandResult = ShellUtils.runWithSu(String.format(SHELL_PROCESS_PID_1, packageName))
+ var result: CommandResult =
+ ShellUtils.runWithSu(
+ String.format(SHELL_PROCESS_PID_1, packageName),
+ useBusyBox = false
+ )
if (result.isSuccessful) {
return result.getStdout()
}
- result = ShellUtils.runWithSu(String.format(SHELL_PROCESS_PID_2, packageName))
+ result = ShellUtils.runWithSu(
+ String.format(SHELL_PROCESS_PID_2, packageName),
+ useBusyBox = false
+ )
if (result.isSuccessful.not()) {
- result = ShellUtils.runWithSu(String.format(SHELL_PROCESS_PID_3, packageName))
+ result = ShellUtils.runWithSu(
+ String.format(SHELL_PROCESS_PID_3, packageName),
+ useBusyBox = false
+ )
}
if (result.isSuccessful.not()) {
return ""
@@ -166,7 +115,11 @@ object ShellManager {
}
fun getSqliteFiles(packageName: String): Array {
- val dbPath = "/data/data/$packageName/databases"
+ return getSqliteFiles(Constant.dataDir, packageName)
+ }
+
+ private fun getSqliteFiles(dir: String, packageName: String): Array {
+ val dbPath = "$dir/$packageName/databases"
val list = lsDir(dbPath)
val files = ArrayList()
for (file in list) {
@@ -182,91 +135,85 @@ object ShellManager {
return files.toTypedArray()
}
- fun openAccessibilityService(callback: Callback? = null) {
- ShellUtils.runWithSu(SHELL_OPEN_ACCESSiBILITY_SERVICE, object : ShellUtils.ShellResultCallback() {
- override fun onComplete(result: CommandResult) {
- callback?.onSuccess(result.isSuccessful && result.getStdout().isEmpty())
- }
-
- override fun onError(msg: String) {
- callback?.onSuccess(false)
- }
- })
+ fun openAccessibilityService(): Single {
+ return ShellUtils.runWithSuAsync(*SHELL_OPEN_ACCESSiBILITY_SERVICE, useBusyBox = false)
+ .map {
+ it.isSuccessful && it.getStdout().isEmpty()
+ }.onErrorReturn { false }.runOnIO()
}
fun catFile(filaPath: String): String {
- val commandResult = ShellUtils.runWithSu("cat $filaPath")
- return commandResult.getStdout()
+ val result = ShellUtils.runWithSu("cat $filaPath")
+ return result.getStdout()
}
fun rmFile(file: String): Boolean {
- val commandResult = ShellUtils.runWithSu("rm -rf $file")
- return commandResult.isSuccessful
+ val result = ShellUtils.runWithSu("rm -rf $file")
+ return result.isSuccessful
}
fun modifyFile(filaPath: String, content: String): Boolean {
- val commandResult = ShellUtils.runWithSu("echo $content>> $filaPath")
- return commandResult.isSuccessful
+ val result = ShellUtils.runWithSu("echo $content>> $filaPath")
+ return result.isSuccessful
}
fun cpFile(source: String, dst: String, mod: String = "666"): Boolean {
val dir = dst.substring(0, dst.lastIndexOf("/"))
- var commandResult = ShellUtils.runWithSu("mkdir -p $dir")
- if (commandResult.isSuccessful.not()) {
+ var result = ShellUtils.runWithSu("mkdir -p $dir")
+ if (result.isSuccessful.not()) {
return false
}
- commandResult = ShellUtils.runWithSu("cp -R $source $dst && chmod $mod $dst")
- return commandResult.isSuccessful || commandResult.getStderr()?.contains("Operation not permitted") ?: false
+ result = ShellUtils.runWithSu("cp -R $source $dst && chmod $mod $dst")
+ return result.isSuccessful || result.getStderr()
+ .contains("Operation not permitted")
}
+ fun tarCF(tarPath: String, srcPath: String): Boolean {
+ val dir = tarPath.substring(0, tarPath.lastIndexOf("/"))
+ var result = ShellUtils.runWithSu("mkdir -p $dir")
+ if (result.isSuccessful.not()) {
+ return false
+ }
+ result = ShellUtils.runWithSu("tar -pcf $tarPath -C $srcPath .")
+ return result.isSuccessful || result.getStderr()
+ .contains("Operation not permitted")
+ }
+
+
fun catFile(source: String, dst: String, mod: String? = null): Boolean {
val cmds = arrayListOf()
cmds.add("cat $source> $dst")
if (mod != null) {
cmds.add("chmod $mod $dst")
}
- val commandResult = ShellUtils.runWithSu(*(cmds.toTypedArray()))
- return commandResult.isSuccessful
- }
-
- fun getZipFileList(path: String): List {
- val file = File(CommonUtils.application.cacheDir, "zip.dex")
- if (file.exists()) {
- ShellUtils.runWithSu("cp ${file.absolutePath} /data/local/tmp", "rm -rf ${file.absolutePath}")
- }
- val commandResult = ShellUtils.runWithSu(String.format(SHELL_GET_ZIP_FILE_LIST, path))
- return commandResult.stdout
+ val result = ShellUtils.runWithSu(*(cmds.toTypedArray()))
+ return result.isSuccessful
}
fun lsDir(path: String): List {
- val commandResult = ShellUtils.runWithSu("ls $path")
- return commandResult.stdout
- }
-
- fun findApkDir(packageName: String): String {
- val cmd = "ls /data/app/|grep $packageName"
- val dir = ShellUtils.runWithSu(cmd).getStdout()
- return "/data/app/$dir/base.apk"
- }
-
- fun uninstallApp(packageName: String): Boolean {
- val commandResult = ShellUtils.runWithSu(String.format(SHELL_UNINSTALL_APP, packageName))
- return commandResult.isSuccessful
+ val result = ShellUtils.runWithSu("ls $path")
+ return result.stdout
}
fun clearAppData(packageName: String): Boolean {
- val commandResult = ShellUtils.runWithSu(String.format(SHELL_CLEAR_APP_DATA, packageName))
- return commandResult.isSuccessful
+ val result = ShellUtils.runWithSu(
+ String.format(SHELL_CLEAR_APP_DATA, packageName),
+ useBusyBox = false
+ )
+ return result.isSuccessful
}
fun forceStopApp(packageName: String): Boolean {
- val commandResult = ShellUtils.runWithSu(String.format(SHELL_FORCE_STOP_APP, packageName))
- return commandResult.isSuccessful
+ val result = ShellUtils.runWithSu(
+ String.format(SHELL_FORCE_STOP_APP, packageName),
+ useBusyBox = false
+ )
+ return result.isSuccessful
}
fun openAdbWifi(): Boolean {
- val commandResult = ShellUtils.runWithSu(*SHELL_OPEN_ADB_WIFI)
- return commandResult.isSuccessful
+// val result = ShellUtils.runWithSu(*SHELL_OPEN_ADB_WIFI)
+ return false
}
}
diff --git a/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellUtils.kt b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellUtils.kt
new file mode 100644
index 0000000..10def5f
--- /dev/null
+++ b/commonutil/src/main/java/com/wrbug/developerhelper/commonutil/shell/ShellUtils.kt
@@ -0,0 +1,84 @@
+package com.wrbug.developerhelper.commonutil.shell
+
+import android.util.Log
+import com.wrbug.developerhelper.commonutil.CommonUtils
+import com.wrbug.developerhelper.commonutil.RootUtils
+import io.reactivex.rxjava3.core.Single
+import java.io.File
+
+object ShellUtils {
+ private const val TAG = "ShellUtils"
+ private const val AVAILABLE_TEST_COMMANDS = "echo -BOC- \n id"
+ val busyBoxFile by lazy {
+ File(CommonUtils.application.cacheDir, "busybox")
+ }
+
+ fun run(cmd: String, useBusyBox: Boolean = true): CommandResult {
+ return Shell.SH.run(appendBusyBox(useBusyBox, cmd)).convert()
+ }
+
+ fun runWithSuAsync(vararg cmds: String, useBusyBox: Boolean = true): Single {
+ return Single.just(cmds).map {
+ if (!RootUtils.isRoot()) {
+ throw ShellException("未开启root权限")
+ }
+ Shell.SU.run(appendBusyBox(useBusyBox, *cmds)).convert()
+ }
+ }
+
+ fun runWithSu(vararg cmd: String, useBusyBox: Boolean = true): CommandResult {
+ if (!RootUtils.isRoot()) {
+ return CommandResult(
+ emptyList(),
+ listOf("未开启root权限"),
+ -1, null
+ )
+ }
+ return Shell.SU.run(appendBusyBox(useBusyBox, *cmd)).convert()
+ }
+
+ private fun Shell.Command.Result.convert(): CommandResult {
+ return apply {
+ if (isSuccess.not()) {
+ Log.e(TAG, stderr.joinToString("\n"))
+ }
+ }.let {
+ CommandResult(it.stdout, it.stderr, exitCode, it.details)
+ }
+ }
+
+ private fun appendBusyBox(useBusyBox: Boolean, vararg cmds: String): String {
+ if (cmds.isEmpty()) {
+ return ""
+ }
+ if (useBusyBox && busyBoxFile.exists() && busyBoxFile.canExecute()) {
+ return cmds.joinToString("\n") { busyBoxFile.absolutePath + " " + it }
+ }
+ return cmds.joinToString("\n")
+ }
+
+ fun isRoot(): Boolean {
+ val result = Shell.SU.run(AVAILABLE_TEST_COMMANDS)
+ return parseAvailableResult(result.stdout, true)
+ }
+
+
+ private fun parseAvailableResult(stdout: List?, checkForRoot: Boolean): Boolean {
+ if (stdout == null) {
+ return false
+ }
+ // this is only one of many ways this can be done
+ var echoSeen = false
+ for (line in stdout) {
+ if (line.contains("uid=")) {
+ // id command is working, let's see if we are actually root
+ return !checkForRoot || line.contains("uid=0")
+ } else if (line.contains("-BOC-")) {
+ // if we end up here, at least the su command starts some kind of shell, let's hope it has root privileges -
+ // no way to know without additional native binaries
+ echoSeen = true
+ }
+ }
+ return echoSeen
+ }
+}
\ No newline at end of file
diff --git a/commonwidget/.gitignore b/commonwidget/.gitignore
deleted file mode 100644
index 796b96d..0000000
--- a/commonwidget/.gitignore
+++ /dev/null
@@ -1 +0,0 @@
-/build
diff --git a/commonwidget/build.gradle b/commonwidget/build.gradle
deleted file mode 100644
index 9e215c4..0000000
--- a/commonwidget/build.gradle
+++ /dev/null
@@ -1,36 +0,0 @@
-apply plugin: 'com.android.library'
-apply plugin: 'kotlin-android'
-
-android {
- compileSdkVersion 28
-
-
-
- defaultConfig {
- minSdkVersion 21
- targetSdkVersion 28
- versionCode 1
- versionName "1.0"
-
- testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
-
- }
-
- buildTypes {
- release {
- minifyEnabled false
- proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- }
- }
-
-}
-
-dependencies {
- implementation fileTree(include: ['*.jar'], dir: 'libs')
- testImplementation 'junit:junit:4.12'
- implementation project(':commonutil')
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
-}
-repositories {
- mavenCentral()
-}
diff --git a/commonwidget/proguard-rules.pro b/commonwidget/proguard-rules.pro
deleted file mode 100644
index f1b4245..0000000
--- a/commonwidget/proguard-rules.pro
+++ /dev/null
@@ -1,21 +0,0 @@
-# Add project specific ProGuard rules here.
-# You can control the set of applied configuration files using the
-# proguardFiles setting in build.gradle.
-#
-# For more details, see
-# http://developer.android.com/guide/developing/tools/proguard.html
-
-# If your project uses WebView with JS, uncomment the following
-# and specify the fully qualified class name to the JavaScript interface
-# class:
-#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
-# public *;
-#}
-
-# Uncomment this to preserve the line number information for
-# debugging stack traces.
-#-keepattributes SourceFile,LineNumberTable
-
-# If you keep the line number information, uncomment this to
-# hide the original source file name.
-#-renamesourcefileattribute SourceFile
diff --git a/commonwidget/src/androidTest/java/com/wrbug/developerhelper/commonwidget/ExampleInstrumentedTest.java b/commonwidget/src/androidTest/java/com/wrbug/developerhelper/commonwidget/ExampleInstrumentedTest.java
deleted file mode 100644
index 78375d2..0000000
--- a/commonwidget/src/androidTest/java/com/wrbug/developerhelper/commonwidget/ExampleInstrumentedTest.java
+++ /dev/null
@@ -1,26 +0,0 @@
-package com.wrbug.developerhelper.commonwidget;
-
-import android.content.Context;
-import android.support.test.InstrumentationRegistry;
-import android.support.test.runner.AndroidJUnit4;
-
-import org.junit.Test;
-import org.junit.runner.RunWith;
-
-import static org.junit.Assert.*;
-
-/**
- * Instrumented test, which will execute on an Android device.
- *
- * @see Testing documentation
- */
-@RunWith(AndroidJUnit4.class)
-public class ExampleInstrumentedTest {
- @Test
- public void useAppContext() {
- // Context of the app under test.
- Context appContext = InstrumentationRegistry.getTargetContext();
-
- assertEquals("com.wrbug.developerhelper.commonwidget.test", appContext.getPackageName());
- }
-}
diff --git a/commonwidget/src/main/AndroidManifest.xml b/commonwidget/src/main/AndroidManifest.xml
deleted file mode 100644
index 1099276..0000000
--- a/commonwidget/src/main/AndroidManifest.xml
+++ /dev/null
@@ -1,2 +0,0 @@
-
diff --git a/commonwidget/src/main/res/values/strings.xml b/commonwidget/src/main/res/values/strings.xml
deleted file mode 100644
index f6c0897..0000000
--- a/commonwidget/src/main/res/values/strings.xml
+++ /dev/null
@@ -1,3 +0,0 @@
-
- commonwidget
-
diff --git a/gradle.properties b/gradle.properties
index 4a5855f..02743a0 100644
--- a/gradle.properties
+++ b/gradle.properties
@@ -14,4 +14,6 @@ org.gradle.jvmargs=-Xmx1536m
# Kotlin code style for this project: "official" or "obsolete":
kotlin.code.style=official
android.useAndroidX=true
-android.enableJetifier=true
\ No newline at end of file
+android.enableJetifier=true
+kotlin.jvm.target.validation.mode = IGNORE
+android.defaults.buildfeatures.buildconfig=true
\ No newline at end of file
diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties
index 9a4163a..534885b 100644
--- a/gradle/wrapper/gradle-wrapper.properties
+++ b/gradle/wrapper/gradle-wrapper.properties
@@ -1,5 +1,6 @@
+#Wed Sep 11 19:29:17 CST 2024
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
-distributionUrl=https\://services.gradle.org/distributions/gradle-4.6-all.zip
+distributionUrl=https\://services.gradle.org/distributions/gradle-8.2-bin.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
diff --git a/ipc/build.gradle b/ipc/build.gradle
index d8f586b..773da94 100644
--- a/ipc/build.gradle
+++ b/ipc/build.gradle
@@ -1,15 +1,16 @@
apply plugin: 'com.android.library'
-apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-android'
+apply from: "${rootDir}/common.gradle"
android {
- compileSdkVersion 28
+ namespace "com.wrbug.developerhelper.ipc"
+ compileSdk 34
defaultConfig {
minSdkVersion 16
- targetSdkVersion 28
+ targetSdkVersion 34
versionCode 1
versionName "1.0"
@@ -28,14 +29,7 @@ android {
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
-
- implementation 'com.android.support:appcompat-v7:28.0.0'
- testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
implementation project(':commonutil')
}
-repositories {
- mavenCentral()
-}
diff --git a/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/ProcessDataCreator.kt b/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/ProcessDataCreator.kt
index 6b41fc4..5c2f834 100644
--- a/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/ProcessDataCreator.kt
+++ b/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/ProcessDataCreator.kt
@@ -2,12 +2,11 @@
package com.wrbug.developerhelper.ipc.processshare
-import androidx.collection.ArrayMap
import com.wrbug.developerhelper.ipc.processshare.data.IpcFileDataManager
import java.lang.reflect.Proxy
object ProcessDataCreator {
- private val map = ArrayMap, Any>()
+ private val map = hashMapOf, Any>()
fun get(clazz: Class): T {
if (map.containsKey(clazz)) {
return map[clazz] as T
diff --git a/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/data/IpcFileInfo.kt b/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/data/IpcFileInfo.kt
index 818b54b..ed6d627 100644
--- a/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/data/IpcFileInfo.kt
+++ b/ipc/src/main/java/com/wrbug/developerhelper/ipc/processshare/data/IpcFileInfo.kt
@@ -1,8 +1,8 @@
package com.wrbug.developerhelper.ipc.processshare.data
-import com.jaredrummler.android.shell.Shell
import com.wrbug.developerhelper.commonutil.Base64
import com.wrbug.developerhelper.commonutil.fromJson
+import com.wrbug.developerhelper.commonutil.shell.Shell
import com.wrbug.developerhelper.commonutil.toJson
import java.io.File
diff --git a/mmkv/build.gradle b/mmkv/build.gradle
index 1f89e36..5c77bb5 100644
--- a/mmkv/build.gradle
+++ b/mmkv/build.gradle
@@ -1,21 +1,26 @@
apply plugin: 'com.android.library'
apply plugin: 'kotlin-android'
+apply from: "${rootDir}/common.gradle"
android {
- compileSdkVersion 28
-
-
+ namespace "com.wrbug.developerhelper.mmkv"
+ compileSdk 34
defaultConfig {
- minSdkVersion 21
- targetSdkVersion 28
+ minSdkVersion 23
+ targetSdkVersion 34
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
+ compileOptions {
+ sourceCompatibility = JavaVersion.VERSION_17
+ targetCompatibility = JavaVersion.VERSION_17
+ }
+ kotlinOptions { jvmTarget = "17" }
buildTypes {
release {
minifyEnabled false
@@ -27,14 +32,7 @@ android {
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
- implementation 'com.tencent:mmkv:1.0.14'
- implementation 'androidx.appcompat:appcompat:1.0.2'
- testImplementation 'junit:junit:4.12'
- androidTestImplementation 'androidx.test:runner:1.1.0-alpha3'
- androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.0-alpha3'
- implementation 'com.google.code.gson:gson:2.8.5'
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
-}
-repositories {
- mavenCentral()
+ implementation 'com.tencent:mmkv:1.3.9'
+ implementation 'androidx.appcompat:appcompat:1.7.0'
+ implementation 'com.google.code.gson:gson:2.10'
}
diff --git a/mmkv/src/androidTest/java/com/wrbug/developerhelper/mmkv/ExampleInstrumentedTest.java b/mmkv/src/androidTest/java/com/wrbug/developerhelper/mmkv/ExampleInstrumentedTest.java
deleted file mode 100644
index 161e03c..0000000
--- a/mmkv/src/androidTest/java/com/wrbug/developerhelper/mmkv/ExampleInstrumentedTest.java
+++ /dev/null
@@ -1,26 +0,0 @@
-package com.wrbug.developerhelper.mmkv;
-
-import android.content.Context;
-import android.support.test.InstrumentationRegistry;
-import android.support.test.runner.AndroidJUnit4;
-
-import org.junit.Test;
-import org.junit.runner.RunWith;
-
-import static org.junit.Assert.*;
-
-/**
- * Instrumented test, which will execute on an Android device.
- *
- * @see Testing documentation
- */
-@RunWith(AndroidJUnit4.class)
-public class ExampleInstrumentedTest {
- @Test
- public void useAppContext() {
- // Context of the app under test.
- Context appContext = InstrumentationRegistry.getTargetContext();
-
- assertEquals("com.wrbug.developerhelper.mmkv.test", appContext.getPackageName());
- }
-}
diff --git a/settings.gradle b/settings.gradle
index fd7bea4..6dde286 100644
--- a/settings.gradle
+++ b/settings.gradle
@@ -1 +1,24 @@
-include ':app', ':basecommon', ':commonutil', ':mmkv', ':xposedmodule', ':commonwidget', ':basemoduleimport', ':ipc'
+pluginManagement {
+ repositories {
+ maven { url 'https://maven.aliyun.com/repository/public/' }
+ gradlePluginPortal()
+ google()
+ mavenCentral()
+ maven {
+ url "https://jitpack.io"
+ }
+ }
+}
+dependencyResolutionManagement {
+ repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
+ repositories {
+ maven { url 'https://maven.aliyun.com/repository/public/' }
+ google()
+ mavenCentral()
+ maven {
+ url "https://jitpack.io"
+ }
+ }
+}
+
+include ':app', ':commonutil', ':mmkv', ":ipc"
diff --git a/xposedmodule/.gitignore b/xposedmodule/.gitignore
deleted file mode 100644
index 796b96d..0000000
--- a/xposedmodule/.gitignore
+++ /dev/null
@@ -1 +0,0 @@
-/build
diff --git a/xposedmodule/CMakeLists.txt b/xposedmodule/CMakeLists.txt
deleted file mode 100644
index 73bac34..0000000
--- a/xposedmodule/CMakeLists.txt
+++ /dev/null
@@ -1,59 +0,0 @@
-# For more information about using CMake with Android Studio, read the
-# documentation: https://d.android.com/studio/projects/add-native-code.html
-
-# Sets the minimum version of CMake required to build the native library.
-
-cmake_minimum_required(VERSION 3.4.1)
-
-# Creates and names a library, sets it as either STATIC
-# or SHARED, and provides the relative paths to its source code.
-# You can define multiple libraries, and CMake builds them for you.
-# Gradle automatically packages shared libraries with your APK.
-
-add_library( # Sets the name of the library.
- nativeDump
-
- # Sets the library as a shared library.
- SHARED
-
- # Provides a relative path to your source file(s).
- src/main/cpp/native.cpp
- src/main/cpp/dlopen.c
- src/main/cpp/inlineHook.c
- src/main/cpp/relocate.c
- src/main/cpp/And64InlineHook.cpp
- src/main/cpp/util/deviceutils.cpp
- src/main/cpp/util/fileutils.cpp)
-
-# Searches for a specified prebuilt library and stores the path as a
-# variable. Because CMake includes system libraries in the search path by
-# default, you only need to specify the name of the public NDK library
-# you want to add. CMake verifies that the library exists before
-# completing its build.
-
-find_library( # Sets the name of the path variable.
- log-lib
-
- # Specifies the name of the NDK library that
- # you want CMake to locate.
- log )
-
-# Specifies libraries CMake should link to your target library. You
-# can link multiple libraries, such as libraries you define in this
-# build script, prebuilt third-party libraries, or system libraries.
-
-target_link_libraries( # Specifies the target library.
- nativeDump
-
- # Links the target library to the log library
- # included in the NDK.
- ${log-lib} )
-
-
-if(${ANDROID_ABI} STREQUAL "armeabi-v7a")
- include_directories(${ANDROID_SYSROOT}/usr/include/arm-linux-androidabi)
-elseif(${ANDROID_ABI} STREQUAL "arm64-v8a")
- include_directories(${ANDROID_SYSROOT}/usr/include/aarch64-linux-android)
-else()
- include_directories(${ANDROID_SYSROOT}/usr/include/arm-linux-androidabi)
-endif()
\ No newline at end of file
diff --git a/xposedmodule/build.gradle b/xposedmodule/build.gradle
deleted file mode 100644
index b26f5df..0000000
--- a/xposedmodule/build.gradle
+++ /dev/null
@@ -1,77 +0,0 @@
-apply plugin: 'com.android.library'
-apply plugin: 'kotlin-android'
-
-android {
- compileSdkVersion 28
-
-
-
- defaultConfig {
- minSdkVersion 21
- targetSdkVersion 28
- versionCode 1
- versionName "1.0"
-
- testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
- if(enableNdk){
- externalNativeBuild {
- cmake {
- cppFlags "-std=c++14", "-fms-extensions"
- }
- }
- ndk {
- abiFilters "armeabi-v7a", "arm64-v8a"
- }
- }
-
-
- }
-
- buildTypes {
- release {
- minifyEnabled false
- proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- }
- }
- if(enableNdk){
- externalNativeBuild {
- cmake {
- path "CMakeLists.txt"
- }
- }
- }
-
-}
-
-dependencies {
- implementation fileTree(include: ['*.jar'], dir: 'libs')
- testImplementation 'junit:junit:4.12'
- androidTestImplementation 'androidx.test:runner:1.1.1'
- androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'
- implementation 'org.jetbrains.anko:anko-commons:0.10.8'
- implementation project(':commonutil')
- implementation 'com.jaredrummler:android-shell:1.0.0'
- implementation project(':mmkv')
- compileOnly 'de.robv.android.xposed:api:82'
- implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
- implementation 'com.google.code.gson:gson:2.8.5'
- implementation project(':basecommon')
- implementation project(':ipc')
- api('com.wrbug:datafinder:0.1.3-2'){
- exclude group: 'com.google.code.gson', module: 'gson'
- }
-
-}
-repositories {
- mavenCentral()
-}
-task buildSo(dependsOn: "compileReleaseSources") {
- doLast {
- println "更新so"
- exec {
- ignoreExitValue true
- commandLine "pwd"
- commandLine "sh", "./script/so.sh"
- }
- }
-}
\ No newline at end of file
diff --git a/xposedmodule/proguard-rules.pro b/xposedmodule/proguard-rules.pro
deleted file mode 100644
index f1b4245..0000000
--- a/xposedmodule/proguard-rules.pro
+++ /dev/null
@@ -1,21 +0,0 @@
-# Add project specific ProGuard rules here.
-# You can control the set of applied configuration files using the
-# proguardFiles setting in build.gradle.
-#
-# For more details, see
-# http://developer.android.com/guide/developing/tools/proguard.html
-
-# If your project uses WebView with JS, uncomment the following
-# and specify the fully qualified class name to the JavaScript interface
-# class:
-#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
-# public *;
-#}
-
-# Uncomment this to preserve the line number information for
-# debugging stack traces.
-#-keepattributes SourceFile,LineNumberTable
-
-# If you keep the line number information, uncomment this to
-# hide the original source file name.
-#-renamesourcefileattribute SourceFile
diff --git a/xposedmodule/script/so.sh b/xposedmodule/script/so.sh
deleted file mode 100644
index 5a3a845..0000000
--- a/xposedmodule/script/so.sh
+++ /dev/null
@@ -1,11 +0,0 @@
-result=$(ls build/intermediates/cmake/release/obj)|grep 'No such file or directory'
-echo $result
-if [ -z "${result}" ]; then
- echo '目录已生成'
- cp build/intermediates/cmake/release/obj/arm64-v8a/libnativeDump.so src/main/assets/nativeDumpV8a.so
- cp build/intermediates/cmake/release/obj/armeabi-v7a/libnativeDump.so src/main/assets/nativeDumpV7a.so
- cp build/intermediates/cmake/release/obj/armeabi/libnativeDump.so src/main/assets/nativeDump.so
- echo '完成'
-else
- echo '目录未生成'
-fi
\ No newline at end of file
diff --git a/xposedmodule/src/androidTest/java/com/wrbug/developerhelper/xposed/ExampleInstrumentedTest.java b/xposedmodule/src/androidTest/java/com/wrbug/developerhelper/xposed/ExampleInstrumentedTest.java
deleted file mode 100644
index 178345d..0000000
--- a/xposedmodule/src/androidTest/java/com/wrbug/developerhelper/xposed/ExampleInstrumentedTest.java
+++ /dev/null
@@ -1,26 +0,0 @@
-package com.wrbug.developerhelper.xposed;
-
-import android.content.Context;
-import android.support.test.InstrumentationRegistry;
-import android.support.test.runner.AndroidJUnit4;
-
-import org.junit.Test;
-import org.junit.runner.RunWith;
-
-import static org.junit.Assert.*;
-
-/**
- * Instrumented test, which will execute on an Android device.
- *
- * @see Testing documentation
- */
-@RunWith(AndroidJUnit4.class)
-public class ExampleInstrumentedTest {
- @Test
- public void useAppContext() {
- // Context of the app under test.
- Context appContext = InstrumentationRegistry.getTargetContext();
-
- assertEquals("com.wrbug.developerhelper.xposed.test", appContext.getPackageName());
- }
-}
diff --git a/xposedmodule/src/main/AndroidManifest.xml b/xposedmodule/src/main/AndroidManifest.xml
deleted file mode 100644
index c249e0a..0000000
--- a/xposedmodule/src/main/AndroidManifest.xml
+++ /dev/null
@@ -1,14 +0,0 @@
-
-
-
-
-
-
-
diff --git a/xposedmodule/src/main/assets/nativeDump.zip b/xposedmodule/src/main/assets/nativeDump.zip
deleted file mode 100644
index 4b7e048..0000000
Binary files a/xposedmodule/src/main/assets/nativeDump.zip and /dev/null differ
diff --git a/xposedmodule/src/main/assets/xposed_init b/xposedmodule/src/main/assets/xposed_init
deleted file mode 100644
index 06ffbeb..0000000
--- a/xposedmodule/src/main/assets/xposed_init
+++ /dev/null
@@ -1 +0,0 @@
-com.wrbug.developerhelper.xposed.XposedInit
\ No newline at end of file
diff --git a/xposedmodule/src/main/cpp/And64InlineHook.cpp b/xposedmodule/src/main/cpp/And64InlineHook.cpp
deleted file mode 100755
index 564fbeb..0000000
--- a/xposedmodule/src/main/cpp/And64InlineHook.cpp
+++ /dev/null
@@ -1,599 +0,0 @@
-/*
- * @date : 2018年01月24日
- * @author : rrrfff@foxmail.com
- * https://github.com/rrrfff/And64InlineHook
- */
-/*
- MIT License
-
- Copyright (c) 2017 rrrfff@foxmail.com
-
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
-
- The above copyright notice and this permission notice shall be included in all
- copies or substantial portions of the Software.
-
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- SOFTWARE.
- */
-#define __STDC_FORMAT_MACROS
-
-#include
-#include
-#include
-#include
-#include
-#include
-#if defined(__aarch64__)
-
-#include "And64InlineHook.hpp"
-
-#define A64_MAX_INSTRUCTIONS 5
-#define A64_MAX_REFERENCES (A64_MAX_INSTRUCTIONS * 2)
-#define A64_NOP 0xd503201fu
-#define A64_JNIEXPORT __attribute__((visibility("default")))
-#define A64_LOGE(...) ((void)__android_log_print(ANDROID_LOG_ERROR, "A64_HOOK", __VA_ARGS__))
-#ifndef NDEBUG
-# define A64_LOGI(...) ((void)__android_log_print(ANDROID_LOG_INFO, "A64_HOOK", __VA_ARGS__))
-#else
-# define A64_LOGI(...) ((void)0)
-#endif // NDEBUG
-typedef uint32_t *__restrict *__restrict instruction;
-typedef struct {
- struct fix_info {
- uint32_t *bp;
- uint32_t ls; // left-shift counts
- uint32_t ad; // & operand
- };
- struct insns_info {
- union {
- uint64_t insu;
- int64_t ins;
- void *insp;
- };
- fix_info fmap[A64_MAX_REFERENCES];
- };
- int64_t basep;
- int64_t endp;
- insns_info dat[A64_MAX_INSTRUCTIONS];
-
-public:
- inline bool is_in_fixing_range(const int64_t absolute_addr) {
- return absolute_addr>= this->basep && absolute_addr < this->endp;
- }
-
- inline intptr_t get_ref_ins_index(const int64_t absolute_addr) {
- return static_cast((absolute_addr - this->basep) / sizeof(uint32_t));
- }
-
- inline intptr_t get_and_set_current_index(uint32_t *__restrict inp, uint32_t *__restrict outp) {
- intptr_t current_idx = this->get_ref_ins_index(reinterpret_cast(inp));
- this->dat[current_idx].insp = outp;
- return current_idx;
- }
-
- inline void reset_current_ins(const intptr_t idx, uint32_t *__restrict outp) {
- this->dat[idx].insp = outp;
- }
-
- void
- insert_fix_map(const intptr_t idx, uint32_t *bp, uint32_t ls = 0u, uint32_t ad = 0xffffffffu) {
- for (auto &f : this->dat[idx].fmap) {
- if (f.bp == NULL) {
- f.bp = bp;
- f.ls = ls;
- f.ad = ad;
- return;
- } //if
- }
- // What? GGing..
- }
-
- void process_fix_map(const intptr_t idx) {
- for (auto &f : this->dat[idx].fmap) {
- if (f.bp == NULL) break;
- *(f.bp) = *(f.bp) |
- (((int32_t(this->dat[idx].ins - reinterpret_cast(f.bp))>> 2)
- << f.ls) & f.ad); - f.bp = NULL; - } - } -} context; - -//------------------------------------------------------------------------- - -static bool __fix_branch_imm(instruction inpp, instruction outpp, context *ctxp) { - constexpr uint32_t mbits = 6u; - constexpr uint32_t mask = 0xfc000000u; // 0b11111100000000000000000000000000 - constexpr uint32_t rmask = 0x03ffffffu; // 0b00000011111111111111111111111111 - constexpr uint32_t op_b = 0x14000000u; // "b" ADDR_PCREL26 - constexpr uint32_t op_bl = 0x94000000u; // "bl" ADDR_PCREL26 - - const uint32_t ins = *(*inpp); - const uint32_t opc = ins & mask; - switch (opc) { - case op_b: - case op_bl: { - intptr_t current_idx = ctxp->get_and_set_current_index(*inpp, *outpp);
- int64_t absolute_addr = reinterpret_cast(*inpp) +
- (static_cast(ins << mbits) ->> (mbits - 2u)); // sign-extended
- int64_t new_pc_offset =
- static_cast(absolute_addr - reinterpret_cast(*outpp))
->> 2; // shifted
- bool special_fix_type = ctxp->is_in_fixing_range(absolute_addr);
- // whether the branch should be converted to absolute jump
- if (!special_fix_type && llabs(new_pc_offset)> rmask) {
- bool b_aligned = (reinterpret_cast(*outpp + 2) & 7u) == 0u;
- if (opc == op_b) {
- if (b_aligned == true) {
- (*outpp)[0] = A64_NOP;
- ctxp->reset_current_ins(current_idx, ++(*outpp));
- } //if
- (*outpp)[0] = 0x58000051u; // LDR X17, #0x8
- (*outpp)[1] = 0xd61f0220u; // BR X17
- memcpy(*outpp + 2, &absolute_addr, sizeof(absolute_addr));
- *outpp += 4;
- } else {
- if (b_aligned == false) {
- (*outpp)[0] = A64_NOP;
- ctxp->reset_current_ins(current_idx, ++(*outpp));
- } //if
- (*outpp)[0] = 0x58000071u; // LDR X17, #12
- (*outpp)[1] = 0x1000009eu; // ADR X30, #16
- (*outpp)[2] = 0xd61f0220u; // BR X17
- memcpy(*outpp + 3, &absolute_addr, sizeof(absolute_addr));
- *outpp += 5;
- } //if
- } else {
- if (special_fix_type) {
- intptr_t ref_idx = ctxp->get_ref_ins_index(absolute_addr);
- if (ref_idx <= current_idx) { - new_pc_offset = static_cast(ctxp->dat[ref_idx].ins -
- reinterpret_cast(*outpp))
->> 2;
- } else {
- ctxp->insert_fix_map(ref_idx, *outpp, 0u, rmask);
- new_pc_offset = 0;
- } //if
- } //if
-
- (*outpp)[0] = opc | (new_pc_offset & ~mask);
- ++(*outpp);
- } //if
-
- ++(*inpp);
- return ctxp->process_fix_map(current_idx), true;
- }
- }
- return false;
-}
-
-//-------------------------------------------------------------------------
-
-static bool __fix_cond_comp_test_branch(instruction inpp, instruction outpp, context *ctxp) {
- constexpr uint32_t lsb = 5u;
- constexpr uint32_t lmask01 = 0xff00001fu; // 0b11111111000000000000000000011111
- constexpr uint32_t mask0 = 0xff000010u; // 0b11111111000000000000000000010000
- constexpr uint32_t op_bc = 0x54000000u; // "b.c" ADDR_PCREL19
- constexpr uint32_t mask1 = 0x7f000000u; // 0b01111111000000000000000000000000
- constexpr uint32_t op_cbz = 0x34000000u; // "cbz" Rt, ADDR_PCREL19
- constexpr uint32_t op_cbnz = 0x35000000u; // "cbnz" Rt, ADDR_PCREL19
- constexpr uint32_t lmask2 = 0xfff8001fu; // 0b11111111111110000000000000011111
- constexpr uint32_t mask2 = 0x7f000000u; // 0b01111111000000000000000000000000
- constexpr uint32_t op_tbz = 0x36000000u; // 0b00110110000000000000000000000000 "tbz" Rt, BIT_NUM, ADDR_PCREL14
- constexpr uint32_t op_tbnz = 0x37000000u; // 0b00110111000000000000000000000000 "tbnz" Rt, BIT_NUM, ADDR_PCREL14
-
- const uint32_t ins = *(*inpp);
- uint32_t lmask = lmask01;
- if ((ins & mask0) != op_bc) {
- uint32_t opc = ins & mask1;
- if (opc != op_cbz && opc != op_cbnz) {
- opc = ins & mask2;
- if (opc != op_tbz && opc != op_tbnz) {
- return false;
- } //if
- lmask = lmask2;
- } //if
- } //if
-
- intptr_t current_idx = ctxp->get_and_set_current_index(*inpp, *outpp);
- int64_t absolute_addr = reinterpret_cast(*inpp) + ((ins & ~lmask)>> (lsb - 2u));
- int64_t new_pc_offset =
- static_cast(absolute_addr - reinterpret_cast(*outpp))>> 2; // shifted
- bool special_fix_type = ctxp->is_in_fixing_range(absolute_addr);
- if (!special_fix_type && llabs(new_pc_offset)> (~lmask>> lsb)) {
- if ((reinterpret_cast(*outpp + 4) & 7u) != 0u) {
- (*outpp)[0] = A64_NOP;
- ctxp->reset_current_ins(current_idx, ++(*outpp));
- } //if
- (*outpp)[0] = (((8u>> 2u) << lsb) & ~lmask) | (ins & lmask); // B.C #0x8 - (*outpp)[1] = 0x14000005u; // B #0x14 - (*outpp)[2] = 0x58000051u; // LDR X17, #0x8 - (*outpp)[3] = 0xd61f0220u; // BR X17 - memcpy(*outpp + 4, &absolute_addr, sizeof(absolute_addr)); - *outpp += 6; - } else { - if (special_fix_type) { - intptr_t ref_idx = ctxp->get_ref_ins_index(absolute_addr);
- if (ref_idx <= current_idx) { - new_pc_offset = static_cast(ctxp->dat[ref_idx].ins -
- reinterpret_cast(*outpp))>> 2;
- } else {
- ctxp->insert_fix_map(ref_idx, *outpp, lsb, ~lmask);
- new_pc_offset = 0;
- } //if
- } //if
-
- (*outpp)[0] = (static_cast(new_pc_offset << lsb) & ~lmask) | (ins & lmask); - ++(*outpp); - } //if - - ++(*inpp); - return ctxp->process_fix_map(current_idx), true;
-}
-
-//-------------------------------------------------------------------------
-
-static bool __fix_loadlit(instruction inpp, instruction outpp, context *ctxp) {
- const uint32_t ins = *(*inpp);
-
- // memory prefetch("prfm"), just skip it
- // http://infocenter.arm.com/help/topic/com.arm.doc.100069_0608_00_en/pge1427897420050.html
- if ((ins & 0xff000000u) == 0xd8000000u) {
- ctxp->process_fix_map(ctxp->get_and_set_current_index(*inpp, *outpp));
- ++(*inpp);
- return true;
- } //if
-
- constexpr uint32_t msb = 8u;
- constexpr uint32_t lsb = 5u;
- constexpr uint32_t mask_30 = 0x40000000u; // 0b01000000000000000000000000000000
- constexpr uint32_t mask_31 = 0x80000000u; // 0b10000000000000000000000000000000
- constexpr uint32_t lmask = 0xff00001fu; // 0b11111111000000000000000000011111
- constexpr uint32_t mask_ldr = 0xbf000000u; // 0b10111111000000000000000000000000
- constexpr uint32_t op_ldr = 0x18000000u; // 0b00011000000000000000000000000000 "LDR Wt/Xt, label" | ADDR_PCREL19
- constexpr uint32_t mask_ldrv = 0x3f000000u; // 0b00111111000000000000000000000000
- constexpr uint32_t op_ldrv = 0x1c000000u; // 0b00011100000000000000000000000000 "LDR St/Dt/Qt, label" | ADDR_PCREL19
- constexpr uint32_t mask_ldrsw = 0xff000000u; // 0b11111111000000000000000000000000
- constexpr uint32_t op_ldrsw = 0x98000000u; // "LDRSW Xt, label" | ADDR_PCREL19 | load register signed word
- // LDR S0, #0 | 0b00011100000000000000000000000000 | 32-bit
- // LDR D0, #0 | 0b01011100000000000000000000000000 | 64-bit
- // LDR Q0, #0 | 0b10011100000000000000000000000000 | 128-bit
- // INVALID | 0b11011100000000000000000000000000 | may be 256-bit
-
- uint32_t mask = mask_ldr;
- uintptr_t faligned = (ins & mask_30) ? 7u : 3u;
- if ((ins & mask_ldr) != op_ldr) {
- mask = mask_ldrv;
- if (faligned != 7u)
- faligned = (ins & mask_31) ? 15u : 3u;
- if ((ins & mask_ldrv) != op_ldrv) {
- if ((ins & mask_ldrsw) != op_ldrsw) {
- return false;
- } //if
- mask = mask_ldrsw;
- faligned = 7u;
- } //if
- } //if
-
- intptr_t current_idx = ctxp->get_and_set_current_index(*inpp, *outpp);
- int64_t absolute_addr = reinterpret_cast(*inpp) +
- ((static_cast(ins << msb)>> (msb + lsb - 2u)) & ~3u);
- int64_t new_pc_offset =
- static_cast(absolute_addr - reinterpret_cast(*outpp))>> 2; // shifted
- bool special_fix_type = ctxp->is_in_fixing_range(absolute_addr);
- // special_fix_type may encounter issue when there are mixed data and code
- if (special_fix_type || (llabs(new_pc_offset) + (faligned + 1u - 4u))> (~lmask>> lsb)) {
- while ((reinterpret_cast(*outpp + 2) & faligned) != 0u) {
- *(*outpp)++ = A64_NOP;
- }
- ctxp->reset_current_ins(current_idx, *outpp);
-
- // Note that if memory at absolute_addr is writeable (non-const), we will fail to fetch it.
- // And what's worse, we may unexpectedly overwrite something if special_fix_type is true...
- uint32_t ns = static_cast((faligned + 1) / sizeof(uint32_t));
- (*outpp)[0] = (((8u>> 2u) << lsb) & ~mask) | (ins & lmask); // LDR #0x8 - (*outpp)[1] = 0x14000001u + ns; // B #0xc - memcpy(*outpp + 2, reinterpret_cast(absolute_addr), faligned + 1);
- *outpp += 2 + ns;
- } else {
- faligned>>= 2; // new_pc_offset is shifted and 4-byte aligned
- while ((new_pc_offset & faligned) != 0) {
- *(*outpp)++ = A64_NOP;
- new_pc_offset =
- static_cast(absolute_addr - reinterpret_cast(*outpp))>> 2;
- }
- ctxp->reset_current_ins(current_idx, *outpp);
-
- (*outpp)[0] = (static_cast(new_pc_offset << lsb) & ~mask) | (ins & lmask); - ++(*outpp); - } //if - - ++(*inpp); - return ctxp->process_fix_map(current_idx), true;
-}
-
-//-------------------------------------------------------------------------
-
-static bool __fix_pcreladdr(instruction inpp, instruction outpp, context *ctxp) {
- // Load a PC-relative address into a register
- // http://infocenter.arm.com/help/topic/com.arm.doc.100069_0608_00_en/pge1427897645644.html
- constexpr uint32_t msb = 8u;
- constexpr uint32_t lsb = 5u;
- constexpr uint32_t mask = 0x9f000000u; // 0b10011111000000000000000000000000
- constexpr uint32_t rmask = 0x0000001fu; // 0b00000000000000000000000000011111
- constexpr uint32_t lmask = 0xff00001fu; // 0b11111111000000000000000000011111
- constexpr uint32_t fmask = 0x00ffffffu; // 0b00000000111111111111111111111111
- constexpr uint32_t max_val = 0x001fffffu; // 0b00000000000111111111111111111111
- constexpr uint32_t op_adr = 0x10000000u; // "adr" Rd, ADDR_PCREL21
- constexpr uint32_t op_adrp = 0x90000000u; // "adrp" Rd, ADDR_ADRP
-
- const uint32_t ins = *(*inpp);
- intptr_t current_idx;
- switch (ins & mask) {
- case op_adr: {
- current_idx = ctxp->get_and_set_current_index(*inpp, *outpp);
- int64_t lsb_bytes = static_cast(ins << 1u)>> 30u;
- int64_t absolute_addr = reinterpret_cast(*inpp) +
- (((static_cast(ins << msb)>> (msb + lsb - 2u)) &
- ~3u) | lsb_bytes);
- int64_t new_pc_offset = static_cast(absolute_addr -
- reinterpret_cast(*outpp));
- bool special_fix_type = ctxp->is_in_fixing_range(absolute_addr);
- if (!special_fix_type && llabs(new_pc_offset)> max_val) {
- if ((reinterpret_cast(*outpp + 2) & 7u) != 0u) {
- (*outpp)[0] = A64_NOP;
- ctxp->reset_current_ins(current_idx, ++(*outpp));
- } //if
-
- (*outpp)[0] =
- 0x58000000u | (((8u>> 2u) << lsb) & ~mask) | (ins & rmask); // LDR #0x8 - (*outpp)[1] = 0x14000003u; // B #0xc - memcpy(*outpp + 2, &absolute_addr, sizeof(absolute_addr)); - *outpp += 4; - } else { - if (special_fix_type) { - intptr_t ref_idx = ctxp->get_ref_ins_index(absolute_addr & ~3ull);
- if (ref_idx <= current_idx) { - new_pc_offset = static_cast(ctxp->dat[ref_idx].ins -
- reinterpret_cast(*outpp));
- } else {
- ctxp->insert_fix_map(ref_idx, *outpp, lsb, fmask);
- new_pc_offset = 0;
- } //if
- } //if
-
- // the lsb_bytes will never be changed, so we can use lmask to keep it
- (*outpp)[0] = (static_cast(new_pc_offset << (lsb - 2u)) & fmask) | - (ins & lmask); - ++(*outpp); - } //if - } - break; - case op_adrp: { - current_idx = ctxp->get_and_set_current_index(*inpp, *outpp);
- int32_t lsb_bytes = static_cast(ins << 1u)>> 30u;
- int64_t absolute_addr = (reinterpret_cast(*inpp) & ~0xfffll) +
- ((((static_cast(ins << msb)>> (msb + lsb - 2u)) &
- ~3u) | lsb_bytes) << 12); - A64_LOGI("ins = 0x%.8X, pc = %p, abs_addr = %p", - ins, *inpp, reinterpret_cast(absolute_addr));
- if (ctxp->is_in_fixing_range(absolute_addr)) {
- intptr_t ref_idx = ctxp->get_ref_ins_index(absolute_addr/* & ~3ull*/);
- if (ref_idx> current_idx) {
- // the bottom 12 bits of absolute_addr are masked out,
- // so ref_idx must be less than or equal to current_idx!
- A64_LOGE("ref_idx must be less than or equal to current_idx!");
- } //if
-
- // *absolute_addr may be changed due to relocation fixing
- A64_LOGI("What is the correct way to fix this?");
- *(*outpp)++ = ins; // 0x90000000u;
- } else {
- if ((reinterpret_cast(*outpp + 2) & 7u) != 0u) {
- (*outpp)[0] = A64_NOP;
- ctxp->reset_current_ins(current_idx, ++(*outpp));
- } //if
-
- (*outpp)[0] =
- 0x58000000u | (((8u>> 2u) << lsb) & ~mask) | (ins & rmask); // LDR #0x8 - (*outpp)[1] = 0x14000003u; // B #0xc - memcpy(*outpp + 2, &absolute_addr, sizeof(absolute_addr)); // potential overflow? - *outpp += 4; - } //if - } - break; - default: - return false; - } - - ctxp->process_fix_map(current_idx);
- ++(*inpp);
- return true;
-}
-
-//-------------------------------------------------------------------------
-
-static void __fix_instructions(uint32_t *__restrict inp, int32_t count, uint32_t *__restrict outp) {
- context ctx;
- ctx.basep = reinterpret_cast(inp);
- ctx.endp = reinterpret_cast(inp + count);
- memset(ctx.dat, 0, sizeof(ctx.dat));
- static_assert(sizeof(ctx.dat) / sizeof(ctx.dat[0]) == A64_MAX_INSTRUCTIONS,
- "please use A64_MAX_INSTRUCTIONS!");
-#ifndef NDEBUG
- if (count> A64_MAX_INSTRUCTIONS) {
- A64_LOGE("too many fixing instructions!");
- } //if
-#endif // NDEBUG
-
- while (--count>= 0) {
- if (__fix_branch_imm(&inp, &outp, &ctx)) continue;
- if (__fix_cond_comp_test_branch(&inp, &outp, &ctx)) continue;
- if (__fix_loadlit(&inp, &outp, &ctx)) continue;
- if (__fix_pcreladdr(&inp, &outp, &ctx)) continue;
-
- // without PC-relative offset
- ctx.process_fix_map(ctx.get_and_set_current_index(inp, outp));
- *(outp++) = *(inp++);
- }
-
- constexpr uint_fast64_t mask = 0x03ffffffu; // 0b00000011111111111111111111111111
- auto callback = reinterpret_cast(inp);
- auto pc_offset = static_cast(callback - reinterpret_cast(outp))>> 2;
- if (llabs(pc_offset)> mask) {
- if ((reinterpret_cast(outp + 2) & 7u) != 0u) {
- outp[0] = A64_NOP;
- ++outp;
- } //if
- outp[0] = 0x58000051u; // LDR X17, #0x8
- outp[1] = 0xd61f0220u; // BR X17
- *reinterpret_cast(outp + 2) = callback;
-// outp += 4;
- } else {
- outp[0] = 0x14000000u | (pc_offset & mask); // "B" ADDR_PCREL26
-// ++outp;
- } //if
-}
-
-//-------------------------------------------------------------------------
-
-extern "C" {
-#define __attribute __attribute__
-#define aligned(x) __aligned__(x)
-#define __intval(p) reinterpret_cast(p)
-#define __uintval(p) reinterpret_cast(p)
-#define __ptr(p) reinterpret_cast(p)
-#define __page_size 4096
-#define __page_align(n) __align_up(static_cast(n), __page_size)
-#define __ptr_align(x) __ptr(__align_down(reinterpret_cast(x), __page_size))
-#define __align_up(x, n) (((x) + ((n) - 1)) & ~((n) - 1))
-#define __align_down(x, n) ((x) & -(n))
-#define __countof(x) static_cast(sizeof(x) / sizeof((x)[0])) // must be signed
-#define __atomic_increase(p) __sync_add_and_fetch(p, 1)
-#define __sync_cmpswap(p, v, n) __sync_bool_compare_and_swap(p, v, n)
-#define __predict_true(exp) __builtin_expect((exp) != 0, 1)
-#define __flush_cache(c, n) __builtin___clear_cache(reinterpret_cast(c), reinterpret_cast(c) + n)
-#define __make_rwx(p, n) ::mprotect(__ptr_align(p), \
- __page_align(__uintval(p) + n) != __page_align(__uintval(p)) ? __page_align(n) + __page_size : __page_align(n), \
- PROT_READ | PROT_WRITE | PROT_EXEC)
-
-//-------------------------------------------------------------------------
-
-static __attribute((aligned(__page_size))) uint32_t __insns_pool[A64_MAX_BACKUPS][
- A64_MAX_INSTRUCTIONS * 10];
-
-//-------------------------------------------------------------------------
-
-class A64HookInit {
-public:
- A64HookInit() {
- __make_rwx(__insns_pool, sizeof(__insns_pool));
- A64_LOGI("insns pool initialized.");
- }
-};
-static A64HookInit __init;
-
-//-------------------------------------------------------------------------
-
-static uint32_t *FastAllocateTrampoline() {
- static_assert((A64_MAX_INSTRUCTIONS * 10 * sizeof(uint32_t)) % 8 == 0, "8-byte align");
- static volatile int32_t __index = -1;
-
- int32_t i = __atomic_increase(&__index);
- if (__predict_true(i>= 0 && i < __countof(__insns_pool))) { - return __insns_pool[i]; - } //if - - A64_LOGE("failed to allocate trampoline!"); - return NULL; -} - -//------------------------------------------------------------------------- - -A64_JNIEXPORT void *A64HookFunctionV(void *const symbol, void *const replace, - void *const rwx, const uintptr_t rwx_size) { - constexpr uint_fast64_t mask = 0x03ffffffu; // 0b00000011111111111111111111111111 - - uint32_t *trampoline = static_cast(rwx), *original = static_cast(symbol);
-
- static_assert(A64_MAX_INSTRUCTIONS>= 5, "please fix A64_MAX_INSTRUCTIONS!");
- auto pc_offset = static_cast(__intval(replace) - __intval(symbol))>> 2;
- if (llabs(pc_offset)> mask) {
- int32_t count = (reinterpret_cast(original + 2) & 7u) != 0u ? 5 : 4;
- if (trampoline) {
- if (rwx_size < count * 10u) { - return NULL; - } //if - __fix_instructions(original, count, trampoline); - } //if - - if (__make_rwx(original, 5 * sizeof(uint32_t)) == 0) { - if (count == 5) { - original[0] = A64_NOP; - ++original; - } //if - original[0] = 0x58000051u; // LDR X17, #0x8 - original[1] = 0xd61f0220u; // BR X17 - *reinterpret_cast(original + 2) = __intval(replace);
- __flush_cache(symbol, 5 * sizeof(uint32_t));
-
- A64_LOGI("inline hook %p->%p successfully! %zu bytes overwritten",
- symbol, replace, 5 * sizeof(uint32_t));
- } else {
-
- trampoline = NULL;
- } //if
- } else {
- if (trampoline) {
- if (rwx_size < 1u * 10u) { - return NULL; - } //if - __fix_instructions(original, 1, trampoline); - } //if - - if (__make_rwx(original, 1 * sizeof(uint32_t)) == 0) { - __sync_cmpswap(original, *original, - 0x14000000u | (pc_offset & mask)); // "B" ADDR_PCREL26 - __flush_cache(symbol, 1 * sizeof(uint32_t)); - - A64_LOGI("inline hook %p->%p successfully! %zu bytes overwritten",
- symbol, replace, 1 * sizeof(uint32_t));
- } else {
-
- trampoline = NULL;
- } //if
- } //if
-
- return trampoline;
-}
-
-//-------------------------------------------------------------------------
-
-A64_JNIEXPORT void A64HookFunction(void *const symbol, void *const replace, void **result) {
- void *trampoline = NULL;
- if (result != NULL) {
- trampoline = FastAllocateTrampoline();
- *result = trampoline;
- if (trampoline == NULL) return;
- } //if
-
- trampoline = A64HookFunctionV(symbol, replace, trampoline, A64_MAX_INSTRUCTIONS * 10u);
- if (trampoline == NULL && result != NULL) {
- *result = NULL;
- } //if
-}
-}
-
-#endif // defined(__aarch64__)
\ No newline at end of file
diff --git a/xposedmodule/src/main/cpp/And64InlineHook.hpp b/xposedmodule/src/main/cpp/And64InlineHook.hpp
deleted file mode 100755
index 2511754..0000000
--- a/xposedmodule/src/main/cpp/And64InlineHook.hpp
+++ /dev/null
@@ -1,42 +0,0 @@
-/*
- * @date : 2018年01月24日
- * @author : rrrfff@foxmail.com
- * https://github.com/rrrfff/And64InlineHook
- */
-/*
- MIT License
-
- Copyright (c) 2017 rrrfff@foxmail.com
-
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
-
- The above copyright notice and this permission notice shall be included in all
- copies or substantial portions of the Software.
-
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- SOFTWARE.
- */
-#pragma once
-#define A64_MAX_BACKUPS 256
-
-#ifdef __cplusplus
-extern "C" {
-#endif
-
- void A64HookFunction(void *const symbol, void *const replace, void **result);
- void *A64HookFunctionV(void *const symbol, void *const replace,
- void *const rwx, const uintptr_t rwx_size);
-
-#ifdef __cplusplus
-}
-#endif
\ No newline at end of file
diff --git a/xposedmodule/src/main/cpp/dlopen.c b/xposedmodule/src/main/cpp/dlopen.c
deleted file mode 100644
index c2a9714..0000000
--- a/xposedmodule/src/main/cpp/dlopen.c
+++ /dev/null
@@ -1,191 +0,0 @@
-/*
- *
- * @author : rrrfff@foxmail.com
- * https://github.com/rrrfff/ndk_dlopen
- *
- */
-#include "dlopen.h"
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-
-#define LOG_TAG "ndk_dlopen"
-#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
-
-static volatile int SDK_INT = 0;
-static void *quick_on_stack_back;
-static union
-{
- void *generic_stub;
- void *(*quick_on_stack_replace)(const void *param1, const void *param2,
- const void *fake_trampoline, const void *called);
-} STUBS;
-
-void JNIEXPORT ndk_init(JNIEnv *env)
-{
- if (SDK_INT <= 0) { - char sdk[PROP_VALUE_MAX]; - __system_property_get("ro.build.version.sdk", sdk); - SDK_INT = atoi(sdk); - LOGI("SDK_INT = %d", SDK_INT); - if (SDK_INT>= 24) {
- static __attribute__((__aligned__(PAGE_SIZE))) uint8_t __insns[PAGE_SIZE];
- STUBS.generic_stub = __insns;
- mprotect(__insns, sizeof(__insns), PROT_READ | PROT_WRITE | PROT_EXEC);
-
- // we are currently hijacking "FatalError" as a fake system-call trampoline
- uintptr_t pv = (uintptr_t)(*env)->FatalError;
- uintptr_t pu = (pv | (PAGE_SIZE - 1)) + 1u;
- uintptr_t pd = (pv & ~(PAGE_SIZE - 1));
- mprotect((void *)pd, pv + 8u>= pu ? PAGE_SIZE * 2u : PAGE_SIZE, PROT_READ | PROT_WRITE | PROT_EXEC);
- quick_on_stack_back = (void *)pv;
-
-#if defined(__i386__)
- /*
- DEFINE_FUNCTION art_quick_on_stack_replace
- movl 12(REG_VAR(esp)), REG_VAR(eax)
- movl (REG_VAR(esp)), REG_VAR(edx)
- movl REG_VAR(eax), (REG_VAR(esp))
- movl REG_VAR(edx), 12(REG_VAR(esp))
- pushl 16(REG_VAR(esp))
- ret
- END_FUNCTION art_quick_on_stack_replace
- */
- memcpy(__insns, "\x8B\x44\x24\x0C\x8B\x14\x24\x89\x04\x24\x89\x54\x24\x0C\xFF\x74\x24\x10\xC3", 19);
- /*
- DEFINE_FUNCTION art_quick_on_stack_back
- push 8(REG_VAR(esp))
- ret
- END_FUNCTION art_quick_on_stack_back
- */
- memcpy(quick_on_stack_back, "\xC3\xFF\x74\x24\x08\xC3", 6);
- quick_on_stack_back = (void *)(pv + 1); // inserts `ret` at first
-#elif defined(__x86_64__)
- // rdi, rsi, rdx, rcx, r8, r9
- /*
- 0x0000000000000000: 52 push rdx
- 0x0000000000000001: FF E1 jmp rcx
- */
- memcpy(__insns, "\x52\xFF\xE1", 3);
- /*
- 0x0000000000000000: C3 ret
- */
- memcpy(quick_on_stack_back, "\xC3", 1);
-#elif defined(__aarch64__)
- // x0~x7
- /*
- 0x0000000000000000: FD 7B BF A9 stp x29, x30, [sp, #-0x10]!
- 0x0000000000000004: FD 03 00 91 mov x29, sp
- 0x0000000000000008: FE 03 02 AA mov x30, x2
- 0x000000000000000C: 60 00 1F D6 br x3
- */
- memcpy(__insns, "\xFD\x7B\xBF\xA9\xFD\x03\x00\x91\xFE\x03\x02\xAA\x60\x00\x1F\xD6", 16);
- /*
- 0x0000000000000000: FD 7B C1 A8 ldp x29, x30, [sp], #0x10
- 0x0000000000000004: C0 03 5F D6 ret
- */
- memcpy(quick_on_stack_back, "\xFD\x7B\xC1\xA8\xC0\x03\x5F\xD6", 8);
-#elif defined(__arm__)
- // r0~r3
- /*
- 0x0000000000000000: 04 E0 2D E5 str lr, [sp, #-4]!
- 0x0000000000000004: 02 E0 A0 E1 mov lr, r2
- 0x0000000000000008: 13 FF 2F E1 bx r3
- */
- memcpy(__insns, "\x04\xE0\x2D\xE5\x02\xE0\xA0\xE1\x13\xFF\x2F\xE1", 12);
- if ((pv & 1u) != 0u) { // Thumb
- /*
- 0x0000000000000000: 08 BC pop {r3}
- 0x0000000000000002: 18 47 bx r3
- */
- memcpy((void *)(pv - 1), "\x08\xBC\x18\x47", 4);
- } else {
- /*
- 0x0000000000000000: 04 30 9D E4 pop {r3}
- 0x0000000000000004: 13 FF 2F E1 bx r3
- */
- memcpy(quick_on_stack_back, "\x04\x30\x9D\xE4\x13\xFF\x2F\xE1", 8);
- } //if
-#else
-# error "not supported"
-#endif
- LOGI("init done! quick_on_stack_replace = %p, quick_on_stack_back = %p",
- STUBS.generic_stub, quick_on_stack_back);
- } //if
- } //if
-}
-
-void *JNIEXPORT ndk_dlopen(const char *filename, int flag)
-{
- if (SDK_INT>= 24) {
-#if defined(__i386__) || defined(__x86_64__) || defined(__aarch64__) || defined(__arm__)
- return STUBS.quick_on_stack_replace(filename, (void *)flag,
- quick_on_stack_back, dlopen);
-#else
-# error "not supported"
-#endif
- } //if
-
- return dlopen(filename, flag);
-}
-
-int JNIEXPORT ndk_dlclose(void *handle)
-{
- if (SDK_INT>= 24) {
-#if defined(__i386__) || defined(__x86_64__) || defined(__aarch64__) || defined(__arm__)
- return (int)STUBS.quick_on_stack_replace(handle, NULL,
- quick_on_stack_back, dlclose);
-#else
-# error "not supported"
-#endif
- } //if
-
- return dlclose(handle);
-}
-
-const char *JNIEXPORT ndk_dlerror(void)
-{
- if (SDK_INT>= 24) {
-#if defined(__i386__) || defined(__x86_64__) || defined(__aarch64__) || defined(__arm__)
- return STUBS.quick_on_stack_replace(NULL, NULL,
- quick_on_stack_back, dlerror);
-#else
-# error "not supported"
-#endif
- } //if
-
- return dlerror();
-}
-
-void *JNIEXPORT ndk_dlsym(void *handle, const char *symbol)
-{
- if (SDK_INT>= 24) {
-#if defined(__i386__) || defined(__x86_64__) || defined(__aarch64__) || defined(__arm__)
- return STUBS.quick_on_stack_replace(handle, symbol,
- quick_on_stack_back, dlsym);
-
-#else
-# error "not supported"
-#endif
- } //if
-
- return dlsym(handle, symbol);
-}
-
-int JNIEXPORT ndk_dladdr(const void *ddr, Dl_info *info)
-{
- if (SDK_INT>= 24) {
-#if defined(__i386__) || defined(__x86_64__) || defined(__aarch64__) || defined(__arm__)
- return (int)STUBS.quick_on_stack_replace(ddr, info,
- quick_on_stack_back, dladdr);
-#else
-# error "not supported"
-#endif
- } //if
-
- return dladdr(ddr, info);
-}
\ No newline at end of file
diff --git a/xposedmodule/src/main/cpp/dlopen.h b/xposedmodule/src/main/cpp/dlopen.h
deleted file mode 100644
index 58374ea..0000000
--- a/xposedmodule/src/main/cpp/dlopen.h
+++ /dev/null
@@ -1,25 +0,0 @@
-/*
- *
- * @author : rrrfff@foxmail.com
- * https://github.com/rrrfff/ndk_dlopen
- *
- */
-#pragma once
-
-#include
-#include