/* * Copyright (c) 2010-2023 Belledonne Communications SARL. * * This file is part of linphone-android * (see https://www.linphone.org). * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package org.linphone import android.annotation.SuppressLint import android.app.Application import android.content.ComponentCallbacks2 import androidx.annotation.MainThread import coil.ImageLoader import coil.ImageLoaderFactory import coil.decode.ImageDecoderDecoder import coil.decode.SvgDecoder import coil.decode.VideoFrameDecoder import coil.disk.DiskCache import coil.imageLoader import coil.memory.MemoryCache import coil.request.CachePolicy import com.google.android.material.color.DynamicColors import org.linphone.compatibility.Compatibility import org.linphone.core.CoreContext import org.linphone.core.CorePreferences import org.linphone.core.Factory import org.linphone.core.LogCollectionState import org.linphone.core.LogLevel import org.linphone.core.VFS import org.linphone.core.tools.Log @MainThread class LinphoneApplication : Application(), ImageLoaderFactory { companion object { private const val TAG = "[Linphone Application]" @SuppressLint("StaticFieldLeak") lateinit var corePreferences: CorePreferences @SuppressLint("StaticFieldLeak") lateinit var coreContext: CoreContext } override fun onCreate() { super.onCreate() val context = applicationContext Factory.instance().setLogCollectionPath(context.filesDir.absolutePath) Factory.instance().enableLogCollection(LogCollectionState.Enabled) // For VFS Factory.instance().setCacheDir(context.cacheDir.absolutePath) corePreferences = CorePreferences(context) corePreferences.copyAssetsFromPackage() if (VFS.isEnabled(context)) { VFS.setup(context) } val config = Factory.instance().createConfigWithFactory( corePreferences.configPath, corePreferences.factoryConfigPath ) corePreferences.config = config val appName = context.getString(R.string.app_name) Factory.instance().setLoggerDomain(appName) Factory.instance().loggingService.setLogLevel(LogLevel.Message) Factory.instance().enableLogcatLogs(corePreferences.printLogsInLogcat) Log.i("$TAG Report Core preferences initialized") Compatibility.setupAppStartupListener(context) coreContext = CoreContext(context) coreContext.start() DynamicColors.applyToActivitiesIfAvailable(this) } override fun onTrimMemory(level: Int) { Log.w("$TAG onTrimMemory called with level [${trimLevelToString(level)}]($level) !") when (level) { ComponentCallbacks2.TRIM_MEMORY_RUNNING_LOW, ComponentCallbacks2.TRIM_MEMORY_RUNNING_CRITICAL, ComponentCallbacks2.TRIM_MEMORY_MODERATE, ComponentCallbacks2.TRIM_MEMORY_COMPLETE -> { Log.i("$TAG Memory trim required, clearing imageLoader memory cache") imageLoader.memoryCache?.clear() } else -> {} } super.onTrimMemory(level) } override fun newImageLoader(): ImageLoader { // When VFS is enabled, prevent Coil from keeping plain version of files on disk val diskCachePolicy = if (VFS.isEnabled(applicationContext)) { CachePolicy.DISABLED } else { CachePolicy.ENABLED } return ImageLoader.Builder(this) .crossfade(false) .components { add(VideoFrameDecoder.Factory()) add(SvgDecoder.Factory()) add(ImageDecoderDecoder.Factory()) } .memoryCache { MemoryCache.Builder(this) .maxSizePercent(0.25) .build() } .diskCache { DiskCache.Builder() .directory(cacheDir.resolve("image_cache")) .maxSizePercent(0.02) .build() } .networkCachePolicy(CachePolicy.ENABLED) .diskCachePolicy(diskCachePolicy) .memoryCachePolicy(CachePolicy.ENABLED) .build() } private fun trimLevelToString(level: Int): String { return when (level) { ComponentCallbacks2.TRIM_MEMORY_UI_HIDDEN -> "Hidden UI" ComponentCallbacks2.TRIM_MEMORY_RUNNING_MODERATE -> "Moderate (Running)" ComponentCallbacks2.TRIM_MEMORY_RUNNING_LOW -> "Low" ComponentCallbacks2.TRIM_MEMORY_RUNNING_CRITICAL -> "Critical" ComponentCallbacks2.TRIM_MEMORY_BACKGROUND -> "Background" ComponentCallbacks2.TRIM_MEMORY_MODERATE -> "Moderate" ComponentCallbacks2.TRIM_MEMORY_COMPLETE -> "Complete" else -> level.toString() } } }