1
0
Fork 0

Fixes #2379 - Generic library selection

master
Tiger Oakes 2019-08-03 18:23:15 -04:00 committed by Emily Kager
parent 4bbb291e8d
commit 3c1ce90f6f
18 changed files with 200 additions and 292 deletions

View File

@ -13,11 +13,45 @@ import android.widget.TextView
import androidx.constraintlayout.widget.ConstraintLayout import androidx.constraintlayout.widget.ConstraintLayout
import androidx.core.view.isVisible import androidx.core.view.isVisible
import kotlinx.android.synthetic.main.library_site_item.view.* import kotlinx.android.synthetic.main.library_site_item.view.*
import mozilla.components.browser.menu.BrowserMenu
import mozilla.components.browser.menu.BrowserMenuBuilder
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.components
import org.mozilla.fenix.ext.increaseTapArea import org.mozilla.fenix.ext.increaseTapArea
import org.mozilla.fenix.ext.loadIntoView import org.mozilla.fenix.ext.loadIntoView
/**
* Interactor for items that can be selected on the bookmarks and history screens.
*/
interface SelectionInteractor<T> {
/**
* Called when an item is tapped to open it.
* @param item the tapped item to open.
*/
fun open(item: T)
/**
* Called when an item is long pressed and selection mode is started,
* or when selection mode has already started an an item is tapped.
* @param item the item to select.
*/
fun select(item: T)
/**
* Called when a selected item is tapped in selection mode and should no longer be selected.
* @param item the item to deselect.
*/
fun deselect(item: T)
}
interface SelectionHolder<T> {
val selectedItems: Set<T>
}
interface LibraryItemMenu {
val menuBuilder: BrowserMenuBuilder
}
class LibrarySiteItemView @JvmOverloads constructor( class LibrarySiteItemView @JvmOverloads constructor(
context: Context, context: Context,
attrs: AttributeSet? = null, attrs: AttributeSet? = null,
@ -63,6 +97,43 @@ class LibrarySiteItemView @JvmOverloads constructor(
context.components.core.icons.loadIntoView(favicon, url) context.components.core.icons.loadIntoView(favicon, url)
} }
fun attachMenu(menu: LibraryItemMenu) {
overflow_menu.setOnClickListener {
menu.menuBuilder.build(context).show(
anchor = it,
orientation = BrowserMenu.Orientation.DOWN
)
}
}
fun <T> setSelectionInteractor(item: T, holder: SelectionHolder<T>, interactor: SelectionInteractor<T>) {
setOnClickListener {
val selected = holder.selectedItems
when {
selected.isEmpty() -> interactor.open(item)
item in selected -> interactor.deselect(item)
else -> interactor.select(item)
}
}
setOnLongClickListener {
if (holder.selectedItems.isEmpty()) {
interactor.select(item)
true
} else {
false
}
}
favicon.setOnClickListener {
if (item in holder.selectedItems) {
interactor.deselect(item)
} else {
interactor.select(item)
}
}
}
enum class ItemType { enum class ItemType {
SITE, FOLDER, SEPARATOR; SITE, FOLDER, SEPARATOR;
} }

View File

@ -8,24 +8,25 @@ import android.view.View
import android.view.ViewGroup import android.view.ViewGroup
import android.view.ViewGroup.LayoutParams.MATCH_PARENT import android.view.ViewGroup.LayoutParams.MATCH_PARENT
import android.view.ViewGroup.LayoutParams.WRAP_CONTENT import android.view.ViewGroup.LayoutParams.WRAP_CONTENT
import androidx.core.view.isVisible
import androidx.recyclerview.widget.DiffUtil import androidx.recyclerview.widget.DiffUtil
import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.RecyclerView
import mozilla.appservices.places.BookmarkRoot import mozilla.appservices.places.BookmarkRoot
import mozilla.components.concept.storage.BookmarkNode import mozilla.components.concept.storage.BookmarkNode
import mozilla.components.concept.storage.BookmarkNodeType import mozilla.components.concept.storage.BookmarkNodeType
import org.mozilla.fenix.library.LibrarySiteItemView import org.mozilla.fenix.library.LibrarySiteItemView
import org.mozilla.fenix.library.SelectionHolder
import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkFolderViewHolder import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkFolderViewHolder
import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkItemViewHolder import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkItemViewHolder
import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkNodeViewHolder import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkNodeViewHolder
import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkSeparatorViewHolder import org.mozilla.fenix.library.bookmarks.viewholders.BookmarkSeparatorViewHolder
class BookmarkAdapter(val emptyView: View, val interactor: BookmarkViewInteractor) : class BookmarkAdapter(val emptyView: View, val interactor: BookmarkViewInteractor) :
RecyclerView.Adapter<BookmarkNodeViewHolder>() { RecyclerView.Adapter<BookmarkNodeViewHolder>(), SelectionHolder<BookmarkNode> {
private var tree: List<BookmarkNode> = listOf() private var tree: List<BookmarkNode> = listOf()
private var mode: BookmarkState.Mode = BookmarkState.Mode.Normal private var mode: BookmarkState.Mode = BookmarkState.Mode.Normal
val selected: Set<BookmarkNode> override val selectedItems: Set<BookmarkNode> get() = mode.selectedItems
get() = (mode as? BookmarkState.Mode.Selecting)?.selectedItems ?: setOf()
private var isFirstRun = true private var isFirstRun = true
fun updateData(tree: BookmarkNode?, mode: BookmarkState.Mode) { fun updateData(tree: BookmarkNode?, mode: BookmarkState.Mode) {
@ -40,7 +41,7 @@ class BookmarkAdapter(val emptyView: View, val interactor: BookmarkViewInteracto
this.tree = tree?.children ?: listOf() this.tree = tree?.children ?: listOf()
isFirstRun = if (isFirstRun) false else { isFirstRun = if (isFirstRun) false else {
emptyView.visibility = if (this.tree.isEmpty()) View.VISIBLE else View.GONE emptyView.isVisible = this.tree.isEmpty()
false false
} }
this.mode = mode this.mode = mode
@ -57,15 +58,8 @@ class BookmarkAdapter(val emptyView: View, val interactor: BookmarkViewInteracto
override fun areItemsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean = override fun areItemsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean =
old[oldItemPosition].guid == new[newItemPosition].guid old[oldItemPosition].guid == new[newItemPosition].guid
override fun areContentsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean { override fun areContentsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean =
val oldSelected = (oldMode as? BookmarkState.Mode.Selecting)?.selectedItems ?: setOf() old[oldItemPosition] in oldMode.selectedItems == new[newItemPosition] in newMode.selectedItems
val newSelected = (newMode as? BookmarkState.Mode.Selecting)?.selectedItems ?: setOf()
val modesEqual = oldMode::class == newMode::class
val selectedEqual =
((oldSelected.contains(old[oldItemPosition]) && newSelected.contains(new[newItemPosition])) ||
(!oldSelected.contains(old[oldItemPosition]) && !newSelected.contains(new[newItemPosition])))
return modesEqual && selectedEqual
}
override fun getOldListSize(): Int = old.size override fun getOldListSize(): Int = old.size
override fun getNewListSize(): Int = new.size override fun getNewListSize(): Int = new.size
@ -77,12 +71,9 @@ class BookmarkAdapter(val emptyView: View, val interactor: BookmarkViewInteracto
} }
return when (viewType) { return when (viewType) {
LibrarySiteItemView.ItemType.SITE.ordinal -> LibrarySiteItemView.ItemType.SITE.ordinal -> BookmarkItemViewHolder(view, interactor, this)
BookmarkItemViewHolder(view, interactor) LibrarySiteItemView.ItemType.FOLDER.ordinal -> BookmarkFolderViewHolder(view, interactor, this)
LibrarySiteItemView.ItemType.FOLDER.ordinal -> LibrarySiteItemView.ItemType.SEPARATOR.ordinal -> BookmarkSeparatorViewHolder(view, interactor)
BookmarkFolderViewHolder(view, interactor)
LibrarySiteItemView.ItemType.SEPARATOR.ordinal ->
BookmarkSeparatorViewHolder(view, interactor)
else -> throw IllegalStateException("ViewType $viewType does not match to a ViewHolder") else -> throw IllegalStateException("ViewType $viewType does not match to a ViewHolder")
} }
} }
@ -99,11 +90,7 @@ class BookmarkAdapter(val emptyView: View, val interactor: BookmarkViewInteracto
override fun getItemCount(): Int = tree.size override fun getItemCount(): Int = tree.size
override fun onBindViewHolder(holder: BookmarkNodeViewHolder, position: Int) { override fun onBindViewHolder(holder: BookmarkNodeViewHolder, position: Int) {
holder.bind( holder.bind(tree[position])
tree[position],
mode,
tree[position] in selected
)
} }
} }

View File

@ -153,7 +153,7 @@ class BookmarkFragment : LibraryPageFragment<BookmarkNode>(), BackHandler, Accou
} }
override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) { override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
when (val mode = bookmarkView.mode) { when (val mode = bookmarkStore.state.mode) {
BookmarkState.Mode.Normal -> { BookmarkState.Mode.Normal -> {
inflater.inflate(R.menu.bookmarks_menu, menu) inflater.inflate(R.menu.bookmarks_menu, menu)
} }

View File

@ -51,24 +51,26 @@ class BookmarkFragmentInteractor(
} }
override fun open(item: BookmarkNode) { override fun open(item: BookmarkNode) {
require(item.type == BookmarkNodeType.ITEM) when (item.type) {
item.url?.let { url -> BookmarkNodeType.ITEM -> {
activity!! item.url?.let { url ->
.openToBrowserAndLoad( activity!!
searchTermOrURL = url, .openToBrowserAndLoad(
newTab = true, searchTermOrURL = url,
from = BrowserDirection.FromBookmarks newTab = true,
from = BrowserDirection.FromBookmarks
)
}
metrics.track(Event.OpenedBookmark)
}
BookmarkNodeType.FOLDER -> {
navController.nav(
R.id.bookmarkFragment,
BookmarkFragmentDirections.actionBookmarkFragmentSelf(item.guid)
) )
}
BookmarkNodeType.SEPARATOR -> throw IllegalStateException("Cannot open separators")
} }
metrics.track(Event.OpenedBookmark)
}
override fun expand(folder: BookmarkNode) {
require(folder.type == BookmarkNodeType.FOLDER)
navController.nav(
R.id.bookmarkFragment,
BookmarkFragmentDirections.actionBookmarkFragmentSelf(folder.guid)
)
} }
override fun switchMode(mode: BookmarkState.Mode) { override fun switchMode(mode: BookmarkState.Mode) {
@ -83,16 +85,16 @@ class BookmarkFragmentInteractor(
) )
} }
override fun select(node: BookmarkNode) { override fun select(item: BookmarkNode) {
if (node.inRoots()) { if (item.inRoots()) {
snackbarPresenter.present(context.getString(R.string.bookmark_cannot_edit_root)) snackbarPresenter.present(context.getString(R.string.bookmark_cannot_edit_root))
return return
} }
bookmarkStore.dispatch(BookmarkAction.Select(node)) bookmarkStore.dispatch(BookmarkAction.Select(item))
} }
override fun deselect(node: BookmarkNode) { override fun deselect(item: BookmarkNode) {
bookmarkStore.dispatch(BookmarkAction.Deselect(node)) bookmarkStore.dispatch(BookmarkAction.Deselect(item))
} }
override fun deselectAll() { override fun deselectAll() {
@ -148,23 +150,14 @@ class BookmarkFragmentInteractor(
} }
} }
override fun delete(node: BookmarkNode) { override fun delete(nodes: Set<BookmarkNode>) {
val eventType = when (node.type) { val eventType = when (nodes.singleOrNull()?.type) {
BookmarkNodeType.ITEM -> { BookmarkNodeType.ITEM -> Event.RemoveBookmark
Event.RemoveBookmark BookmarkNodeType.FOLDER -> Event.RemoveBookmarkFolder
} BookmarkNodeType.SEPARATOR -> throw IllegalStateException("Cannot delete separators")
BookmarkNodeType.FOLDER -> { null -> Event.RemoveBookmarks
Event.RemoveBookmarkFolder
}
BookmarkNodeType.SEPARATOR -> {
throw IllegalStateException("Cannot delete separators")
}
} }
deleteBookmarkNodes(setOf(node), eventType) deleteBookmarkNodes(nodes, eventType)
}
override fun deleteMulti(nodes: Set<BookmarkNode>) {
deleteBookmarkNodes(nodes, Event.RemoveBookmarks)
} }
override fun backPressed() { override fun backPressed() {

View File

@ -11,12 +11,13 @@ import mozilla.components.concept.storage.BookmarkNode
import mozilla.components.concept.storage.BookmarkNodeType import mozilla.components.concept.storage.BookmarkNodeType
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.ThemeManager import org.mozilla.fenix.ThemeManager
import org.mozilla.fenix.library.LibraryItemMenu
class BookmarkItemMenu( class BookmarkItemMenu(
private val context: Context, private val context: Context,
private val item: BookmarkNode, private val item: BookmarkNode,
private val onItemTapped: (BookmarkItemMenu.Item) -> Unit = {} private val onItemTapped: (BookmarkItemMenu.Item) -> Unit = {}
) { ) : LibraryItemMenu {
sealed class Item { sealed class Item {
object Edit : Item() object Edit : Item()
@ -28,7 +29,7 @@ class BookmarkItemMenu(
object Delete : Item() object Delete : Item()
} }
val menuBuilder by lazy { BrowserMenuBuilder(menuItems) } override val menuBuilder by lazy { BrowserMenuBuilder(menuItems) }
private val menuItems by lazy { private val menuItems by lazy {
listOfNotNull( listOfNotNull(

View File

@ -13,13 +13,14 @@ import mozilla.components.concept.storage.BookmarkNode
import mozilla.components.support.base.feature.BackHandler import mozilla.components.support.base.feature.BackHandler
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.library.LibraryPageView import org.mozilla.fenix.library.LibraryPageView
import org.mozilla.fenix.library.SelectionInteractor
/** /**
* Interface for the Bookmarks view. * Interface for the Bookmarks view.
* This interface is implemented by objects that want to respond to user interaction on the bookmarks management UI. * This interface is implemented by objects that want to respond to user interaction on the bookmarks management UI.
*/ */
@SuppressWarnings("TooManyFunctions") @SuppressWarnings("TooManyFunctions")
interface BookmarkViewInteractor { interface BookmarkViewInteractor : SelectionInteractor<BookmarkNode> {
/** /**
* Swaps the head of the bookmarks tree, replacing it with a new, updated bookmarks tree. * Swaps the head of the bookmarks tree, replacing it with a new, updated bookmarks tree.
@ -28,20 +29,6 @@ interface BookmarkViewInteractor {
*/ */
fun change(node: BookmarkNode) fun change(node: BookmarkNode)
/**
* Opens a tab for a bookmark item.
*
* @param item the bookmark item to open
*/
fun open(item: BookmarkNode)
/**
* Expands a bookmark folder in the bookmarks tree, providing a view of a different folder elsewhere in the tree.
*
* @param folder the bookmark folder to expand
*/
fun expand(folder: BookmarkNode)
/** /**
* Switches the current bookmark multi-selection mode. * Switches the current bookmark multi-selection mode.
* *
@ -56,20 +43,6 @@ interface BookmarkViewInteractor {
*/ */
fun edit(node: BookmarkNode) fun edit(node: BookmarkNode)
/**
* Selects a bookmark node in multi-selection.
*
* @param node the bookmark node to select
*/
fun select(node: BookmarkNode)
/**
* De-selects a bookmark node in multi-selection.
*
* @param node the bookmark node to deselect
*/
fun deselect(node: BookmarkNode)
/** /**
* De-selects all bookmark nodes, clearing the multi-selection mode. * De-selects all bookmark nodes, clearing the multi-selection mode.
* *
@ -105,18 +78,11 @@ interface BookmarkViewInteractor {
fun openInPrivateTab(item: BookmarkNode) fun openInPrivateTab(item: BookmarkNode)
/** /**
* Deletes a bookmark node. * Deletes a set of bookmark node.
* *
* @param node the bookmark node to delete * @param nodes the bookmark nodes to delete
*/ */
fun delete(node: BookmarkNode) fun delete(nodes: Set<BookmarkNode>)
/**
* Deletes a set of bookmark nodes.
*
* @param nodes the set of bookmark nodes to delete
*/
fun deleteMulti(nodes: Set<BookmarkNode>)
/** /**
* Handles back presses for the bookmark screen, so navigation up the tree is possible. * Handles back presses for the bookmark screen, so navigation up the tree is possible.
@ -133,10 +99,8 @@ class BookmarkView(
val view: View = LayoutInflater.from(container.context) val view: View = LayoutInflater.from(container.context)
.inflate(R.layout.component_bookmark, container, true) .inflate(R.layout.component_bookmark, container, true)
var mode: BookmarkState.Mode = BookmarkState.Mode.Normal private var mode: BookmarkState.Mode = BookmarkState.Mode.Normal
private set private var tree: BookmarkNode? = null
var tree: BookmarkNode? = null
private set
private var canGoBack = false private var canGoBack = false
private val bookmarkAdapter: BookmarkAdapter private val bookmarkAdapter: BookmarkAdapter
@ -157,7 +121,7 @@ class BookmarkView(
} }
bookmarkAdapter.updateData(state.tree, mode) bookmarkAdapter.updateData(state.tree, mode)
when (state.mode) { when (mode) {
is BookmarkState.Mode.Normal -> is BookmarkState.Mode.Normal ->
setUiForNormalMode(state.tree) setUiForNormalMode(state.tree)
is BookmarkState.Mode.Selecting -> is BookmarkState.Mode.Selecting ->

View File

@ -10,7 +10,7 @@ import mozilla.components.concept.storage.BookmarkNode
import org.jetbrains.anko.image import org.jetbrains.anko.image
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.library.LibrarySiteItemView import org.mozilla.fenix.library.LibrarySiteItemView
import org.mozilla.fenix.library.bookmarks.BookmarkState import org.mozilla.fenix.library.SelectionHolder
import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor
import org.mozilla.fenix.library.bookmarks.inRoots import org.mozilla.fenix.library.bookmarks.inRoots
@ -19,14 +19,15 @@ import org.mozilla.fenix.library.bookmarks.inRoots
*/ */
class BookmarkFolderViewHolder( class BookmarkFolderViewHolder(
view: LibrarySiteItemView, view: LibrarySiteItemView,
interactor: BookmarkViewInteractor interactor: BookmarkViewInteractor,
private val selectionHolder: SelectionHolder<BookmarkNode>
) : BookmarkNodeViewHolder(view, interactor) { ) : BookmarkNodeViewHolder(view, interactor) {
override fun bind(item: BookmarkNode, mode: BookmarkState.Mode, selected: Boolean) { override fun bind(item: BookmarkNode) {
containerView.displayAs(LibrarySiteItemView.ItemType.FOLDER) containerView.displayAs(LibrarySiteItemView.ItemType.FOLDER)
setClickListeners(mode, item, selected) setSelectionListeners(item, selectionHolder)
if (!item.inRoots()) { if (!item.inRoots()) {
setupMenu(item) setupMenu(item)
@ -34,31 +35,10 @@ class BookmarkFolderViewHolder(
containerView.overflowView.visibility = View.GONE containerView.overflowView.visibility = View.GONE
} }
containerView.changeSelected(selected) containerView.changeSelected(item in selectionHolder.selectedItems)
containerView.iconView.image = containerView.context.getDrawable(R.drawable.ic_folder_icon)?.apply { containerView.iconView.image = containerView.context.getDrawable(R.drawable.ic_folder_icon)?.apply {
setTint(ContextCompat.getColor(containerView.context, R.color.primary_text_light_theme)) setTint(ContextCompat.getColor(containerView.context, R.color.primary_text_light_theme))
} }
containerView.titleView.text = item.title containerView.titleView.text = item.title
} }
private fun setClickListeners(
mode: BookmarkState.Mode,
item: BookmarkNode,
selected: Boolean
) {
containerView.setOnClickListener {
when {
mode == BookmarkState.Mode.Normal -> interactor.expand(item)
selected -> interactor.deselect(item)
else -> interactor.select(item)
}
}
containerView.setOnLongClickListener {
if (mode == BookmarkState.Mode.Normal && !item.inRoots()) {
interactor.select(item)
true
} else false
}
}
} }

View File

@ -6,7 +6,7 @@ package org.mozilla.fenix.library.bookmarks.viewholders
import mozilla.components.concept.storage.BookmarkNode import mozilla.components.concept.storage.BookmarkNode
import org.mozilla.fenix.library.LibrarySiteItemView import org.mozilla.fenix.library.LibrarySiteItemView
import org.mozilla.fenix.library.bookmarks.BookmarkState import org.mozilla.fenix.library.SelectionHolder
import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor
/** /**
@ -14,10 +14,11 @@ import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor
*/ */
class BookmarkItemViewHolder( class BookmarkItemViewHolder(
view: LibrarySiteItemView, view: LibrarySiteItemView,
interactor: BookmarkViewInteractor interactor: BookmarkViewInteractor,
private val selectionHolder: SelectionHolder<BookmarkNode>
) : BookmarkNodeViewHolder(view, interactor) { ) : BookmarkNodeViewHolder(view, interactor) {
override fun bind(item: BookmarkNode, mode: BookmarkState.Mode, selected: Boolean) { override fun bind(item: BookmarkNode) {
containerView.displayAs(LibrarySiteItemView.ItemType.SITE) containerView.displayAs(LibrarySiteItemView.ItemType.SITE)
@ -25,8 +26,9 @@ class BookmarkItemViewHolder(
containerView.titleView.text = if (item.title.isNullOrBlank()) item.url else item.title containerView.titleView.text = if (item.title.isNullOrBlank()) item.url else item.title
containerView.urlView.text = item.url containerView.urlView.text = item.url
setClickListeners(mode, item, selected) setSelectionListeners(item, selectionHolder)
containerView.changeSelected(selected)
containerView.changeSelected(item in selectionHolder.selectedItems)
setColorsAndIcons(item.url) setColorsAndIcons(item.url)
} }
@ -37,32 +39,4 @@ class BookmarkItemViewHolder(
containerView.iconView.setImageDrawable(null) containerView.iconView.setImageDrawable(null)
} }
} }
private fun setClickListeners(
mode: BookmarkState.Mode,
item: BookmarkNode,
selected: Boolean
) {
containerView.setOnClickListener {
when {
mode == BookmarkState.Mode.Normal -> interactor.open(item)
selected -> interactor.deselect(item)
else -> interactor.select(item)
}
}
containerView.setOnLongClickListener {
if (mode == BookmarkState.Mode.Normal) {
interactor.select(item)
true
} else false
}
containerView.iconView.setOnClickListener({
when {
selected -> interactor.deselect(item)
else -> interactor.select(item)
}
})
}
} }

View File

@ -6,11 +6,10 @@ package org.mozilla.fenix.library.bookmarks.viewholders
import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.RecyclerView
import kotlinx.android.extensions.LayoutContainer import kotlinx.android.extensions.LayoutContainer
import mozilla.components.browser.menu.BrowserMenu
import mozilla.components.concept.storage.BookmarkNode import mozilla.components.concept.storage.BookmarkNode
import org.mozilla.fenix.library.LibrarySiteItemView import org.mozilla.fenix.library.LibrarySiteItemView
import org.mozilla.fenix.library.SelectionHolder
import org.mozilla.fenix.library.bookmarks.BookmarkItemMenu import org.mozilla.fenix.library.bookmarks.BookmarkItemMenu
import org.mozilla.fenix.library.bookmarks.BookmarkState
import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor
/** /**
@ -18,11 +17,15 @@ import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor
*/ */
abstract class BookmarkNodeViewHolder( abstract class BookmarkNodeViewHolder(
override val containerView: LibrarySiteItemView, override val containerView: LibrarySiteItemView,
val interactor: BookmarkViewInteractor private val interactor: BookmarkViewInteractor
) : ) :
RecyclerView.ViewHolder(containerView), LayoutContainer { RecyclerView.ViewHolder(containerView), LayoutContainer {
abstract fun bind(item: BookmarkNode, mode: BookmarkState.Mode, selected: Boolean) abstract fun bind(item: BookmarkNode)
protected fun setSelectionListeners(item: BookmarkNode, selectionHolder: SelectionHolder<BookmarkNode>) {
containerView.setSelectionInteractor(item, selectionHolder, interactor)
}
protected fun setupMenu(item: BookmarkNode) { protected fun setupMenu(item: BookmarkNode) {
val bookmarkItemMenu = BookmarkItemMenu(containerView.context, item) { val bookmarkItemMenu = BookmarkItemMenu(containerView.context, item) {
@ -33,15 +36,10 @@ abstract class BookmarkNodeViewHolder(
BookmarkItemMenu.Item.Share -> interactor.share(item) BookmarkItemMenu.Item.Share -> interactor.share(item)
BookmarkItemMenu.Item.OpenInNewTab -> interactor.openInNewTab(item) BookmarkItemMenu.Item.OpenInNewTab -> interactor.openInNewTab(item)
BookmarkItemMenu.Item.OpenInPrivateTab -> interactor.openInPrivateTab(item) BookmarkItemMenu.Item.OpenInPrivateTab -> interactor.openInPrivateTab(item)
BookmarkItemMenu.Item.Delete -> interactor.delete(item) BookmarkItemMenu.Item.Delete -> interactor.delete(setOf(item))
} }
} }
containerView.overflowView.setOnClickListener { containerView.attachMenu(bookmarkItemMenu)
bookmarkItemMenu.menuBuilder.build(containerView.context).show(
anchor = it,
orientation = BrowserMenu.Orientation.DOWN
)
}
} }
} }

View File

@ -6,7 +6,6 @@ package org.mozilla.fenix.library.bookmarks.viewholders
import mozilla.components.concept.storage.BookmarkNode import mozilla.components.concept.storage.BookmarkNode
import org.mozilla.fenix.library.LibrarySiteItemView import org.mozilla.fenix.library.LibrarySiteItemView
import org.mozilla.fenix.library.bookmarks.BookmarkState
import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor import org.mozilla.fenix.library.bookmarks.BookmarkViewInteractor
/** /**
@ -17,7 +16,7 @@ class BookmarkSeparatorViewHolder(
interactor: BookmarkViewInteractor interactor: BookmarkViewInteractor
) : BookmarkNodeViewHolder(view, interactor) { ) : BookmarkNodeViewHolder(view, interactor) {
override fun bind(item: BookmarkNode, mode: BookmarkState.Mode, selected: Boolean) { override fun bind(item: BookmarkNode) {
containerView.displayAs(LibrarySiteItemView.ItemType.SEPARATOR) containerView.displayAs(LibrarySiteItemView.ItemType.SEPARATOR)
setupMenu(item) setupMenu(item)
} }

View File

@ -11,6 +11,7 @@ import android.view.ViewGroup
import androidx.paging.PagedListAdapter import androidx.paging.PagedListAdapter
import androidx.recyclerview.widget.DiffUtil import androidx.recyclerview.widget.DiffUtil
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.library.SelectionHolder
import org.mozilla.fenix.library.history.viewholders.HistoryListItemViewHolder import org.mozilla.fenix.library.history.viewholders.HistoryListItemViewHolder
import java.util.Calendar import java.util.Calendar
import java.util.Date import java.util.Date
@ -28,14 +29,16 @@ enum class HistoryItemTimeGroup {
class HistoryAdapter( class HistoryAdapter(
private val historyInteractor: HistoryInteractor private val historyInteractor: HistoryInteractor
) : PagedListAdapter<HistoryItem, HistoryListItemViewHolder>(historyDiffCallback) { ) : PagedListAdapter<HistoryItem, HistoryListItemViewHolder>(historyDiffCallback), SelectionHolder<HistoryItem> {
private var mode: HistoryState.Mode = HistoryState.Mode.Normal private var mode: HistoryState.Mode = HistoryState.Mode.Normal
override val selectedItems get() = mode.selectedItems
override fun getItemViewType(position: Int): Int = HistoryListItemViewHolder.LAYOUT_ID override fun getItemViewType(position: Int): Int = HistoryListItemViewHolder.LAYOUT_ID
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): HistoryListItemViewHolder { override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): HistoryListItemViewHolder {
val view = LayoutInflater.from(parent.context).inflate(viewType, parent, false) val view = LayoutInflater.from(parent.context).inflate(viewType, parent, false)
return HistoryListItemViewHolder(view, historyInteractor) return HistoryListItemViewHolder(view, historyInteractor, this)
} }
fun updateMode(mode: HistoryState.Mode) { fun updateMode(mode: HistoryState.Mode) {

View File

@ -94,10 +94,11 @@ class HistoryFragment : LibraryPageFragment<HistoryItem>(), BackHandler {
private fun deleteHistoryItems(items: Set<HistoryItem>) { private fun deleteHistoryItems(items: Set<HistoryItem>) {
lifecycleScope.launch { lifecycleScope.launch {
val storage = context?.components?.core?.historyStorage context?.components?.run {
for (item in items) { for (item in items) {
context?.components?.analytics?.metrics?.track(Event.HistoryItemRemoved) analytics.metrics.track(Event.HistoryItemRemoved)
storage?.deleteVisit(item.url, item.visitedAt) core.historyStorage.deleteVisit(item.url, item.visitedAt)
}
} }
viewModel.invalidate() viewModel.invalidate()
historyStore.dispatch(HistoryAction.ExitDeletionMode) historyStore.dispatch(HistoryAction.ExitDeletionMode)
@ -134,7 +135,7 @@ class HistoryFragment : LibraryPageFragment<HistoryItem>(), BackHandler {
if (mode is HistoryState.Mode.Editing) { if (mode is HistoryState.Mode.Editing) {
menu.findItem(R.id.share_history_multi_select)?.run { menu.findItem(R.id.share_history_multi_select)?.run {
isVisible = mode.selectedItems.isNotEmpty() isVisible = true
icon.colorFilter = PorterDuffColorFilter( icon.colorFilter = PorterDuffColorFilter(
ContextCompat.getColor(context!!, R.color.white_color), ContextCompat.getColor(context!!, R.color.white_color),
SRC_IN SRC_IN
@ -212,7 +213,7 @@ class HistoryFragment : LibraryPageFragment<HistoryItem>(), BackHandler {
) )
} }
fun displayDeleteAllDialog() { private fun displayDeleteAllDialog() {
activity?.let { activity -> activity?.let { activity ->
AlertDialog.Builder(activity).apply { AlertDialog.Builder(activity).apply {
setMessage(R.string.history_delete_all_dialog) setMessage(R.string.history_delete_all_dialog)

View File

@ -15,29 +15,16 @@ class HistoryInteractor(
private val invalidateOptionsMenu: () -> Unit, private val invalidateOptionsMenu: () -> Unit,
private val deleteHistoryItems: (Set<HistoryItem>) -> Unit private val deleteHistoryItems: (Set<HistoryItem>) -> Unit
) : HistoryViewInteractor { ) : HistoryViewInteractor {
override fun onItemPress(item: HistoryItem) { override fun open(item: HistoryItem) {
when (val mode = store.state.mode) { openToBrowser(item)
is HistoryState.Mode.Normal -> openToBrowser(item)
is HistoryState.Mode.Editing -> {
val isSelected = mode.selectedItems.contains(item)
if (isSelected) {
store.dispatch(HistoryAction.RemoveItemForRemoval(item))
} else {
store.dispatch(HistoryAction.AddItemForRemoval(item))
}
}
}
} }
override fun onItemLongPress(item: HistoryItem) { override fun select(item: HistoryItem) {
val isSelected = store.state.mode.selectedItems.contains(item) store.dispatch(HistoryAction.AddItemForRemoval(item))
}
if (isSelected) { override fun deselect(item: HistoryItem) {
store.dispatch(HistoryAction.RemoveItemForRemoval(item)) store.dispatch(HistoryAction.RemoveItemForRemoval(item))
} else {
store.dispatch(HistoryAction.AddItemForRemoval(item))
}
} }
override fun onBackPressed(): Boolean { override fun onBackPressed(): Boolean {
@ -57,10 +44,6 @@ class HistoryInteractor(
displayDeleteAll.invoke() displayDeleteAll.invoke()
} }
override fun onDeleteOne(item: HistoryItem) {
deleteHistoryItems.invoke(setOf(item))
}
override fun onDeleteSome(items: Set<HistoryItem>) { override fun onDeleteSome(items: Set<HistoryItem>) {
deleteHistoryItems.invoke(items) deleteHistoryItems.invoke(items)
} }

View File

@ -9,16 +9,17 @@ import mozilla.components.browser.menu.BrowserMenuBuilder
import mozilla.components.browser.menu.item.SimpleBrowserMenuItem import mozilla.components.browser.menu.item.SimpleBrowserMenuItem
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.ThemeManager import org.mozilla.fenix.ThemeManager
import org.mozilla.fenix.library.LibraryItemMenu
class HistoryItemMenu( class HistoryItemMenu(
private val context: Context, private val context: Context,
private val onItemTapped: (Item) -> Unit = {} private val onItemTapped: (Item) -> Unit = {}
) { ) : LibraryItemMenu {
sealed class Item { sealed class Item {
object Delete : Item() object Delete : Item()
} }
val menuBuilder by lazy { BrowserMenuBuilder(menuItems) } override val menuBuilder by lazy { BrowserMenuBuilder(menuItems) }
private val menuItems by lazy { private val menuItems by lazy {
listOf( listOf(

View File

@ -15,21 +15,13 @@ import kotlinx.android.synthetic.main.component_history.view.*
import mozilla.components.support.base.feature.BackHandler import mozilla.components.support.base.feature.BackHandler
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.library.LibraryPageView import org.mozilla.fenix.library.LibraryPageView
import org.mozilla.fenix.library.SelectionInteractor
/** /**
* Interface for the HistoryViewInteractor. This interface is implemented by objects that want * Interface for the HistoryViewInteractor. This interface is implemented by objects that want
* to respond to user interaction on the HistoryView * to respond to user interaction on the HistoryView
*/ */
interface HistoryViewInteractor { interface HistoryViewInteractor : SelectionInteractor<HistoryItem> {
/**
* Called when a user taps a history item
*/
fun onItemPress(item: HistoryItem)
/**
* Called when a user long clicks a user
*/
fun onItemLongPress(item: HistoryItem)
/** /**
* Called on backpressed to exit edit mode * Called on backpressed to exit edit mode
@ -46,12 +38,6 @@ interface HistoryViewInteractor {
*/ */
fun onDeleteAll() fun onDeleteAll()
/**
* Called when one history item is deleted
* @param item the history item to delete
*/
fun onDeleteOne(item: HistoryItem)
/** /**
* Called when multiple history items are deleted * Called when multiple history items are deleted
* @param items the history items to delete * @param items the history items to delete

View File

@ -7,8 +7,8 @@ package org.mozilla.fenix.library.history.viewholders
import android.view.View import android.view.View
import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.RecyclerView
import kotlinx.android.synthetic.main.history_list_item.view.* import kotlinx.android.synthetic.main.history_list_item.view.*
import mozilla.components.browser.menu.BrowserMenu
import org.mozilla.fenix.R import org.mozilla.fenix.R
import org.mozilla.fenix.library.SelectionHolder
import org.mozilla.fenix.library.history.HistoryInteractor import org.mozilla.fenix.library.history.HistoryInteractor
import org.mozilla.fenix.library.history.HistoryItem import org.mozilla.fenix.library.history.HistoryItem
import org.mozilla.fenix.library.history.HistoryItemMenu import org.mozilla.fenix.library.history.HistoryItemMenu
@ -17,34 +17,21 @@ import org.mozilla.fenix.library.history.HistoryState
class HistoryListItemViewHolder( class HistoryListItemViewHolder(
view: View, view: View,
private val historyInteractor: HistoryInteractor private val historyInteractor: HistoryInteractor,
private val selectionHolder: SelectionHolder<HistoryItem>
) : RecyclerView.ViewHolder(view) { ) : RecyclerView.ViewHolder(view) {
private var item: HistoryItem? = null private var item: HistoryItem? = null
private var mode: HistoryState.Mode = HistoryState.Mode.Normal
init { init {
setupMenu() setupMenu()
itemView.history_layout.setOnLongClickListener {
item?.also(historyInteractor::onItemLongPress)
true
}
itemView.history_layout.setOnClickListener {
item?.also(historyInteractor::onItemPress)
}
itemView.history_layout.iconView.setOnClickListener {
item?.apply {
historyInteractor.onItemLongPress(this)
}
}
itemView.delete_button.setOnClickListener { itemView.delete_button.setOnClickListener {
when (val mode = this.mode) { val selected = selectionHolder.selectedItems
HistoryState.Mode.Normal -> historyInteractor.onDeleteAll() if (selected.isEmpty()) {
is HistoryState.Mode.Editing -> historyInteractor.onDeleteSome(mode.selectedItems) historyInteractor.onDeleteAll()
} else {
historyInteractor.onDeleteSome(selected)
} }
} }
} }
@ -56,24 +43,24 @@ class HistoryListItemViewHolder(
mode: HistoryState.Mode mode: HistoryState.Mode
) { ) {
this.item = item this.item = item
this.mode = mode
itemView.history_layout.titleView.text = item.title itemView.history_layout.titleView.text = item.title
itemView.history_layout.urlView.text = item.url itemView.history_layout.urlView.text = item.url
toggleDeleteButton(showDeleteButton, mode) toggleDeleteButton(showDeleteButton, mode === HistoryState.Mode.Normal)
val headerText = timeGroup?.humanReadable(itemView.context) val headerText = timeGroup?.humanReadable(itemView.context)
toggleHeader(headerText) toggleHeader(headerText)
itemView.history_layout.changeSelected(item in mode.selectedItems) itemView.history_layout.setSelectionInteractor(item, selectionHolder, historyInteractor)
itemView.history_layout.changeSelected(item in selectionHolder.selectedItems)
itemView.history_layout.loadFavicon(item.url) itemView.history_layout.loadFavicon(item.url)
} }
private fun toggleHeader(text: String?) { private fun toggleHeader(headerText: String?) {
if (text != null) { if (headerText != null) {
itemView.header_title.visibility = View.VISIBLE itemView.header_title.visibility = View.VISIBLE
itemView.header_title.text = text itemView.header_title.text = headerText
} else { } else {
itemView.header_title.visibility = View.GONE itemView.header_title.visibility = View.GONE
} }
@ -81,18 +68,18 @@ class HistoryListItemViewHolder(
private fun toggleDeleteButton( private fun toggleDeleteButton(
showDeleteButton: Boolean, showDeleteButton: Boolean,
mode: HistoryState.Mode isNormalMode: Boolean
) { ) {
if (showDeleteButton) { if (showDeleteButton) {
itemView.delete_button.run { itemView.delete_button.run {
visibility = View.VISIBLE visibility = View.VISIBLE
if (mode === HistoryState.Mode.Deleting || mode.selectedItems.isNotEmpty()) { if (isNormalMode) {
isEnabled = false
alpha = DELETE_BUTTON_DISABLED_ALPHA
} else {
isEnabled = true isEnabled = true
alpha = 1f alpha = 1f
} else {
isEnabled = false
alpha = DELETE_BUTTON_DISABLED_ALPHA
} }
} }
} else { } else {
@ -102,17 +89,13 @@ class HistoryListItemViewHolder(
private fun setupMenu() { private fun setupMenu() {
val historyMenu = HistoryItemMenu(itemView.context) { val historyMenu = HistoryItemMenu(itemView.context) {
val item = this.item ?: return@HistoryItemMenu
when (it) { when (it) {
HistoryItemMenu.Item.Delete -> item?.also(historyInteractor::onDeleteOne) HistoryItemMenu.Item.Delete -> historyInteractor.onDeleteSome(setOf(item))
} }
} }
itemView.history_layout.overflowView.setOnClickListener { itemView.history_layout.attachMenu(historyMenu)
historyMenu.menuBuilder.build(itemView.context).show(
anchor = it,
orientation = BrowserMenu.Orientation.DOWN
)
}
} }
companion object { companion object {

View File

@ -117,7 +117,7 @@ class BookmarkFragmentInteractorTest {
@Test @Test
fun `expand a level of bookmarks`() { fun `expand a level of bookmarks`() {
interactor.expand(tree) interactor.open(tree)
verify { verify {
navController.navigate(BookmarkFragmentDirections.actionBookmarkFragmentSelf(tree.guid)) navController.navigate(BookmarkFragmentDirections.actionBookmarkFragmentSelf(tree.guid))
@ -236,7 +236,7 @@ class BookmarkFragmentInteractorTest {
@Test @Test
fun `delete a bookmark item`() { fun `delete a bookmark item`() {
interactor.delete(item) interactor.delete(setOf(item))
verify { verify {
deleteBookmarkNodes(setOf(item), Event.RemoveBookmark) deleteBookmarkNodes(setOf(item), Event.RemoveBookmark)
@ -245,7 +245,7 @@ class BookmarkFragmentInteractorTest {
@Test @Test
fun `delete a bookmark folder`() { fun `delete a bookmark folder`() {
interactor.delete(subfolder) interactor.delete(setOf(subfolder))
verify { verify {
deleteBookmarkNodes(setOf(subfolder), Event.RemoveBookmarkFolder) deleteBookmarkNodes(setOf(subfolder), Event.RemoveBookmarkFolder)
@ -254,7 +254,7 @@ class BookmarkFragmentInteractorTest {
@Test @Test
fun `delete multiple bookmarks`() { fun `delete multiple bookmarks`() {
interactor.deleteMulti(setOf(item, subfolder)) interactor.delete(setOf(item, subfolder))
verify { verify {
deleteBookmarkNodes(setOf(item, subfolder), Event.RemoveBookmarks) deleteBookmarkNodes(setOf(item, subfolder), Event.RemoveBookmarks)

View File

@ -31,7 +31,7 @@ class HistoryInteractorTest {
mockk() mockk()
) )
interactor.onItemPress(historyItem) interactor.open(historyItem)
assertEquals(historyItem, historyItemReceived) assertEquals(historyItem, historyItemReceived)
} }
@ -51,7 +51,7 @@ class HistoryInteractorTest {
mockk() mockk()
) )
interactor.onItemPress(historyItem) interactor.select(historyItem)
verify { verify {
store.dispatch(HistoryAction.AddItemForRemoval(historyItem)) store.dispatch(HistoryAction.AddItemForRemoval(historyItem))
@ -74,7 +74,7 @@ class HistoryInteractorTest {
mockk() mockk()
) )
interactor.onItemPress(historyItem) interactor.deselect(historyItem)
verify { verify {
store.dispatch(HistoryAction.RemoveItemForRemoval(historyItem)) store.dispatch(HistoryAction.RemoveItemForRemoval(historyItem))
@ -135,22 +135,6 @@ class HistoryInteractorTest {
assertEquals(true, deleteAllDialogShown) assertEquals(true, deleteAllDialogShown)
} }
@Test
fun onDeleteOne() {
var itemsToDelete: Set<HistoryItem>? = null
val historyItem = HistoryItem(0, "title", "url", 0.toLong())
val interactor =
HistoryInteractor(
mockk(),
mockk(),
mockk(),
mockk(),
{ itemsToDelete = it }
)
interactor.onDeleteOne(historyItem)
assertEquals(itemsToDelete, setOf(historyItem))
}
@Test @Test
fun onDeleteSome() { fun onDeleteSome() {
var itemsToDelete: Set<HistoryItem>? = null var itemsToDelete: Set<HistoryItem>? = null