You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
372 lines
14 KiB
Kotlin
372 lines
14 KiB
Kotlin
/* 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()
|
|
}
|