/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ package org.mozilla.fenix.addons import android.os.Bundle import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import androidx.annotation.VisibleForTesting import androidx.core.view.isVisible import androidx.fragment.app.Fragment import androidx.lifecycle.lifecycleScope import androidx.navigation.Navigation import androidx.navigation.findNavController import androidx.navigation.fragment.findNavController import com.google.android.material.switchmaterial.SwitchMaterial import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import mozilla.components.concept.engine.webextension.EnableSource import mozilla.components.feature.addons.Addon import mozilla.components.feature.addons.AddonManager import mozilla.components.feature.addons.AddonManagerException import mozilla.components.feature.addons.ui.translateName import org.mozilla.fenix.HomeActivity import org.mozilla.fenix.R import org.mozilla.fenix.databinding.FragmentInstalledAddOnDetailsBinding import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.runIfFragmentIsAttached import org.mozilla.fenix.ext.showToolbar /** * An activity to show the details of a installed add-on. */ @Suppress("LargeClass", "TooManyFunctions") class InstalledAddonDetailsFragment : Fragment() { @VisibleForTesting(otherwise = VisibleForTesting.PRIVATE) internal lateinit var addon: Addon private var _binding: FragmentInstalledAddOnDetailsBinding? = null private val binding get() = _binding!! override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?, ): View { if (!::addon.isInitialized) { addon = AddonDetailsFragmentArgs.fromBundle(requireNotNull(arguments)).addon } _binding = FragmentInstalledAddOnDetailsBinding.inflate( inflater, container, false, ) bindUI() return binding.root } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) bindAddon() } override fun onResume() { super.onResume() context?.let { showToolbar(title = addon.translateName(it)) } } override fun onDestroyView() { super.onDestroyView() _binding = null } private fun bindAddon() { lifecycleScope.launch(Dispatchers.IO) { try { val addons = requireContext().components.addonManager.getAddons() lifecycleScope.launch(Dispatchers.Main) { runIfFragmentIsAttached { addons.find { addon.id == it.id }.let { if (it == null) { throw AddonManagerException(Exception("Addon ${addon.id} not found")) } else { addon = it bindUI() } binding.addOnProgressBar.isVisible = false binding.addonContainer.isVisible = true } } } } catch (e: AddonManagerException) { lifecycleScope.launch(Dispatchers.Main) { runIfFragmentIsAttached { showSnackBar( binding.root, getString(R.string.mozac_feature_addons_failed_to_query_add_ons), ) findNavController().popBackStack() } } } } } private fun bindUI() { bindEnableSwitch() bindSettings() bindDetails() bindPermissions() bindAllowInPrivateBrowsingSwitch() bindRemoveButton() } @SuppressWarnings("LongMethod") private fun bindEnableSwitch() { val switch = binding.enableSwitch val privateBrowsingSwitch = binding.allowInPrivateBrowsingSwitch switch.setState(addon.isEnabled()) switch.setOnCheckedChangeListener { v, isChecked -> val addonManager = v.context.components.addonManager switch.isClickable = false binding.removeAddOn.isEnabled = false if (isChecked) { enableAddon( addonManager, onSuccess = { runIfFragmentIsAttached { this.addon = it switch.isClickable = true privateBrowsingSwitch.isVisible = it.isEnabled() privateBrowsingSwitch.isChecked = it.isAllowedInPrivateBrowsing() switch.setText(R.string.mozac_feature_addons_enabled) binding.settings.isVisible = shouldSettingsBeVisible() binding.removeAddOn.isEnabled = true context?.let { showSnackBar( binding.root, getString( R.string.mozac_feature_addons_successfully_enabled, addon.translateName(it), ), ) } } }, onError = { runIfFragmentIsAttached { switch.isClickable = true binding.removeAddOn.isEnabled = true switch.setState(addon.isEnabled()) context?.let { showSnackBar( binding.root, getString( R.string.mozac_feature_addons_failed_to_enable, addon.translateName(it), ), ) } } }, ) } else { binding.settings.isVisible = false addonManager.disableAddon( addon, onSuccess = { runIfFragmentIsAttached { this.addon = it switch.isClickable = true privateBrowsingSwitch.isVisible = it.isEnabled() switch.setText(R.string.mozac_feature_addons_disabled) binding.removeAddOn.isEnabled = true context?.let { showSnackBar( binding.root, getString( R.string.mozac_feature_addons_successfully_disabled, addon.translateName(it), ), ) } } }, onError = { runIfFragmentIsAttached { switch.isClickable = true privateBrowsingSwitch.isClickable = true binding.removeAddOn.isEnabled = true switch.setState(addon.isEnabled()) context?.let { showSnackBar( binding.root, getString( R.string.mozac_feature_addons_failed_to_disable, addon.translateName(it), ), ) } } }, ) } } } @VisibleForTesting(otherwise = VisibleForTesting.PRIVATE) internal fun enableAddon( addonManager: AddonManager, onSuccess: (Addon) -> Unit, onError: (Throwable) -> Unit, ) { // If the addon is migrated from Fennec and supported in Fenix, for the addon to be enabled, // we need to also request the addon to be enabled as supported by the app if (addon.isSupported() && addon.isDisabledAsUnsupported()) { addonManager.enableAddon( addon, EnableSource.APP_SUPPORT, { enabledAddon -> addonManager.enableAddon(enabledAddon, EnableSource.USER, onSuccess, onError) }, onError, ) } else { addonManager.enableAddon(addon, EnableSource.USER, onSuccess, onError) } } private fun bindSettings() { binding.settings.apply { isVisible = shouldSettingsBeVisible() setOnClickListener { val settingUrl = addon.installedState?.optionsPageUrl ?: return@setOnClickListener val directions = if (addon.installedState?.openOptionsPageInTab == true) { val components = it.context.components val shouldCreatePrivateSession = (activity as HomeActivity).browsingModeManager.mode.isPrivate // If the addon settings page is already open in a tab, select that one components.useCases.tabsUseCases.selectOrAddTab( url = settingUrl, private = shouldCreatePrivateSession, ignoreFragment = true, ) InstalledAddonDetailsFragmentDirections.actionGlobalBrowser(null) } else { InstalledAddonDetailsFragmentDirections .actionInstalledAddonFragmentToAddonInternalSettingsFragment(addon) } Navigation.findNavController(this).navigate(directions) } } } private fun bindDetails() { binding.details.setOnClickListener { val directions = InstalledAddonDetailsFragmentDirections.actionInstalledAddonFragmentToAddonDetailsFragment( addon, ) Navigation.findNavController(binding.root).navigate(directions) } } private fun bindPermissions() { binding.permissions.setOnClickListener { val directions = InstalledAddonDetailsFragmentDirections.actionInstalledAddonFragmentToAddonPermissionsDetailsFragment( addon, ) Navigation.findNavController(binding.root).navigate(directions) } } private fun bindAllowInPrivateBrowsingSwitch() { val switch = binding.allowInPrivateBrowsingSwitch switch.isChecked = addon.isAllowedInPrivateBrowsing() switch.isVisible = addon.isEnabled() switch.setOnCheckedChangeListener { v, isChecked -> val addonManager = v.context.components.addonManager switch.isClickable = false binding.removeAddOn.isEnabled = false addonManager.setAddonAllowedInPrivateBrowsing( addon, isChecked, onSuccess = { runIfFragmentIsAttached { this.addon = it switch.isClickable = true binding.removeAddOn.isEnabled = true } }, onError = { runIfFragmentIsAttached { switch.isChecked = addon.isAllowedInPrivateBrowsing() switch.isClickable = true binding.removeAddOn.isEnabled = true } }, ) } } private fun bindRemoveButton() { binding.removeAddOn.setOnClickListener { setAllInteractiveViewsClickable(binding, false) requireContext().components.addonManager.uninstallAddon( addon, onSuccess = { runIfFragmentIsAttached { setAllInteractiveViewsClickable(binding, true) context?.let { showSnackBar( binding.root, getString( R.string.mozac_feature_addons_successfully_uninstalled, addon.translateName(it), ), ) } binding.root.findNavController().popBackStack() } }, onError = { _, _ -> runIfFragmentIsAttached { setAllInteractiveViewsClickable(binding, true) context?.let { showSnackBar( binding.root, getString( R.string.mozac_feature_addons_failed_to_uninstall, addon.translateName(it), ), ) } } }, ) } } private fun setAllInteractiveViewsClickable( binding: FragmentInstalledAddOnDetailsBinding, clickable: Boolean, ) { binding.enableSwitch.isClickable = clickable binding.settings.isClickable = clickable binding.details.isClickable = clickable binding.permissions.isClickable = clickable binding.removeAddOn.isClickable = clickable } private fun SwitchMaterial.setState(checked: Boolean) { val text = if (checked) { R.string.mozac_feature_addons_enabled } else { R.string.mozac_feature_addons_disabled } setText(text) isChecked = checked } private fun shouldSettingsBeVisible() = !addon.installedState?.optionsPageUrl.isNullOrEmpty() }