diff --git a/app/src/main/java/eu/kanade/tachiyomi/data/notification/Notifications.kt b/app/src/main/java/eu/kanade/tachiyomi/data/notification/Notifications.kt
index 32c7ad2e1..6c75236ab 100644
--- a/app/src/main/java/eu/kanade/tachiyomi/data/notification/Notifications.kt
+++ b/app/src/main/java/eu/kanade/tachiyomi/data/notification/Notifications.kt
@@ -62,13 +62,19 @@ object Notifications {
const val ID_BACKUP_COMPLETE = -502
const val ID_RESTORE_COMPLETE = -504
+ /**
+ * Notification channel used for crash log file sharing.
+ */
+ const val CHANNEL_CRASH_LOGS = "crash_logs_channel"
+ const val ID_CRASH_LOGS = -601
+
// SY -->
/**
* Notification channel and ids used for backup and restore.
*/
const val CHANNEL_SIMILAR = "similar_channel"
- const val ID_SIMILAR_PROGRESS = -601
- const val ID_SIMILAR_COMPLETE = -602
+ const val ID_SIMILAR_PROGRESS = -901
+ const val ID_SIMILAR_COMPLETE = -902
// SY <--
private val deprecatedChannels = listOf(
@@ -153,6 +159,11 @@ object Notifications {
setShowBadge(false)
setSound(null, null)
},
+ NotificationChannel(
+ CHANNEL_CRASH_LOGS,
+ context.getString(R.string.channel_crash_logs),
+ NotificationManager.IMPORTANCE_HIGH
+ ),
NotificationChannel(
CHANNEL_SIMILAR,
context.getString(R.string.similar),
diff --git a/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsAdvancedController.kt b/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsAdvancedController.kt
index 3a2deed23..fc88b4b5a 100644
--- a/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsAdvancedController.kt
+++ b/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsAdvancedController.kt
@@ -25,6 +25,7 @@ import eu.kanade.tachiyomi.source.SourceManager
import eu.kanade.tachiyomi.source.SourceManager.Companion.DELEGATED_SOURCES
import eu.kanade.tachiyomi.ui.base.controller.DialogController
import eu.kanade.tachiyomi.ui.base.controller.withFadeTransaction
+import eu.kanade.tachiyomi.util.CrashLogUtil
import eu.kanade.tachiyomi.util.lang.launchUI
import eu.kanade.tachiyomi.util.preference.defaultValue
import eu.kanade.tachiyomi.util.preference.editTextPreference
@@ -68,6 +69,16 @@ class SettingsAdvancedController : SettingsController() {
override fun setupPreferenceScreen(screen: PreferenceScreen) = screen.apply {
titleRes = R.string.pref_category_advanced
+ preference {
+ key = "dump_crash_logs"
+ titleRes = R.string.pref_dump_crash_logs
+ summaryRes = R.string.pref_dump_crash_logs_summary
+
+ onClick {
+ CrashLogUtil(context).dumpLogs()
+ }
+ }
+
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
preference {
key = "pref_disable_battery_optimization"
diff --git a/app/src/main/java/eu/kanade/tachiyomi/util/CrashLogUtil.kt b/app/src/main/java/eu/kanade/tachiyomi/util/CrashLogUtil.kt
new file mode 100644
index 000000000..751bd8cd0
--- /dev/null
+++ b/app/src/main/java/eu/kanade/tachiyomi/util/CrashLogUtil.kt
@@ -0,0 +1,54 @@
+package eu.kanade.tachiyomi.util
+
+import android.content.Context
+import android.net.Uri
+import eu.kanade.tachiyomi.R
+import eu.kanade.tachiyomi.data.notification.NotificationReceiver
+import eu.kanade.tachiyomi.data.notification.Notifications
+import eu.kanade.tachiyomi.util.storage.getUriCompat
+import eu.kanade.tachiyomi.util.system.notificationBuilder
+import eu.kanade.tachiyomi.util.system.notificationManager
+import eu.kanade.tachiyomi.util.system.toast
+import java.io.File
+import java.io.IOException
+
+class CrashLogUtil(private val context: Context) {
+
+ private val notificationBuilder = context.notificationBuilder(Notifications.CHANNEL_CRASH_LOGS) {
+ setSmallIcon(R.drawable.ic_tachi)
+ }
+
+ fun dumpLogs() {
+ try {
+ val file = File(context.externalCacheDir, "tachiyomi_crash_logs.txt")
+ if (file.exists()) {
+ file.delete()
+ }
+ file.createNewFile()
+ Runtime.getRuntime().exec("logcat *:E -d -f ${file.absolutePath}")
+
+ showNotification(file.getUriCompat(context))
+ } catch (e: IOException) {
+ context.toast("Failed to get logs")
+ }
+ }
+
+ private fun showNotification(uri: Uri) {
+ context.notificationManager.cancel(Notifications.ID_CRASH_LOGS)
+
+ with(notificationBuilder) {
+ setContentTitle(context.getString(R.string.crash_log_saved))
+
+ // Clear old actions if they exist
+ clearActions()
+
+ addAction(
+ R.drawable.ic_folder_24dp,
+ context.getString(R.string.action_open_log),
+ NotificationReceiver.openErrorLogPendingActivity(context, uri)
+ )
+
+ context.notificationManager.notify(Notifications.ID_CRASH_LOGS, build())
+ }
+ }
+}
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index c8b5247fc..ef48de0f0 100755
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -273,6 +273,11 @@
Volume keys
Invert volume keys
Tapping
+ Invert tapping
+ None
+ Horizontal
+ Vertical
+ Both
Long tap dialog
Background color
White
@@ -412,6 +417,9 @@
Refresh library manga covers
Refresh tracking
Updates status, score and last chapter read from the tracking services
+ Dump crash logs
+ Saves error logs to a file for sharing with the developers
+ Crash logs saved
Disable battery optimization
Helps with background library updates and backups
Battery optimization is already disabled
@@ -731,11 +739,7 @@
Backup and restore
Chapter updates
Extension updates
- Invert tapping
- None
- Horizontal
- Vertical
- Both
+ Crash logs
Previous page