forked from mozilla-mobile/android-components
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Closes mozilla-mobile#6915: Add addon installation confirmation dialog
- Loading branch information
Showing
6 changed files
with
621 additions
and
12 deletions.
There are no files selected for viewing
271 changes: 271 additions & 0 deletions
271
...ons/src/main/java/mozilla/components/feature/addons/ui/AddonInstallationDialogFragment.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,271 @@ | ||
/* 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 mozilla.components.feature.addons.ui | ||
|
||
import android.annotation.SuppressLint | ||
import android.app.Dialog | ||
import android.content.DialogInterface | ||
import android.graphics.Color | ||
import android.graphics.drawable.BitmapDrawable | ||
import android.graphics.drawable.ColorDrawable | ||
import android.graphics.drawable.GradientDrawable | ||
import android.os.Bundle | ||
import android.view.Gravity | ||
import android.view.LayoutInflater | ||
import android.view.View | ||
import android.view.ViewGroup | ||
import android.view.Window | ||
import android.widget.Button | ||
import android.widget.ImageView | ||
import android.widget.LinearLayout | ||
import android.widget.TextView | ||
import androidx.annotation.ColorRes | ||
import androidx.annotation.VisibleForTesting | ||
import androidx.appcompat.app.AppCompatDialogFragment | ||
import androidx.appcompat.widget.AppCompatCheckBox | ||
import androidx.core.content.ContextCompat | ||
import kotlinx.android.synthetic.main.mozac_feature_addons_fragment_dialog_addon_installed.view.* | ||
import kotlinx.coroutines.CoroutineScope | ||
import kotlinx.coroutines.Dispatchers | ||
import kotlinx.coroutines.Job | ||
import kotlinx.coroutines.launch | ||
import mozilla.components.feature.addons.Addon | ||
import mozilla.components.feature.addons.R | ||
import mozilla.components.feature.addons.amo.AddonCollectionProvider | ||
import mozilla.components.support.base.log.logger.Logger | ||
import mozilla.components.support.ktx.android.content.appName | ||
import mozilla.components.support.ktx.android.content.res.resolveAttribute | ||
import java.io.IOException | ||
|
||
private const val KEY_DIALOG_GRAVITY = "KEY_DIALOG_GRAVITY" | ||
private const val KEY_DIALOG_WIDTH_MATCH_PARENT = "KEY_DIALOG_WIDTH_MATCH_PARENT" | ||
private const val KEY_POSITIVE_BUTTON_BACKGROUND_COLOR = "KEY_POSITIVE_BUTTON_BACKGROUND_COLOR" | ||
private const val KEY_POSITIVE_BUTTON_TEXT_COLOR = "KEY_POSITIVE_BUTTON_TEXT_COLOR" | ||
private const val KEY_POSITIVE_BUTTON_RADIUS = "KEY_POSITIVE_BUTTON_RADIUS" | ||
private const val DEFAULT_VALUE = Int.MAX_VALUE | ||
|
||
/** | ||
* A dialog that shows [Addon] installation confirmation. | ||
*/ | ||
class AddonInstallationDialogFragment( | ||
private val addonCollectionProvider: AddonCollectionProvider | ||
) : AppCompatDialogFragment() { | ||
private val scope = CoroutineScope(Dispatchers.IO) | ||
private val logger = Logger("AddonInstallationDialogFragment") | ||
/** | ||
* A lambda called when the allow button is clicked. | ||
*/ | ||
var onPositiveButtonClicked: ((Addon, Boolean) -> Unit)? = null | ||
|
||
private val safeArguments get() = requireNotNull(arguments) | ||
|
||
internal val addon get() = requireNotNull(safeArguments.getParcelable<Addon>(KEY_ADDON)) | ||
private var allowPrivateBrowsing: Boolean = false | ||
|
||
internal val positiveButtonRadius | ||
get() = | ||
safeArguments.getFloat(KEY_POSITIVE_BUTTON_RADIUS, DEFAULT_VALUE.toFloat()) | ||
|
||
internal val dialogGravity: Int | ||
get() = | ||
safeArguments.getInt( | ||
KEY_DIALOG_GRAVITY, | ||
DEFAULT_VALUE | ||
) | ||
internal val dialogShouldWidthMatchParent: Boolean | ||
get() = | ||
safeArguments.getBoolean(KEY_DIALOG_WIDTH_MATCH_PARENT) | ||
|
||
internal val positiveButtonBackgroundColor | ||
get() = | ||
safeArguments.getInt( | ||
KEY_POSITIVE_BUTTON_BACKGROUND_COLOR, | ||
DEFAULT_VALUE | ||
) | ||
|
||
internal val positiveButtonTextColor | ||
get() = | ||
safeArguments.getInt( | ||
KEY_POSITIVE_BUTTON_TEXT_COLOR, | ||
DEFAULT_VALUE | ||
) | ||
|
||
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { | ||
val sheetDialog = Dialog(requireContext()) | ||
sheetDialog.requestWindowFeature(Window.FEATURE_NO_TITLE) | ||
sheetDialog.setCanceledOnTouchOutside(true) | ||
|
||
val rootView = createContainer() | ||
|
||
sheetDialog.setContainerView(rootView) | ||
|
||
sheetDialog.window?.apply { | ||
if (dialogGravity != DEFAULT_VALUE) { | ||
setGravity(dialogGravity) | ||
} | ||
|
||
if (dialogShouldWidthMatchParent) { | ||
setBackgroundDrawable(ColorDrawable(Color.TRANSPARENT)) | ||
// This must be called after addContentView, or it won't fully fill to the edge. | ||
setLayout(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT) | ||
} | ||
} | ||
|
||
return sheetDialog | ||
} | ||
|
||
override fun onDismiss(dialog: DialogInterface) { | ||
super.onDismiss(dialog) | ||
onPositiveButtonClicked?.invoke(addon, allowPrivateBrowsing) | ||
} | ||
|
||
private fun Dialog.setContainerView(rootView: View) { | ||
if (dialogShouldWidthMatchParent) { | ||
setContentView(rootView) | ||
} else { | ||
addContentView( | ||
rootView, | ||
LinearLayout.LayoutParams( | ||
LinearLayout.LayoutParams.MATCH_PARENT, | ||
LinearLayout.LayoutParams.MATCH_PARENT | ||
) | ||
) | ||
} | ||
} | ||
|
||
@SuppressLint("InflateParams") | ||
private fun createContainer(): View { | ||
val rootView = LayoutInflater.from(requireContext()).inflate( | ||
R.layout.mozac_feature_addons_fragment_dialog_addon_installed, | ||
null, | ||
false | ||
) | ||
|
||
rootView.findViewById<TextView>(R.id.title).text = | ||
requireContext().getString( | ||
R.string.mozac_feature_addons_installed_dialog_title, | ||
addon.translatedName, | ||
requireContext().appName | ||
) | ||
|
||
fetchIcon(addon, rootView.icon) | ||
|
||
val allowedInPrivateBrowsing = rootView.findViewById<AppCompatCheckBox>(R.id.allow_in_private_browsing) | ||
allowedInPrivateBrowsing.setOnCheckedChangeListener { _, isChecked -> | ||
allowPrivateBrowsing = isChecked | ||
} | ||
|
||
val positiveButton = rootView.findViewById<Button>(R.id.confirm_button) | ||
positiveButton.setOnClickListener { | ||
onPositiveButtonClicked?.invoke(addon, allowPrivateBrowsing) | ||
dismiss() | ||
} | ||
|
||
if (positiveButtonBackgroundColor != DEFAULT_VALUE) { | ||
val backgroundTintList = | ||
ContextCompat.getColorStateList(requireContext(), positiveButtonBackgroundColor) | ||
positiveButton.backgroundTintList = backgroundTintList | ||
} | ||
|
||
if (positiveButtonTextColor != DEFAULT_VALUE) { | ||
val color = ContextCompat.getColor(requireContext(), positiveButtonTextColor) | ||
positiveButton.setTextColor(color) | ||
} | ||
|
||
if (positiveButtonRadius != DEFAULT_VALUE.toFloat()) { | ||
val shape = GradientDrawable() | ||
shape.shape = GradientDrawable.RECTANGLE | ||
shape.setColor( | ||
ContextCompat.getColor( | ||
requireContext(), | ||
positiveButtonBackgroundColor | ||
) | ||
) | ||
shape.cornerRadius = positiveButtonRadius | ||
positiveButton.background = shape | ||
} | ||
|
||
return rootView | ||
} | ||
|
||
@VisibleForTesting(otherwise = VisibleForTesting.PRIVATE) | ||
internal fun fetchIcon(addon: Addon, iconView: ImageView, scope: CoroutineScope = this.scope): Job { | ||
return scope.launch { | ||
try { | ||
val iconBitmap = addonCollectionProvider.getAddonIconBitmap(addon) | ||
iconBitmap?.let { | ||
scope.launch(Dispatchers.Main) { | ||
iconView.setImageDrawable(BitmapDrawable(iconView.resources, it)) | ||
} | ||
} | ||
} catch (e: IOException) { | ||
scope.launch(Dispatchers.Main) { | ||
val context = iconView.context | ||
val att = context.theme.resolveAttribute(android.R.attr.textColorPrimary) | ||
iconView.setColorFilter(ContextCompat.getColor(context, att)) | ||
iconView.setImageDrawable(context.getDrawable(R.drawable.mozac_ic_extensions)) | ||
} | ||
logger.error("Attempt to fetch the ${addon.id} icon failed", e) | ||
} | ||
} | ||
} | ||
|
||
@Suppress("LongParameterList") | ||
companion object { | ||
/** | ||
* Returns a new instance of [AddonInstallationDialogFragment]. | ||
* @param addon The addon to show in the dialog. | ||
* @param promptsStyling Styling properties for the dialog. | ||
* @param onPositiveButtonClicked A lambda called when the allow button is clicked. | ||
*/ | ||
fun newInstance( | ||
addon: Addon, | ||
addonCollectionProvider: AddonCollectionProvider, | ||
promptsStyling: PromptsStyling? = PromptsStyling( | ||
gravity = Gravity.BOTTOM, | ||
shouldWidthMatchParent = true | ||
), | ||
onPositiveButtonClicked: ((Addon, Boolean) -> Unit)? = null | ||
): AddonInstallationDialogFragment { | ||
|
||
val fragment = AddonInstallationDialogFragment(addonCollectionProvider) | ||
val arguments = fragment.arguments ?: Bundle() | ||
|
||
arguments.apply { | ||
putParcelable(KEY_ADDON, addon) | ||
|
||
promptsStyling?.gravity?.apply { | ||
putInt(KEY_DIALOG_GRAVITY, this) | ||
} | ||
promptsStyling?.shouldWidthMatchParent?.apply { | ||
putBoolean(KEY_DIALOG_WIDTH_MATCH_PARENT, this) | ||
} | ||
promptsStyling?.positiveButtonBackgroundColor?.apply { | ||
putInt(KEY_POSITIVE_BUTTON_BACKGROUND_COLOR, this) | ||
} | ||
|
||
promptsStyling?.positiveButtonTextColor?.apply { | ||
putInt(KEY_POSITIVE_BUTTON_TEXT_COLOR, this) | ||
} | ||
} | ||
fragment.onPositiveButtonClicked = onPositiveButtonClicked | ||
fragment.arguments = arguments | ||
return fragment | ||
} | ||
} | ||
|
||
/** | ||
* Styling for the addon installation dialog. | ||
*/ | ||
data class PromptsStyling( | ||
val gravity: Int, | ||
val shouldWidthMatchParent: Boolean = false, | ||
@ColorRes | ||
val positiveButtonBackgroundColor: Int? = null, | ||
@ColorRes | ||
val positiveButtonTextColor: Int? = null, | ||
val positiveButtonRadius: Float? = null | ||
) | ||
} |
90 changes: 90 additions & 0 deletions
90
...ature/addons/src/main/res/layout/mozac_feature_addons_fragment_dialog_addon_installed.xml
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,90 @@ | ||
<!-- 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/. --> | ||
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||
xmlns:app="http://schemas.android.com/apk/res-auto" | ||
xmlns:tools="http://schemas.android.com/tools" | ||
android:layout_width="match_parent" | ||
android:layout_height="wrap_content" | ||
android:background="?android:windowBackground" | ||
android:orientation="vertical" | ||
tools:ignore="Overdraw"> | ||
|
||
<androidx.appcompat.widget.AppCompatImageView | ||
android:id="@+id/icon" | ||
android:layout_width="32dp" | ||
android:layout_height="32dp" | ||
android:layout_alignParentTop="true" | ||
android:layout_marginStart="16dp" | ||
android:layout_marginTop="16dp" | ||
android:importantForAccessibility="no" | ||
android:scaleType="fitCenter" | ||
app:srcCompat="@drawable/mozac_ic_extensions" /> | ||
|
||
<TextView | ||
android:id="@+id/title" | ||
android:layout_width="wrap_content" | ||
android:layout_height="wrap_content" | ||
android:layout_alignBaseline="@id/icon" | ||
android:layout_alignParentTop="true" | ||
android:layout_marginStart="3dp" | ||
android:layout_marginTop="16dp" | ||
android:layout_marginEnd="11dp" | ||
android:layout_toEndOf="@id/icon" | ||
android:paddingStart="5dp" | ||
android:paddingTop="4dp" | ||
android:paddingEnd="5dp" | ||
android:textColor="?android:attr/textColorPrimary" | ||
android:textSize="16sp" | ||
tools:text="@string/mozac_feature_addons_installed_dialog_title" | ||
tools:textColor="#000000" /> | ||
|
||
<TextView | ||
android:id="@+id/description" | ||
android:layout_width="wrap_content" | ||
android:layout_height="wrap_content" | ||
android:layout_below="@id/title" | ||
android:layout_alignStart="@id/title" | ||
android:layout_marginTop="16dp" | ||
android:paddingStart="5dp" | ||
android:paddingTop="4dp" | ||
android:paddingEnd="5dp" | ||
android:textColor="?android:attr/textColorPrimary" | ||
android:text="@string/mozac_feature_addons_installed_dialog_description" /> | ||
|
||
<androidx.appcompat.widget.AppCompatImageView | ||
android:layout_width="wrap_content" | ||
android:layout_height="wrap_content" | ||
android:layout_toEndOf="@+id/description" | ||
android:layout_below="@id/title" | ||
android:layout_marginTop="16dp" | ||
app:tint="?android:attr/textColorPrimary" | ||
app:srcCompat="@drawable/mozac_ic_menu" | ||
/> | ||
|
||
<androidx.appcompat.widget.AppCompatCheckBox | ||
android:id="@+id/allow_in_private_browsing" | ||
android:layout_width="wrap_content" | ||
android:layout_height="wrap_content" | ||
android:layout_below="@id/description" | ||
android:layout_alignStart="@id/title" | ||
android:layout_marginTop="16dp" | ||
android:paddingStart="5dp" | ||
android:paddingTop="4dp" | ||
android:paddingEnd="5dp" | ||
android:text="@string/mozac_feature_addons_settings_allow_in_private_browsing" /> | ||
|
||
<Button | ||
android:id="@+id/confirm_button" | ||
android:layout_width="wrap_content" | ||
android:layout_height="wrap_content" | ||
android:layout_below="@id/allow_in_private_browsing" | ||
android:layout_alignParentEnd="true" | ||
android:layout_marginStart="8dp" | ||
android:layout_marginTop="16dp" | ||
android:layout_marginEnd="16dp" | ||
android:layout_marginBottom="16dp" | ||
android:text="@string/mozac_feature_addons_installed_dialog_okay_button" | ||
android:textAllCaps="false" /> | ||
|
||
</RelativeLayout> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.