diff --git a/README.md b/README.md index 89bfe8c..9af4493 100644 --- a/README.md +++ b/README.md @@ -44,7 +44,7 @@ Added more features to app's crash dialog, fixed custom rom deleted dialog, the - 打印异常堆栈到控制台功能 -- 异常历史记录功能,可通过通知栏磁贴“异常历史记录”进入和模块主界面进入(正在开发) +- 异常历史记录功能,可通过通知栏磁贴“异常历史记录”进入和模块主界面进入 - 多进程 APP 的异常将会显示异常的进程名 @@ -56,10 +56,6 @@ Added more features to app's crash dialog, fixed custom rom deleted dialog, the - 排除列表功能 -- 模块主界面进入“异常历史记录”功能 - -- 隐藏 APP 多进程和后台进程崩溃对话框功能 - - 已忽略异常的 APP 查看功能 - 更多功能 diff --git a/app/src/main/java/com/fankes/apperrorstracking/data/DataConst.kt b/app/src/main/java/com/fankes/apperrorstracking/data/DataConst.kt index 210f43d..df1b1a7 100644 --- a/app/src/main/java/com/fankes/apperrorstracking/data/DataConst.kt +++ b/app/src/main/java/com/fankes/apperrorstracking/data/DataConst.kt @@ -26,4 +26,6 @@ import com.highcapable.yukihookapi.hook.xposed.prefs.data.PrefsData object DataConst { val ENABLE_HIDE_ICON = PrefsData("_hide_icon", false) + val ENABLE_ONLY_SHOW_ERRORS_IN_FRONT = PrefsData("_enable_only_show_errors_in_front", false) + val ENABLE_ONLY_SHOW_ERRORS_IN_MAIN = PrefsData("_enable_only_show_errors_in_main", false) } \ No newline at end of file diff --git a/app/src/main/java/com/fankes/apperrorstracking/hook/HookEntry.kt b/app/src/main/java/com/fankes/apperrorstracking/hook/HookEntry.kt index 97be73f..6a68394 100644 --- a/app/src/main/java/com/fankes/apperrorstracking/hook/HookEntry.kt +++ b/app/src/main/java/com/fankes/apperrorstracking/hook/HookEntry.kt @@ -34,6 +34,7 @@ class HookEntry : IYukiHookXposedInit { override fun onInit() = configs { debugTag = "AppErrorsTracking" isDebug = false + isEnableModulePrefsCache = false } override fun onHook() = encase { diff --git a/app/src/main/java/com/fankes/apperrorstracking/hook/entity/FrameworkHooker.kt b/app/src/main/java/com/fankes/apperrorstracking/hook/entity/FrameworkHooker.kt index d70da22..2f29378 100644 --- a/app/src/main/java/com/fankes/apperrorstracking/hook/entity/FrameworkHooker.kt +++ b/app/src/main/java/com/fankes/apperrorstracking/hook/entity/FrameworkHooker.kt @@ -30,6 +30,7 @@ import android.os.Message import com.fankes.apperrorstracking.BuildConfig import com.fankes.apperrorstracking.bean.AppErrorsDisplayBean import com.fankes.apperrorstracking.bean.AppErrorsInfoBean +import com.fankes.apperrorstracking.data.DataConst import com.fankes.apperrorstracking.locale.LocaleString import com.fankes.apperrorstracking.ui.activity.errors.AppErrorsDisplayActivity import com.fankes.apperrorstracking.utils.factory.appName @@ -47,10 +48,10 @@ import com.highcapable.yukihookapi.hook.type.android.MessageClass object FrameworkHooker : YukiBaseHooker() { + private const val ActivityManagerServiceClass = "com.android.server.am.ActivityManagerService" + private const val UserControllerClass = "com.android.server.am.UserController" private const val AppErrorsClass = "com.android.server.am.AppErrors" - private const val AppErrorDialog_DataClass = "com.android.server.am.AppErrorDialog\$Data" - private const val ProcessRecordClass = "com.android.server.am.ProcessRecord" private val PackageListClass = VariousClass( @@ -127,6 +128,9 @@ object FrameworkHooker : YukiBaseHooker() { /** 当前进程信息 */ val proc = AppErrorDialog_DataClass.clazz.field { name = "proc" }.get(errData).any() + /** 当前 UserId 信息 */ + val userId = ProcessRecordClass.clazz.field { name = "userId" }.get(proc).int() + /** 当前 APP 信息 */ val appInfo = ProcessRecordClass.clazz.field { name = "info" }.get(proc).cast() @@ -153,6 +157,15 @@ object FrameworkHooker : YukiBaseHooker() { name = "pkgList" }.get(proc).self).int() == 1 && appInfo != null) + /** 是否为主进程 */ + val isMainProcess = packageName == processName + + /** 是否为后台进程 */ + val isBackgroundProcess = UserControllerClass.clazz.method { name = "getCurrentProfileIds" } + .get(ActivityManagerServiceClass.clazz.field { name = "mUserController" } + .get(field { name = "mService" }.get(instance).any()).any()) + .invoke()?.takeIf { it.isNotEmpty() }?.any { it != userId } ?: false + /** 是否短时内重复错误 */ val isRepeating = AppErrorDialog_DataClass.clazz.field { name = "repeating" }.get(errData).boolean() /** 打印错误日志 */ @@ -165,19 +178,26 @@ object FrameworkHooker : YukiBaseHooker() { context.toast(msg = "AppErrorsTracking has crashed, please see the log in console") return@afterHook } + /** 判断是否为已忽略的 APP */ + if (ignoredErrorsIfUnlockApps.contains(packageName) || ignoredErrorsIfRestartApps.contains(packageName)) return@afterHook + /** 判断是否为后台进程 */ + if ((isBackgroundProcess || context.isAppCanOpened(packageName).not()) + && prefs.get(DataConst.ENABLE_ONLY_SHOW_ERRORS_IN_FRONT) + ) return@afterHook + /** 判断是否为主进程 */ + if (isMainProcess.not() && prefs.get(DataConst.ENABLE_ONLY_SHOW_ERRORS_IN_MAIN)) return@afterHook /** 启动错误对话框显示窗口 */ - if (ignoredErrorsIfUnlockApps.contains(packageName).not() && ignoredErrorsIfRestartApps.contains(packageName).not()) - AppErrorsDisplayActivity.start( - context, AppErrorsDisplayBean( - packageName = packageName, - processName = processName, - appName = appName, - title = if (isRepeating) LocaleString.aerrRepeatedTitle(appName) else LocaleString.aerrTitle(appName), - isShowAppInfoButton = isApp, - isShowReopenButton = isApp && isRepeating.not() && context.isAppCanOpened(packageName) && packageName == processName, - isShowCloseAppButton = isApp - ) + AppErrorsDisplayActivity.start( + context, AppErrorsDisplayBean( + packageName = packageName, + processName = processName, + appName = appName, + title = if (isRepeating) LocaleString.aerrRepeatedTitle(appName) else LocaleString.aerrTitle(appName), + isShowAppInfoButton = isApp, + isShowReopenButton = isApp && isRepeating.not() && context.isAppCanOpened(packageName) && isMainProcess, + isShowCloseAppButton = isApp ) + ) } } injectMember { diff --git a/app/src/main/java/com/fankes/apperrorstracking/ui/activity/main/MainActivity.kt b/app/src/main/java/com/fankes/apperrorstracking/ui/activity/main/MainActivity.kt index f071eb1..1158721 100644 --- a/app/src/main/java/com/fankes/apperrorstracking/ui/activity/main/MainActivity.kt +++ b/app/src/main/java/com/fankes/apperrorstracking/ui/activity/main/MainActivity.kt @@ -33,7 +33,10 @@ import com.fankes.apperrorstracking.data.DataConst import com.fankes.apperrorstracking.databinding.ActivityMainBinding import com.fankes.apperrorstracking.locale.LocaleString import com.fankes.apperrorstracking.ui.activity.base.BaseActivity +import com.fankes.apperrorstracking.ui.activity.errors.AppErrorsRecordActivity +import com.fankes.apperrorstracking.utils.factory.navigate import com.fankes.apperrorstracking.utils.factory.openBrowser +import com.fankes.apperrorstracking.utils.factory.toast import com.fankes.apperrorstracking.utils.tool.FrameworkTool import com.highcapable.yukihookapi.YukiHookAPI import com.highcapable.yukihookapi.hook.factory.modulePrefs @@ -47,6 +50,8 @@ class MainActivity : BaseActivity() { binding.mainTextVersion.text = LocaleString.moduleVersion(BuildConfig.VERSION_NAME) binding.mainTextSystemVersion.text = LocaleString.systemVersion("${Build.VERSION.RELEASE} (API ${Build.VERSION.SDK_INT}) ${Build.DISPLAY}") + binding.onlyShowErrorsInFrontSwitch.isChecked = modulePrefs.get(DataConst.ENABLE_ONLY_SHOW_ERRORS_IN_FRONT) + binding.onlyShowErrorsInMainProcessSwitch.isChecked = modulePrefs.get(DataConst.ENABLE_ONLY_SHOW_ERRORS_IN_MAIN) binding.hideIconInLauncherSwitch.isChecked = modulePrefs.get(DataConst.ENABLE_HIDE_ICON) binding.hideIconInLauncherSwitch.setOnCheckedChangeListener { btn, b -> if (btn.isPressed.not()) return@setOnCheckedChangeListener @@ -57,6 +62,23 @@ class MainActivity : BaseActivity() { PackageManager.DONT_KILL_APP ) } + binding.onlyShowErrorsInFrontSwitch.setOnCheckedChangeListener { btn, b -> + if (btn.isPressed.not()) return@setOnCheckedChangeListener + modulePrefs.put(DataConst.ENABLE_ONLY_SHOW_ERRORS_IN_FRONT, b) + } + binding.onlyShowErrorsInMainProcessSwitch.setOnCheckedChangeListener { btn, b -> + if (btn.isPressed.not()) return@setOnCheckedChangeListener + modulePrefs.put(DataConst.ENABLE_ONLY_SHOW_ERRORS_IN_MAIN, b) + } + binding.enableAppsConfigsTemplateSwitch.setOnCheckedChangeListener { btn, b -> + if (b) btn.isChecked = false + toastComingSooon() + } + /** 管理应用配置模板按钮点击事件 */ + binding.mgrAppsConfigsTemplateButton.setOnClickListener { toastComingSooon() } + /** 功能管理按钮点击事件 */ + binding.viewErrorsRecordButton.setOnClickListener { navigate() } + binding.viewIgnoredErrorsAppsButton.setOnClickListener { toastComingSooon() } /** 重启按钮点击事件 */ binding.titleRestartIcon.setOnClickListener { FrameworkTool.restartSystem(context = this) } /** 项目地址按钮点击事件 */ @@ -88,6 +110,9 @@ class MainActivity : BaseActivity() { binding.mainTextApiWay.text = "Activated by ${YukiHookAPI.Status.executorName} API ${YukiHookAPI.Status.executorVersion}" } + /** 敬请期待 */ + private fun toastComingSooon() = toast(msg = "Coming soon") + override fun onResume() { super.onResume() /** 刷新模块状态 */ diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml index fef047f..23bffd7 100644 --- a/app/src/main/res/layout/activity_main.xml +++ b/app/src/main/res/layout/activity_main.xml @@ -196,18 +196,223 @@ android:textSize="11sp" /> - + + + + + + + + + + + + + + + + + + + + + + + + + android:orientation="vertical" + android:paddingTop="15dp"> + + + + + + + + + + + + + + + + 年前 エラープロセス「%1$s」 エラースタックをシェアする + 好み設定 + 機能管理 + フロントアプリのエラーダイアログを表示 + メインプロセスのエラーダイアログを表示 + アプリ設定テンプレートを有効にする + アプリ設定テンプレートを管理する + エラー履歴を表示する + エラーを無視したアプリを表示する + 有効にすると、エラーが発生したアプリがフォアグラウンド(使用中)の場合にのみエラーダイアログが表示されます。エラーが発生したバックグラウンドアプリではエラーダイアログは表示されませんが、すべてがに記録されます。エラー履歴。 + 有効にすると、エラーダイアログは、アプリで発生したエラーがメインプロセス(最初のアプリケーションインスタンスオブジェクト)にある場合にのみ表示されます。 + ここでは、アプリごとに個別にエラーが発生したときにエラーダイアログを表示するかどうかを設定できます。 + ここでは、システムの電源を入れてから現在までのすべてのアプリエラーレコードを確認できます。エラー履歴はリスタート後に自動的にクリアされます。レコードの表示、エクスポート、共有、およびクリアが可能です。 + ここでは、さまざまな形式のエラーを手動で無視したアプリを見つけることができます。このリストは、リスタート後に自動的にクリアされます。これらの無視されたアプリを管理し、無視リストから削除できます。 \ No newline at end of file diff --git a/app/src/main/res/values-zh-rCN/strings.xml b/app/src/main/res/values-zh-rCN/strings.xml index c112a31..a044326 100644 --- a/app/src/main/res/values-zh-rCN/strings.xml +++ b/app/src/main/res/values-zh-rCN/strings.xml @@ -71,4 +71,17 @@ 年前 异常进程 \"%1$s\" 分享异常堆栈 + 偏好设置 + 功能管理 + 仅对前台应用展示错误对话框 + 仅对应用主进程展示错误对话框 + 启用应用配置模板 + 管理应用配置模板 + 查看异常历史记录 + 查看已忽略异常的应用 + 启用后,只有发生异常的应用处于前台(正在使用中)时才会展示错误对话框,发生异常的后台应用虽然不会展示错误对话框,但是它们都会被记录到异常历史记录中。 + 启用后,只有应用发生的异常位于主进程(第一个 Application 实例对象)时才会展示错误对话框。 + 你可以在这里对每个应用发生异常时,单独配置其在发生异常时是否展示错误对话框。 + 在这里,你可以找到从系统开机以来到现在为止的全部应用异常记录,异常历史记录在重新启动后会自动清空,你可以对记录进行查看、导出和分享以及清空。 + 在这里,你可以找到已被你以不同形式手动忽略异常的应用,这个列表将会在重新启动后自动清空,你可以对这些已忽略的应用进行管理以及从忽略列表中移除它们。 \ No newline at end of file diff --git a/app/src/main/res/values-zh-rHK/strings.xml b/app/src/main/res/values-zh-rHK/strings.xml index e0f9b80..5391323 100644 --- a/app/src/main/res/values-zh-rHK/strings.xml +++ b/app/src/main/res/values-zh-rHK/strings.xml @@ -71,4 +71,17 @@ 年前 異常進程 \"%1$s\" 分享異常堆棧 + 偏好設置 + 功能管理 + 僅為前台程式展示錯誤對話框 + 僅為程式主進程展示錯誤對話框 + 啟用程式配置模板 + 管理程式配置模板 + 查看異常歷史記錄 + 查看已忽略異常的程式 + 啟用後,只有發生異常的程式處於前台(正在使用中)時才會展示錯誤對話框,發生異常的後台程式雖然不會展示錯誤對話框,但是它們都會被記錄到異常歷史記錄中。 + 啟用後,只有程式發生的異常位於主進程(第一個 Application 實例對象)時才會展示錯誤對話框。 + 你可以在這裡對每個程式發生異常時,單獨配置其在發生異常時是否展示錯誤對話框。 + 在這裡,你可以找到從系統開機以來到現在為止的全部程式異常記錄,異常歷史記錄在重新啟動後會自動清空,你可以對記錄進行查看、導出和分享以及清空。 + 在這裡,你可以找到已被你以不同形式手動忽略異常的程式,這個列表將會在重新啟動後自動清空,你可以對這些已忽略的程式進行管理以及從忽略列表中移除它們。 \ No newline at end of file diff --git a/app/src/main/res/values-zh-rMO/strings.xml b/app/src/main/res/values-zh-rMO/strings.xml index 013f65d..2276b14 100644 --- a/app/src/main/res/values-zh-rMO/strings.xml +++ b/app/src/main/res/values-zh-rMO/strings.xml @@ -71,4 +71,17 @@ 年前 異常進程 \"%1$s\" 分享異常堆棧 + 偏好設置 + 功能管理 + 僅為前台程式展示錯誤對話框 + 僅為程式主進程展示錯誤對話框 + 啟用程式配置模板 + 管理程式配置模板 + 查看異常歷史記錄 + 查看已忽略異常的程式 + 啟用後,只有發生異常的程式處於前台(正在使用中)時才會展示錯誤對話框,發生異常的後台程式雖然不會展示錯誤對話框,但是它們都會被記錄到異常歷史記錄中。 + 啟用後,只有程式發生的異常位於主進程(第一個 Application 實例對象)時才會展示錯誤對話框。 + 你可以在這裡對每個程式發生異常時,單獨配置其在發生異常時是否展示錯誤對話框。 + 在這裡,你可以找到從系統開機以來到現在為止的全部程式異常記錄,異常歷史記錄在重新啟動後會自動清空,你可以對記錄進行查看、導出和分享以及清空。 + 在這裡,你可以找到已被你以不同形式手動忽略異常的程式,這個列表將會在重新啟動後自動清空,你可以對這些已忽略的程式進行管理以及從忽略列表中移除它們。 \ No newline at end of file diff --git a/app/src/main/res/values-zh-rTW/strings.xml b/app/src/main/res/values-zh-rTW/strings.xml index fed99fb..72297a4 100644 --- a/app/src/main/res/values-zh-rTW/strings.xml +++ b/app/src/main/res/values-zh-rTW/strings.xml @@ -71,4 +71,17 @@ 年前 異常進程 \"%1$s\" 分享異常堆棧 + 偏好設置 + 功能管理 + 僅為前台程式展示錯誤對話框 + 僅為程式主進程展示錯誤對話框 + 啟用程式配置模板 + 管理程式配置模板 + 查看異常歷史記錄 + 查看已忽略異常的程式 + 啟用後,只有發生異常的程式處於前台(正在使用中)時才會展示錯誤對話框,發生異常的後台程式雖然不會展示錯誤對話框,但是它們都會被記錄到異常歷史記錄中。 + 啟用後,只有程式發生的異常位於主進程(第一個 Application 實例對象)時才會展示錯誤對話框。 + 你可以在這裡對每個程式發生異常時,單獨配置其在發生異常時是否展示錯誤對話框。 + 在這裡,你可以找到從系統開機以來到現在為止的全部程式異常記錄,異常歷史記錄在重新啟動後會自動清空,你可以對記錄進行查看、導出和分享以及清空。 + 在這裡,你可以找到已被你以不同形式手動忽略異常的程式,這個列表將會在重新啟動後自動清空,你可以對這些已忽略的程式進行管理以及從忽略列表中移除它們。 \ 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 6dc94bb..1fce34c 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -69,6 +69,19 @@ days ago month ago year ago - Crash Process \"%1$s\" + Crashed Process \"%1$s\" Share error stack + Preference Settings + Function Management + Show error dialogs only for foreground apps + Show error dialogs only for main process apps + Enable apps config template + Management apps config template + View apps errors record + View ignored errors apps + After enabling, the error dialog will only be displayed when the abnormal apps is in the foreground (in use). Although the abnormal background application will not display the error dialog, they will be recorded in the errors history. + After enabling, the error dialog will only be displayed if the exception occurred in the application is in the main process (the first Application instance object). + Here you can individually configure whether to display an error dialog when an exception occurs for each apps. + Here you can find all apps exception records since the system was turned on until now. The exception history will be automatically cleared after restarting. You can view, export, share and clear the records. + Here you can find apps that you have manually ignored exceptions in different forms. This list will be automatically cleared after restarting. You can manage these ignored applications and remove them from the ignore list. \ No newline at end of file