Fixes #2379 - Generic library selection
parent
4bbb291e8d
commit
3c1ce90f6f
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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 ->
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue