1
0
Fork 0

Group history by date

master
Jeff Boek 2019-03-25 11:49:37 -07:00 committed by Colin Lee
parent b3ac37078e
commit bd9b7d8d8d
4 changed files with 93 additions and 43 deletions

View File

@ -4,6 +4,7 @@
package org.mozilla.fenix.library.history
import android.content.Context
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
@ -16,10 +17,85 @@ import kotlinx.android.synthetic.main.history_delete.view.*
import kotlinx.android.synthetic.main.history_header.view.*
import kotlinx.android.synthetic.main.history_list_item.view.*
import mozilla.components.browser.menu.BrowserMenu
import org.mozilla.fenix.components.SectionedAdapter
import java.util.*
class HistoryList(val history: List<HistoryItem>) {
enum class Range {
Today, ThisWeek, ThisMonth, Older;
fun humanReadable(context: Context) : String = when (this) {
Today -> context.getString(R.string.history_today)
ThisWeek -> context.getString(R.string.history_this_week)
ThisMonth -> context.getString(R.string.history_this_month)
Older -> context.getString(R.string.history_older)
}
}
val ranges: List<Range>
get() = grouped.keys.toList()
fun itemsInRange(range: Range): List<HistoryItem> {
return grouped[range] ?: listOf()
}
private val grouped: Map<Range, List<HistoryItem>>
init {
val oneDayAgo = getDaysAgo(1).time
val sevenDaysAgo = getDaysAgo(7).time
val thirtyDaysAgo = getDaysAgo(30).time
val lastWeek = LongRange(sevenDaysAgo, oneDayAgo)
val lastMonth = LongRange(thirtyDaysAgo, sevenDaysAgo)
grouped = history.groupBy { item ->
when {
item.visitedAt > oneDayAgo -> Range.Today
lastWeek.contains(item.visitedAt) -> Range.ThisWeek
lastMonth.contains(item.visitedAt) -> Range.ThisMonth
else -> Range.Older
}
}
}
private fun getDaysAgo(daysAgo: Int): Date {
val calendar = Calendar.getInstance()
calendar.add(Calendar.DAY_OF_YEAR, -daysAgo)
return calendar.time
}
}
class HistoryAdapter(
private val actionEmitter: Observer<HistoryAction>
) : RecyclerView.Adapter<RecyclerView.ViewHolder>() {
) : SectionedAdapter() {
override fun numberOfSections(): Int = historyList.ranges.size
override fun numberOfRowsInSection(section: Int): Int = historyList.itemsInRange(historyList.ranges[section]).size
override fun onCreateHeaderViewHolder(parent: ViewGroup): RecyclerView.ViewHolder {
val view = LayoutInflater.from(parent.context).inflate(HistoryHeaderViewHolder.LAYOUT_ID, parent, false)
return HistoryHeaderViewHolder(view)
}
override fun onBindHeaderViewHolder(holder: RecyclerView.ViewHolder, header: SectionType.Header) {
val sectionTitle = historyList.ranges[header.index].humanReadable(holder.itemView.context)
when (holder) {
is HistoryHeaderViewHolder -> holder.bind(sectionTitle)
}
}
override fun onCreateItemViewHolder(parent: ViewGroup): RecyclerView.ViewHolder {
val view = LayoutInflater.from(parent.context).inflate(HistoryListItemViewHolder.LAYOUT_ID, parent, false)
return HistoryListItemViewHolder(view, actionEmitter)
}
override fun onBindItemViewHolder(holder: RecyclerView.ViewHolder, row: SectionType.Row) {
(holder as? HistoryListItemViewHolder)?.bind(historyList.itemsInRange(historyList.ranges[row.section])[row.row], mode)
}
class HistoryListItemViewHolder(
view: View,
private val actionEmitter: Observer<HistoryAction>
@ -176,49 +252,12 @@ class HistoryAdapter(
}
}
private var items: List<HistoryItem> = emptyList()
private var historyList: HistoryList = HistoryList(emptyList())
private var mode: HistoryState.Mode = HistoryState.Mode.Normal
fun updateData(items: List<HistoryItem>, mode: HistoryState.Mode) {
this.items = items
this.historyList = HistoryList(items)
this.mode = mode
notifyDataSetChanged()
}
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
val view = LayoutInflater.from(parent.context).inflate(viewType, parent, false)
return when (viewType) {
HistoryDeleteViewHolder.LAYOUT_ID -> HistoryDeleteViewHolder(view, actionEmitter)
HistoryHeaderViewHolder.LAYOUT_ID -> HistoryHeaderViewHolder(view)
HistoryListItemViewHolder.LAYOUT_ID -> HistoryListItemViewHolder(view, actionEmitter)
else -> throw IllegalStateException("viewType $viewType does not match to a ViewHolder")
}
}
override fun getItemViewType(position: Int): Int {
return when (position) {
0 -> HistoryDeleteViewHolder.LAYOUT_ID
1 -> HistoryHeaderViewHolder.LAYOUT_ID
else -> HistoryListItemViewHolder.LAYOUT_ID
}
}
override fun getItemCount(): Int = items.count() + NUMBER_OF_SECTIONS
override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) {
when (holder) {
is HistoryDeleteViewHolder -> holder.bind(mode)
// Temporarily hard code the header until
// we build out support for time based sections
is HistoryHeaderViewHolder -> holder.bind("Today")
is HistoryListItemViewHolder -> holder.bind(items[position - NUMBER_OF_SECTIONS], mode)
}
}
companion object {
// Temporarily hard code the number of sections until
// we build out support for time based sections
private const val NUMBER_OF_SECTIONS = 2
}
}

View File

@ -12,7 +12,7 @@ import org.mozilla.fenix.mvi.UIComponent
import org.mozilla.fenix.mvi.ViewState
import java.net.URL
data class HistoryItem(val id: Int, val url: String) {
data class HistoryItem(val id: Int, val url: String, val visitedAt: Long) {
val title: String
get() = siteTitle()

View File

@ -83,8 +83,9 @@ class HistoryFragment : Fragment(), CoroutineScope, BackHandler {
super.onViewCreated(view, savedInstanceState)
launch(Dispatchers.IO) {
val items = requireComponents.core.historyStorage.getVisited()
.mapIndexed { id, item -> HistoryItem(id, item) }
val items = requireComponents.core.historyStorage.getDetailedVisits(0)
.asReversed()
.mapIndexed { id, item -> HistoryItem(id, item.url, item.visitTime) }
launch(Dispatchers.Main) {
getManagedEmitter<HistoryChange>().onNext(HistoryChange.Change(items))

View File

@ -261,6 +261,15 @@
is the number of items you have selected -->
<string name="history_delete_some">Delete %1$d items</string>
<!-- Text for the header that groups the history for Today -->
<string name="history_today">Today</string>
<!-- Text for the header that groups the history the past week -->
<string name="history_this_week">This Week</string>
<!-- Text for the header that groups the history the past month -->
<string name="history_this_month">This Month</string>
<!-- Text for the header that groups the history older than the last month -->
<string name="history_older">Older</string>
<!-- Text displayed in a notification when the user enters full screen mode -->
<string name="full_screen_notification">Entering full screen mode</string>
@ -281,4 +290,5 @@
<!-- Content Description for session item share button -->
<string name="content_description_session_share">Share Session</string>
</resources>