(cherry picked from commit aa498360db90350f2642e6320dc55e7d474df1fd) # Conflicts: # app/build.gradle.kts # app/src/main/java/eu/kanade/domain/base/BasePreferences.kt # app/src/main/java/eu/kanade/domain/chapter/interactor/SetReadStatus.kt # app/src/main/java/eu/kanade/presentation/library/LibrarySettingsDialog.kt # app/src/main/java/eu/kanade/presentation/more/settings/screen/SettingsAdvancedScreen.kt # app/src/main/java/eu/kanade/presentation/more/settings/screen/SettingsBrowseScreen.kt # app/src/main/java/eu/kanade/tachiyomi/App.kt # app/src/main/java/eu/kanade/tachiyomi/Migrations.kt # app/src/main/java/eu/kanade/tachiyomi/data/backup/create/BackupCreator.kt # app/src/main/java/eu/kanade/tachiyomi/data/library/LibraryUpdateJob.kt # app/src/main/java/eu/kanade/tachiyomi/data/updater/AppUpdateChecker.kt # app/src/main/java/eu/kanade/tachiyomi/ui/browse/migration/search/MigrateDialog.kt # app/src/main/java/eu/kanade/tachiyomi/ui/browse/migration/search/SourceSearchScreen.kt # app/src/main/java/eu/kanade/tachiyomi/ui/browse/source/SourcesScreenModel.kt # app/src/main/java/eu/kanade/tachiyomi/ui/browse/source/browse/BrowseSourceScreenModel.kt # app/src/main/java/eu/kanade/tachiyomi/ui/browse/source/browse/SourceFilterDialog.kt # app/src/main/java/eu/kanade/tachiyomi/ui/library/LibraryScreenModel.kt # app/src/main/java/eu/kanade/tachiyomi/ui/library/LibraryTab.kt # app/src/main/java/eu/kanade/tachiyomi/ui/main/MainActivity.kt # app/src/main/java/eu/kanade/tachiyomi/ui/manga/MangaScreen.kt # app/src/main/java/eu/kanade/tachiyomi/ui/manga/MangaScreenModel.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/ReaderActivity.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/ReaderViewModel.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/loader/ChapterLoader.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/loader/RarPageLoader.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/loader/ZipPageLoader.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/setting/ReaderPreferences.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/viewer/pager/PagerPageHolder.kt # app/src/main/java/eu/kanade/tachiyomi/ui/reader/viewer/pager/PagerViewerAdapter.kt # app/src/main/java/eu/kanade/tachiyomi/ui/stats/StatsScreenModel.kt # app/src/main/java/eu/kanade/tachiyomi/util/CrashLogUtil.kt # core/common/src/main/kotlin/eu/kanade/tachiyomi/util/storage/CbzCrypto.kt # data/src/main/java/tachiyomi/data/manga/MangaRepositoryImpl.kt # data/src/main/java/tachiyomi/data/source/SourcePagingSource.kt # domain/src/main/java/tachiyomi/domain/library/service/LibraryPreferences.kt # domain/src/main/java/tachiyomi/domain/manga/model/Manga.kt # settings.gradle.kts # source-api/src/commonMain/kotlin/eu/kanade/tachiyomi/source/online/HttpSource.kt # source-local/src/androidMain/kotlin/tachiyomi/source/local/image/LocalCoverManager.kt
355 lines
13 KiB
Kotlin
Executable File
355 lines
13 KiB
Kotlin
Executable File
package eu.kanade.tachiyomi
|
|
|
|
import android.annotation.SuppressLint
|
|
import android.app.Application
|
|
import android.app.PendingIntent
|
|
import android.content.BroadcastReceiver
|
|
import android.content.Context
|
|
import android.content.Intent
|
|
import android.content.IntentFilter
|
|
import android.os.Build
|
|
import android.os.Looper
|
|
import android.webkit.WebView
|
|
import androidx.core.content.ContextCompat
|
|
import androidx.lifecycle.DefaultLifecycleObserver
|
|
import androidx.lifecycle.LifecycleOwner
|
|
import androidx.lifecycle.ProcessLifecycleOwner
|
|
import androidx.lifecycle.lifecycleScope
|
|
import coil.ImageLoader
|
|
import coil.ImageLoaderFactory
|
|
import coil.decode.GifDecoder
|
|
import coil.decode.ImageDecoderDecoder
|
|
import coil.disk.DiskCache
|
|
import coil.util.DebugLogger
|
|
import com.elvishew.xlog.LogConfiguration
|
|
import com.elvishew.xlog.LogLevel
|
|
import com.elvishew.xlog.XLog
|
|
import com.elvishew.xlog.printer.AndroidPrinter
|
|
import com.elvishew.xlog.printer.Printer
|
|
import com.elvishew.xlog.printer.file.backup.NeverBackupStrategy
|
|
import com.elvishew.xlog.printer.file.naming.DateFileNameGenerator
|
|
import com.google.firebase.crashlytics.ktx.crashlytics
|
|
import com.google.firebase.ktx.Firebase
|
|
import eu.kanade.domain.DomainModule
|
|
import eu.kanade.domain.SYDomainModule
|
|
import eu.kanade.domain.base.BasePreferences
|
|
import eu.kanade.domain.ui.UiPreferences
|
|
import eu.kanade.domain.ui.model.setAppCompatDelegateThemeMode
|
|
import eu.kanade.tachiyomi.crash.CrashActivity
|
|
import eu.kanade.tachiyomi.crash.GlobalExceptionHandler
|
|
import eu.kanade.tachiyomi.data.coil.MangaCoverFetcher
|
|
import eu.kanade.tachiyomi.data.coil.MangaCoverKeyer
|
|
import eu.kanade.tachiyomi.data.coil.MangaKeyer
|
|
import eu.kanade.tachiyomi.data.coil.PagePreviewFetcher
|
|
import eu.kanade.tachiyomi.data.coil.PagePreviewKeyer
|
|
import eu.kanade.tachiyomi.data.coil.TachiyomiImageDecoder
|
|
import eu.kanade.tachiyomi.data.notification.Notifications
|
|
import eu.kanade.tachiyomi.di.AppModule
|
|
import eu.kanade.tachiyomi.di.PreferenceModule
|
|
import eu.kanade.tachiyomi.di.SYPreferenceModule
|
|
import eu.kanade.tachiyomi.network.NetworkHelper
|
|
import eu.kanade.tachiyomi.network.NetworkPreferences
|
|
import eu.kanade.tachiyomi.ui.base.delegate.SecureActivityDelegate
|
|
import eu.kanade.tachiyomi.util.system.DeviceUtil
|
|
import eu.kanade.tachiyomi.util.system.WebViewUtil
|
|
import eu.kanade.tachiyomi.util.system.animatorDurationScale
|
|
import eu.kanade.tachiyomi.util.system.cancelNotification
|
|
import eu.kanade.tachiyomi.util.system.isDevFlavor
|
|
import eu.kanade.tachiyomi.util.system.isReleaseBuildType
|
|
import eu.kanade.tachiyomi.util.system.notify
|
|
import exh.log.CrashlyticsPrinter
|
|
import exh.log.EHLogLevel
|
|
import exh.log.EnhancedFilePrinter
|
|
import exh.log.XLogLogcatLogger
|
|
import exh.log.xLogD
|
|
import exh.syDebugVersion
|
|
import kotlinx.coroutines.Dispatchers
|
|
import kotlinx.coroutines.flow.launchIn
|
|
import kotlinx.coroutines.flow.onEach
|
|
import logcat.LogPriority
|
|
import logcat.LogcatLogger
|
|
import org.conscrypt.Conscrypt
|
|
import tachiyomi.core.common.i18n.stringResource
|
|
import tachiyomi.core.common.util.system.logcat
|
|
import tachiyomi.domain.storage.service.StorageManager
|
|
import tachiyomi.i18n.MR
|
|
import tachiyomi.presentation.widget.WidgetManager
|
|
import uy.kohesive.injekt.Injekt
|
|
import uy.kohesive.injekt.api.get
|
|
import uy.kohesive.injekt.injectLazy
|
|
import java.security.Security
|
|
import java.text.SimpleDateFormat
|
|
import java.util.Locale
|
|
|
|
class App : Application(), DefaultLifecycleObserver, ImageLoaderFactory {
|
|
|
|
private val basePreferences: BasePreferences by injectLazy()
|
|
private val networkPreferences: NetworkPreferences by injectLazy()
|
|
|
|
private val disableIncognitoReceiver = DisableIncognitoReceiver()
|
|
|
|
@SuppressLint("LaunchActivityFromNotification")
|
|
override fun onCreate() {
|
|
super<Application>.onCreate()
|
|
|
|
// SY -->
|
|
if (!isDevFlavor) {
|
|
Firebase.crashlytics.setCrashlyticsCollectionEnabled(isReleaseBuildType)
|
|
}
|
|
// SY <--
|
|
GlobalExceptionHandler.initialize(applicationContext, CrashActivity::class.java)
|
|
|
|
// TLS 1.3 support for Android < 10
|
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q) {
|
|
Security.insertProviderAt(Conscrypt.newProvider(), 1)
|
|
}
|
|
|
|
// Avoid potential crashes
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
|
|
val process = getProcessName()
|
|
if (packageName != process) WebView.setDataDirectorySuffix(process)
|
|
}
|
|
|
|
Injekt.importModule(PreferenceModule(this))
|
|
Injekt.importModule(AppModule(this))
|
|
Injekt.importModule(DomainModule())
|
|
// SY -->
|
|
Injekt.importModule(SYPreferenceModule(this))
|
|
Injekt.importModule(SYDomainModule())
|
|
// SY <--
|
|
|
|
setupExhLogging() // EXH logging
|
|
LogcatLogger.install(XLogLogcatLogger()) // SY Redirect Logcat to XLog
|
|
|
|
setupNotificationChannels()
|
|
|
|
ProcessLifecycleOwner.get().lifecycle.addObserver(this)
|
|
|
|
// Show notification to disable Incognito Mode when it's enabled
|
|
basePreferences.incognitoMode().changes()
|
|
.onEach { enabled ->
|
|
if (enabled) {
|
|
disableIncognitoReceiver.register()
|
|
notify(
|
|
Notifications.ID_INCOGNITO_MODE,
|
|
Notifications.CHANNEL_INCOGNITO_MODE,
|
|
) {
|
|
setContentTitle(stringResource(MR.strings.pref_incognito_mode))
|
|
setContentText(stringResource(MR.strings.notification_incognito_text))
|
|
setSmallIcon(R.drawable.ic_glasses_24dp)
|
|
setOngoing(true)
|
|
|
|
val pendingIntent = PendingIntent.getBroadcast(
|
|
this@App,
|
|
0,
|
|
Intent(ACTION_DISABLE_INCOGNITO_MODE),
|
|
PendingIntent.FLAG_ONE_SHOT or PendingIntent.FLAG_IMMUTABLE,
|
|
)
|
|
setContentIntent(pendingIntent)
|
|
}
|
|
} else {
|
|
disableIncognitoReceiver.unregister()
|
|
cancelNotification(Notifications.ID_INCOGNITO_MODE)
|
|
}
|
|
}
|
|
.launchIn(ProcessLifecycleOwner.get().lifecycleScope)
|
|
|
|
setAppCompatDelegateThemeMode(Injekt.get<UiPreferences>().themeMode().get())
|
|
|
|
// Updates widget update
|
|
with(WidgetManager(Injekt.get(), Injekt.get())) {
|
|
init(ProcessLifecycleOwner.get().lifecycleScope)
|
|
}
|
|
|
|
/*if (!LogcatLogger.isInstalled && networkPreferences.verboseLogging().get()) {
|
|
LogcatLogger.install(AndroidLogcatLogger(LogPriority.VERBOSE))
|
|
}*/
|
|
}
|
|
|
|
override fun newImageLoader(): ImageLoader {
|
|
return ImageLoader.Builder(this).apply {
|
|
val callFactoryInit = { Injekt.get<NetworkHelper>().client }
|
|
val diskCacheInit = { CoilDiskCache.get(this@App) }
|
|
components {
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
|
|
add(ImageDecoderDecoder.Factory())
|
|
} else {
|
|
add(GifDecoder.Factory())
|
|
}
|
|
add(TachiyomiImageDecoder.Factory())
|
|
add(MangaCoverFetcher.MangaFactory(lazy(callFactoryInit), lazy(diskCacheInit)))
|
|
add(MangaCoverFetcher.MangaCoverFactory(lazy(callFactoryInit), lazy(diskCacheInit)))
|
|
add(MangaKeyer())
|
|
add(MangaCoverKeyer())
|
|
// SY -->
|
|
add(PagePreviewKeyer())
|
|
add(PagePreviewFetcher.Factory(lazy(callFactoryInit), lazy(diskCacheInit)))
|
|
// SY <--
|
|
}
|
|
callFactory(callFactoryInit)
|
|
diskCache(diskCacheInit)
|
|
crossfade((300 * this@App.animatorDurationScale).toInt())
|
|
allowRgb565(DeviceUtil.isLowRamDevice(this@App))
|
|
if (networkPreferences.verboseLogging().get()) logger(DebugLogger())
|
|
|
|
// Coil spawns a new thread for every image load by default
|
|
fetcherDispatcher(Dispatchers.IO.limitedParallelism(8))
|
|
decoderDispatcher(Dispatchers.IO.limitedParallelism(2))
|
|
transformationDispatcher(Dispatchers.IO.limitedParallelism(2))
|
|
}.build()
|
|
}
|
|
|
|
override fun onStart(owner: LifecycleOwner) {
|
|
SecureActivityDelegate.onApplicationStart()
|
|
}
|
|
|
|
override fun onStop(owner: LifecycleOwner) {
|
|
SecureActivityDelegate.onApplicationStopped()
|
|
}
|
|
|
|
override fun getPackageName(): String {
|
|
// This causes freezes in Android 6/7 for some reason
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
|
try {
|
|
// Override the value passed as X-Requested-With in WebView requests
|
|
val stackTrace = Looper.getMainLooper().thread.stackTrace
|
|
val chromiumElement = stackTrace.find {
|
|
it.className.equals(
|
|
"org.chromium.base.BuildInfo",
|
|
ignoreCase = true,
|
|
)
|
|
}
|
|
if (chromiumElement?.methodName.equals("getAll", ignoreCase = true)) {
|
|
return WebViewUtil.SPOOF_PACKAGE_NAME
|
|
}
|
|
} catch (_: Exception) {
|
|
}
|
|
}
|
|
return super.getPackageName()
|
|
}
|
|
|
|
private fun setupNotificationChannels() {
|
|
try {
|
|
Notifications.createChannels(this)
|
|
} catch (e: Exception) {
|
|
logcat(LogPriority.ERROR, e) { "Failed to modify notification channels" }
|
|
}
|
|
}
|
|
|
|
// EXH
|
|
private fun setupExhLogging() {
|
|
EHLogLevel.init(this)
|
|
|
|
val logLevel = when {
|
|
EHLogLevel.shouldLog(EHLogLevel.EXTREME) -> LogLevel.ALL
|
|
EHLogLevel.shouldLog(EHLogLevel.EXTRA) || BuildConfig.DEBUG -> LogLevel.DEBUG
|
|
else -> LogLevel.WARN
|
|
}
|
|
|
|
val logConfig = LogConfiguration.Builder()
|
|
.logLevel(logLevel)
|
|
.disableStackTrace()
|
|
.disableBorder()
|
|
.build()
|
|
|
|
val printers = mutableListOf<Printer>(AndroidPrinter())
|
|
|
|
val logFolder = Injekt.get<StorageManager>().getLogsDirectory()
|
|
|
|
if (logFolder != null) {
|
|
val dateFormat = SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS", Locale.getDefault())
|
|
|
|
printers += EnhancedFilePrinter
|
|
.Builder(logFolder) {
|
|
fileNameGenerator = object : DateFileNameGenerator() {
|
|
override fun generateFileName(logLevel: Int, timestamp: Long): String {
|
|
return super.generateFileName(
|
|
logLevel,
|
|
timestamp,
|
|
) + "-${BuildConfig.BUILD_TYPE}.log"
|
|
}
|
|
}
|
|
flattener { timeMillis, level, tag, message ->
|
|
"${dateFormat.format(timeMillis)} ${LogLevel.getShortLevelName(level)}/$tag: $message"
|
|
}
|
|
backupStrategy = NeverBackupStrategy()
|
|
}
|
|
}
|
|
|
|
// Install Crashlytics in prod
|
|
if (!BuildConfig.DEBUG) {
|
|
printers += CrashlyticsPrinter(LogLevel.ERROR)
|
|
}
|
|
|
|
XLog.init(
|
|
logConfig,
|
|
*printers.toTypedArray(),
|
|
)
|
|
|
|
xLogD("Application booting...")
|
|
xLogD(
|
|
"""
|
|
App version: ${BuildConfig.VERSION_NAME} (${BuildConfig.FLAVOR}, ${BuildConfig.COMMIT_SHA}, ${BuildConfig.VERSION_CODE})
|
|
Preview build: $syDebugVersion
|
|
Android version: ${Build.VERSION.RELEASE} (SDK ${Build.VERSION.SDK_INT})
|
|
Android build ID: ${Build.DISPLAY}
|
|
Device brand: ${Build.BRAND}
|
|
Device manufacturer: ${Build.MANUFACTURER}
|
|
Device name: ${Build.DEVICE}
|
|
Device model: ${Build.MODEL}
|
|
Device product name: ${Build.PRODUCT}
|
|
""".trimIndent(),
|
|
)
|
|
}
|
|
|
|
private inner class DisableIncognitoReceiver : BroadcastReceiver() {
|
|
private var registered = false
|
|
|
|
override fun onReceive(context: Context, intent: Intent) {
|
|
basePreferences.incognitoMode().set(false)
|
|
}
|
|
|
|
fun register() {
|
|
if (!registered) {
|
|
ContextCompat.registerReceiver(
|
|
this@App,
|
|
this,
|
|
IntentFilter(ACTION_DISABLE_INCOGNITO_MODE),
|
|
ContextCompat.RECEIVER_NOT_EXPORTED,
|
|
)
|
|
registered = true
|
|
}
|
|
}
|
|
|
|
fun unregister() {
|
|
if (registered) {
|
|
unregisterReceiver(this)
|
|
registered = false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private const val ACTION_DISABLE_INCOGNITO_MODE = "tachi.action.DISABLE_INCOGNITO_MODE"
|
|
|
|
/**
|
|
* Direct copy of Coil's internal SingletonDiskCache so that [MangaCoverFetcher] can access it.
|
|
*/
|
|
private object CoilDiskCache {
|
|
|
|
private const val FOLDER_NAME = "image_cache"
|
|
private var instance: DiskCache? = null
|
|
|
|
@Synchronized
|
|
fun get(context: Context): DiskCache {
|
|
return instance ?: run {
|
|
val safeCacheDir = context.cacheDir.apply { mkdirs() }
|
|
// Create the singleton disk cache instance.
|
|
DiskCache.Builder()
|
|
.directory(safeCacheDir.resolve(FOLDER_NAME))
|
|
.build()
|
|
.also { instance = it }
|
|
}
|
|
}
|
|
}
|