diff --git a/app/src/main/java/foundation/e/apps/home/HomeFragment.kt b/app/src/main/java/foundation/e/apps/home/HomeFragment.kt index 69b9a2fc72e7beb5b0f9e42f590a24b52e372955..e3af670ba88a08a173bc85f227dd306924420c69 100644 --- a/app/src/main/java/foundation/e/apps/home/HomeFragment.kt +++ b/app/src/main/java/foundation/e/apps/home/HomeFragment.kt @@ -54,7 +54,11 @@ import javax.inject.Inject @AndroidEntryPoint class HomeFragment : TimeoutFragment(R.layout.fragment_home), FusedAPIInterface { - private lateinit var homeParentRVAdapter: HomeParentRVAdapter + /* + * Make adapter nullable to avoid memory leaks. + * Issue: https://gitlab.e.foundation/e/os/backlog/-/issues/485 + */ + private var homeParentRVAdapter: HomeParentRVAdapter? = null private var _binding: FragmentHomeBinding? = null private val binding get() = _binding!! @@ -155,7 +159,7 @@ class HomeFragment : TimeoutFragment(R.layout.fragment_home), FusedAPIInterface stopLoadingUI() if (it.second == ResultStatus.OK) { dismissTimeoutDialog() - homeParentRVAdapter.setData(it.first) + homeParentRVAdapter?.setData(it.first) } else { onTimeout() } @@ -211,10 +215,10 @@ class HomeFragment : TimeoutFragment(R.layout.fragment_home), FusedAPIInterface } private fun updateProgressOfDownloadingAppItemViews( - homeParentRVAdapter: HomeParentRVAdapter, + homeParentRVAdapter: HomeParentRVAdapter?, downloadProgress: DownloadProgress ) { - homeParentRVAdapter.currentList.forEach { fusedHome -> + homeParentRVAdapter?.currentList?.forEach { fusedHome -> val viewHolder = binding.parentRV.findViewHolderForAdapterPosition( homeParentRVAdapter.currentList.indexOf(fusedHome) ) @@ -269,6 +273,11 @@ class HomeFragment : TimeoutFragment(R.layout.fragment_home), FusedAPIInterface override fun onDestroyView() { super.onDestroyView() _binding = null + /* + * Nullify adapter to avoid leaks. + * Issue: https://gitlab.e.foundation/e/os/backlog/-/issues/485 + */ + homeParentRVAdapter = null } override fun getApplication(app: FusedApp, appIcon: ImageView?) { diff --git a/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt b/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt index 59fe1b90827e7abf4862a40e305e437e149d378b..ae17f2793e3084fa94aa995b2fda2b02d26f480d 100644 --- a/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt +++ b/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt @@ -48,14 +48,14 @@ import foundation.e.apps.utils.enums.User import foundation.e.apps.utils.modules.PWAManagerModule class HomeChildRVAdapter( - private val fusedAPIInterface: FusedAPIInterface, + private var fusedAPIInterface: FusedAPIInterface?, private val pkgManagerModule: PkgManagerModule, private val pwaManagerModule: PWAManagerModule, private val appInfoFetchViewModel: AppInfoFetchViewModel, private val mainActivityViewModel: MainActivityViewModel, private val user: User, - private val lifecycleOwner: LifecycleOwner, - private val paidAppHandler: ((FusedApp) -> Unit)? = null + private var lifecycleOwner: LifecycleOwner?, + private var paidAppHandler: ((FusedApp) -> Unit)? = null ) : ListAdapter(HomeChildFusedAppDiffUtil()) { private val shimmer = Shimmer.ColorHighlightBuilder() @@ -293,11 +293,13 @@ class HomeChildRVAdapter( materialButton.isEnabled = false materialButton.text = "" homeChildListItemBinding.progressBarInstall.visibility = View.VISIBLE - appInfoFetchViewModel.isAppPurchased(homeApp).observe(lifecycleOwner) { - materialButton.isEnabled = true - homeChildListItemBinding.progressBarInstall.visibility = View.GONE - materialButton.text = - if (it) materialButton.context.getString(R.string.install) else homeApp.price + lifecycleOwner?.let { + appInfoFetchViewModel.isAppPurchased(homeApp).observe(it) { + materialButton.isEnabled = true + homeChildListItemBinding.progressBarInstall.visibility = View.GONE + materialButton.text = + if (it) materialButton.context.getString(R.string.install) else homeApp.price + } } } } @@ -308,10 +310,17 @@ class HomeChildRVAdapter( } private fun installApplication(homeApp: FusedApp, appIcon: ImageView) { - fusedAPIInterface.getApplication(homeApp, appIcon) + fusedAPIInterface?.getApplication(homeApp, appIcon) } private fun cancelDownload(homeApp: FusedApp) { - fusedAPIInterface.cancelDownload(homeApp) + fusedAPIInterface?.cancelDownload(homeApp) + } + + override fun onDetachedFromRecyclerView(recyclerView: RecyclerView) { + super.onDetachedFromRecyclerView(recyclerView) + lifecycleOwner = null + paidAppHandler = null + fusedAPIInterface = null } } diff --git a/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt b/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt index 7e81364fa73ae51263a5e17ba92fb6ae95a6c7b5..1af6ce558a8a147cf8380c3fdb0eccc48ce379d7 100644 --- a/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt +++ b/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt @@ -41,12 +41,12 @@ class HomeParentRVAdapter( private val user: User, private val mainActivityViewModel: MainActivityViewModel, private val appInfoFetchViewModel: AppInfoFetchViewModel, - private val lifecycleOwner: LifecycleOwner, + private var lifecycleOwner: LifecycleOwner?, private val paidAppHandler: ((FusedApp) -> Unit)? = null ) : ListAdapter(FusedHomeDiffUtil()) { private val viewPool = RecyclerView.RecycledViewPool() - private var isDownloadObserverAdded = false + private var isDetachedFromRecyclerView = false inner class ViewHolder(val binding: HomeParentListItemBinding) : RecyclerView.ViewHolder(binding.root) @@ -91,13 +91,28 @@ class HomeParentRVAdapter( fusedHome: FusedHome, homeChildRVAdapter: RecyclerView.Adapter<*>? ) { - mainActivityViewModel.downloadList.observe(lifecycleOwner) { - mainActivityViewModel.updateStatusOfFusedApps(fusedHome.list, it) - (homeChildRVAdapter as HomeChildRVAdapter).setData(fusedHome.list) + lifecycleOwner?.let { + mainActivityViewModel.downloadList.observe(it) { + mainActivityViewModel.updateStatusOfFusedApps(fusedHome.list, it) + (homeChildRVAdapter as HomeChildRVAdapter).setData(fusedHome.list) + } } } fun setData(newList: List) { submitList(newList.map { it.copy() }) } + + override fun onDetachedFromRecyclerView(recyclerView: RecyclerView) { + super.onDetachedFromRecyclerView(recyclerView) + isDetachedFromRecyclerView = true + lifecycleOwner = null + } + + override fun onViewDetachedFromWindow(holder: ViewHolder) { + super.onViewDetachedFromWindow(holder) + if(isDetachedFromRecyclerView) { + holder.binding.childRV.adapter = null + } + } } diff --git a/app/src/main/java/foundation/e/apps/setup/tos/TOSFragment.kt b/app/src/main/java/foundation/e/apps/setup/tos/TOSFragment.kt index 2909d8c29294a9fc3fb2b658ce2548a4d458a40c..920a70f492e971d514fa8b965a5cbd7d80ce6a02 100644 --- a/app/src/main/java/foundation/e/apps/setup/tos/TOSFragment.kt +++ b/app/src/main/java/foundation/e/apps/setup/tos/TOSFragment.kt @@ -3,6 +3,7 @@ package foundation.e.apps.setup.tos import android.content.res.Configuration import android.os.Bundle import android.view.View +import android.webkit.WebView import androidx.constraintlayout.widget.ConstraintSet import androidx.fragment.app.Fragment import androidx.fragment.app.viewModels @@ -18,11 +19,18 @@ class TOSFragment : Fragment(R.layout.fragment_tos) { private var _binding: FragmentTosBinding? = null private val binding get() = _binding!! + /* + * Fix memory leaks related to WebView. + * Issue: https://gitlab.e.foundation/e/os/backlog/-/issues/485 + */ + private var webView: WebView? = null + private val viewModel: TOSViewModel by viewModels() override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) _binding = FragmentTosBinding.bind(view) + webView = binding.tosWebView var canNavigate = false viewModel.tocStatus.observe(viewLifecycleOwner) { @@ -30,7 +38,7 @@ class TOSFragment : Fragment(R.layout.fragment_tos) { view.findNavController().navigate(R.id.action_TOSFragment_to_signInFragment) } - if (it == true) { + if (it == true && webView != null) { binding.TOSWarning.visibility = View.GONE binding.TOSButtons.visibility = View.GONE binding.toolbar.visibility = View.VISIBLE @@ -38,7 +46,7 @@ class TOSFragment : Fragment(R.layout.fragment_tos) { val constraintSet = ConstraintSet() constraintSet.clone(binding.root) constraintSet.connect( - binding.tosWebView.id, + webView!!.id, ConstraintSet.TOP, binding.acceptDateTV.id, ConstraintSet.BOTTOM, @@ -66,6 +74,16 @@ class TOSFragment : Fragment(R.layout.fragment_tos) { override fun onDestroyView() { super.onDestroyView() + /* + * Fix WebView memory leaks. https://stackoverflow.com/a/19391512 + * Issue: https://gitlab.e.foundation/e/os/backlog/-/issues/485 + */ + webView?.run { + setOnScrollChangeListener(null) + removeAllViews() + destroy() + } + webView = null _binding = null } @@ -92,12 +110,12 @@ class TOSFragment : Fragment(R.layout.fragment_tos) { ) .append(body.toString()) .append("") - binding.tosWebView.loadDataWithBaseURL( + webView?.loadDataWithBaseURL( "file:///android_asset/", sb.toString(), "text/html", "utf-8", null ) - binding.tosWebView.setOnScrollChangeListener { _, scrollX, scrollY, _, _ -> + webView?.setOnScrollChangeListener { _, scrollX, scrollY, _, _ -> if (scrollX == 0 && scrollY == 0 && viewModel.tocStatus.value == true) { binding.acceptDateTV.visibility = View.VISIBLE } else {