Moved drag & drop sorting to library presenter

Where it always belonged
also added radio buttons sort category by (used to be called reorder)
also added unread sorting to that^
This commit is contained in:
Jay 2020-01-14 21:29:32 -08:00
parent ab75f44d9c
commit 7008752639
8 changed files with 119 additions and 71 deletions

View File

@ -156,35 +156,24 @@ class LibraryCategoryView @JvmOverloads constructor(context: Context, attrs: Att
subscriptions += controller.reorganizeRelay subscriptions += controller.reorganizeRelay
.subscribe { .subscribe {
if (it.first == category.id) { if (it.first == category.id) {
var items:List<LibraryItem> if (it.second in -2..-1) {
if (it.second in 5..6) { val items = adapter.currentItems.toMutableList()
items = adapter.currentItems.toMutableList()
val mangas = controller.selectedMangas val mangas = controller.selectedMangas
val selectedManga = items.filter { item -> item.manga in mangas } val selectedManga = items.filter { item -> item.manga in mangas }
items.removeAll(selectedManga) items.removeAll(selectedManga)
if (it.second == 5) items.addAll(0, selectedManga) if (it.second == -1) items.addAll(0, selectedManga)
else items.addAll(selectedManga) else items.addAll(selectedManga)
adapter.setItems(items) adapter.setItems(items)
adapter.notifyDataSetChanged() adapter.notifyDataSetChanged()
saveDragSort() saveDragSort()
} }
else { else {
items = when (it.second) {
1, 2 -> adapter.currentItems.sortedBy {
if (preferences.removeArticles().getOrDefault()) it.manga.title.removeArticles()
else it.manga.title
}
3, 4 -> adapter.currentItems.sortedBy { it.manga.last_update }
else -> adapter.currentItems.sortedBy { it.manga.title }
}
if (it.second % 2 == 0) items = items.reversed()
adapter.setItems(items)
adapter.notifyDataSetChanged()
category.mangaSort = ('a' + (it.second - 1)) category.mangaSort = ('a' + (it.second - 1))
if (category.name == "Default") if (category.name == "Default")
preferences.defaultMangaOrder().set(category.mangaSort.toString()) preferences.defaultMangaOrder().set(category.mangaSort.toString())
else else
db.insertCategory(category).asRxObservable().subscribe() db.insertCategory(category).asRxObservable().subscribe()
controller.enableReorderItems(category)
} }
} }
} }
@ -218,11 +207,8 @@ class LibraryCategoryView @JvmOverloads constructor(context: Context, attrs: Att
*/ */
fun onNextLibraryManga(event: LibraryMangaEvent) { fun onNextLibraryManga(event: LibraryMangaEvent) {
// Get the manga list for this category. // Get the manga list for this category.
val sortingMode = preferences.librarySortingMode().getOrDefault()
adapter.isLongPressDragEnabled = canDrag() adapter.isLongPressDragEnabled = canDrag()
var mangaForCategory = event.getMangaForCategory(category).orEmpty() val mangaForCategory = event.getMangaForCategory(category).orEmpty()
if (sortingMode == LibrarySort.DRAG_AND_DROP)
mangaForCategory = sortMangaInDragAnDrop(mangaForCategory)
// Update the category with its manga. // Update the category with its manga.
adapter.setItems(mangaForCategory) adapter.setItems(mangaForCategory)
@ -238,32 +224,6 @@ class LibraryCategoryView @JvmOverloads constructor(context: Context, attrs: Att
} }
} }
fun sortMangaInDragAnDrop(mangaForCategory: List<LibraryItem>): List<LibraryItem> {
if (category.name == "Default") {
val defOrder = preferences.defaultMangaOrder().getOrDefault()
if (defOrder.first().isLetter()) category.mangaSort = defOrder.first()
else category.mangaOrder = defOrder.split("/").mapNotNull { it.toLongOrNull() }
}
return if (category.mangaSort != null) {
var mangas = when (category.mangaSort) {
'a', 'b' -> mangaForCategory.sortedBy {
if (preferences.removeArticles().getOrDefault()) it.manga.title.removeArticles()
else it.manga.title
}
'c', 'd' -> mangaForCategory.sortedBy { it.manga.last_update }
else -> mangaForCategory.sortedBy { it.manga.title }
}
if (category.mangaSort == 'b' || category.mangaSort == 'd')
mangas = mangas.asReversed()
mangas
} else mangaForCategory.sortedBy {
category.mangaOrder.indexOf(
it.manga.id
)
}
}
/** /**
* Subscribe to [LibrarySelectionEvent]. When an event is received, it updates the selection * Subscribe to [LibrarySelectionEvent]. When an event is received, it updates the selection
* depending on the type of event received. * depending on the type of event received.
@ -373,6 +333,8 @@ class LibraryCategoryView @JvmOverloads constructor(context: Context, attrs: Att
preferences.defaultMangaOrder().set(mangaIds.joinToString("/")) preferences.defaultMangaOrder().set(mangaIds.joinToString("/"))
else else
db.insertCategory(category).asRxObservable().subscribe() db.insertCategory(category).asRxObservable().subscribe()
controller.onSortChanged()
controller.enableReorderItems(category)
} }
override fun shouldMoveItem(fromPosition: Int, toPosition: Int): Boolean { override fun shouldMoveItem(fromPosition: Int, toPosition: Int): Boolean {
if (adapter.selectedItemCount > 1) if (adapter.selectedItemCount > 1)

View File

@ -22,6 +22,7 @@ import androidx.core.content.ContextCompat.getSystemService
import androidx.core.graphics.drawable.DrawableCompat import androidx.core.graphics.drawable.DrawableCompat
import androidx.core.view.GravityCompat import androidx.core.view.GravityCompat
import androidx.drawerlayout.widget.DrawerLayout import androidx.drawerlayout.widget.DrawerLayout
import androidx.viewpager.widget.ViewPager
import com.bluelinelabs.conductor.ControllerChangeHandler import com.bluelinelabs.conductor.ControllerChangeHandler
import com.bluelinelabs.conductor.ControllerChangeType import com.bluelinelabs.conductor.ControllerChangeType
import com.f2prateek.rx.preferences.Preference import com.f2prateek.rx.preferences.Preference
@ -158,6 +159,8 @@ class LibraryController(
var snack: Snackbar? = null var snack: Snackbar? = null
private var reorderMenuItem:MenuItem? = null
init { init {
setHasOptionsMenu(true) setHasOptionsMenu(true)
retainViewMode = RetainViewMode.RETAIN_DETACH retainViewMode = RetainViewMode.RETAIN_DETACH
@ -185,6 +188,20 @@ class LibraryController(
activeCategory = it activeCategory = it
} }
library_pager.addOnPageChangeListener(object : ViewPager.OnPageChangeListener {
override fun onPageSelected(position: Int) {
enableReorderItems(position)
}
override fun onPageScrolled(
position: Int,
positionOffset: Float,
positionOffsetPixels: Int
) { }
override fun onPageScrollStateChanged(state: Int) { }
})
getColumnsPreferenceForCurrentOrientation().asObservable() getColumnsPreferenceForCurrentOrientation().asObservable()
.doOnNext { mangaPerRow = it } .doOnNext { mangaPerRow = it }
.skip(1) .skip(1)
@ -196,6 +213,31 @@ class LibraryController(
} }
} }
fun enableReorderItems(category: Category) {
adapter?.categories?.getOrNull(library_pager.currentItem)?.mangaSort = category.mangaSort
enableReorderItems(sortType = category.mangaSort)
}
private fun enableReorderItems(position: Int? = null, sortType: Char? = null) {
val pos = position ?: library_pager.currentItem
val orderOfCat = sortType ?: adapter?.categories?.getOrNull(pos)?.mangaSort
if (reorderMenuItem?.isVisible != true) return
val subMenu = reorderMenuItem?.subMenu ?: return
if (orderOfCat != null) {
subMenu.setGroupCheckable(R.id.reorder_group, true, true)
when (orderOfCat) {
'a' -> subMenu.findItem(R.id.action_alpha_asc)?.isChecked = true
'b' -> subMenu.findItem(R.id.action_alpha_dsc)?.isChecked = true
'c' -> subMenu.findItem(R.id.action_update_asc)?.isChecked = true
'd' -> subMenu.findItem(R.id.action_update_dsc)?.isChecked = true
'e' -> subMenu.findItem(R.id.action_unread)?.isChecked = true
}
}
else {
subMenu.setGroupCheckable(R.id.reorder_group, false, false)
}
}
override fun onChangeStarted(handler: ControllerChangeHandler, type: ControllerChangeType) { override fun onChangeStarted(handler: ControllerChangeHandler, type: ControllerChangeType) {
super.onChangeStarted(handler, type) super.onChangeStarted(handler, type)
if (type.isEnter) { if (type.isEnter) {
@ -337,7 +379,7 @@ class LibraryController(
/** /**
* Called when the sorting mode is changed. * Called when the sorting mode is changed.
*/ */
private fun onSortChanged() { fun onSortChanged() {
activity?.invalidateOptionsMenu() activity?.invalidateOptionsMenu()
presenter.requestSortUpdate() presenter.requestSortUpdate()
} }
@ -381,6 +423,8 @@ class LibraryController(
val reorganizeItem = menu.findItem(R.id.action_reorganize) val reorganizeItem = menu.findItem(R.id.action_reorganize)
reorganizeItem.isVisible = preferences.librarySortingMode().getOrDefault() == LibrarySort.DRAG_AND_DROP reorganizeItem.isVisible = preferences.librarySortingMode().getOrDefault() == LibrarySort.DRAG_AND_DROP
reorderMenuItem = reorganizeItem
enableReorderItems()
val searchItem = menu.findItem(R.id.action_search) val searchItem = menu.findItem(R.id.action_search)
val searchView = searchItem.actionView as SearchView val searchView = searchItem.actionView as SearchView
@ -439,8 +483,9 @@ class LibraryController(
} }
R.id.action_alpha_asc -> reOrder(1) R.id.action_alpha_asc -> reOrder(1)
R.id.action_alpha_dsc -> reOrder(2) R.id.action_alpha_dsc -> reOrder(2)
R.id.action_update_dsc -> reOrder(3) R.id.action_update_asc -> reOrder(3)
R.id.action_update_asc -> reOrder(4) R.id.action_update_dsc -> reOrder(4)
R.id.action_unread -> reOrder(5)
else -> return super.onOptionsItemSelected(item) else -> return super.onOptionsItemSelected(item)
} }
@ -450,6 +495,7 @@ class LibraryController(
private fun reOrder(type: Int) { private fun reOrder(type: Int) {
adapter?.categories?.getOrNull(library_pager.currentItem)?.id?.let { adapter?.categories?.getOrNull(library_pager.currentItem)?.id?.let {
reorganizeRelay.call(it to type) reorganizeRelay.call(it to type)
onSortChanged()
} }
} }
@ -527,7 +573,7 @@ class LibraryController(
} }
R.id.action_to_top, R.id.action_to_bottom -> { R.id.action_to_top, R.id.action_to_bottom -> {
adapter?.categories?.getOrNull(library_pager.currentItem)?.id?.let { adapter?.categories?.getOrNull(library_pager.currentItem)?.id?.let {
reorganizeRelay.call(it to if (item.itemId == R.id.action_to_top) 5 else 6) reorganizeRelay.call(it to if (item.itemId == R.id.action_to_top) -1 else -2)
} }
destroyActionModeIfNeeded() destroyActionModeIfNeeded()
} }

View File

@ -200,6 +200,14 @@ class LibraryPresenter(
var counter = 0 var counter = 0
db.getTotalChapterManga().executeAsBlocking().associate { it.id!! to counter++ } db.getTotalChapterManga().executeAsBlocking().associate { it.id!! to counter++ }
} }
val catListing by lazy {
val default = Category.createDefault()
val defOrder = preferences.defaultMangaOrder().getOrDefault()
if (defOrder.firstOrNull()?.isLetter() == true) default.mangaSort = defOrder.first()
else default.mangaOrder = defOrder.split("/").mapNotNull { it.toLongOrNull() }
listOf(default) + db.getCategories().executeAsBlocking()
}
val sortFn: (LibraryItem, LibraryItem) -> Int = { i1, i2 -> val sortFn: (LibraryItem, LibraryItem) -> Int = { i1, i2 ->
when (sortingMode) { when (sortingMode) {
@ -220,9 +228,35 @@ class LibraryPresenter(
if (mangaCompare == 0) sortAlphabetical(i1, i2) else mangaCompare if (mangaCompare == 0) sortAlphabetical(i1, i2) else mangaCompare
} }
LibrarySort.DRAG_AND_DROP -> { LibrarySort.DRAG_AND_DROP -> {
0 if (i1.manga.category == i2.manga.category) {
val category = catListing.find { it.id == i1.manga.category }
when {
category?.mangaSort != null -> {
when (category.mangaSort) {
'a' -> sortAlphabetical(i1, i2)
'b' -> sortAlphabetical(i2, i1)
'c' -> i2.manga.last_update.compareTo(i1.manga.last_update)
'd' -> i1.manga.last_update.compareTo(i2.manga.last_update)
'e' -> i2.manga.unread.compareTo(i1.manga.unread)
else -> sortAlphabetical(i1, i2)
}
}
category?.mangaOrder?.isEmpty() == false -> {
val order = category.mangaOrder
val index1 = order.indexOf(i1.manga.id!!)
val index2 = order.indexOf(i2.manga.id!!)
when {
index1 == -1 -> -1
index2 == -1 -> 1
else -> index1.compareTo(index2)
}
}
else -> 0
}
}
else 0
} }
else -> sortAlphabetical(i1, i2) else -> 0
} }
} }

View File

@ -68,7 +68,7 @@ abstract class PagerViewer(val activity: ReaderActivity) : BaseViewer {
pager.offscreenPageLimit = 1 pager.offscreenPageLimit = 1
pager.id = R.id.reader_pager pager.id = R.id.reader_pager
pager.adapter = adapter pager.adapter = adapter
pager.addOnPageChangeListener(object : androidx.viewpager.widget.ViewPager.SimpleOnPageChangeListener() { pager.addOnPageChangeListener(object : ViewPager.SimpleOnPageChangeListener() {
override fun onPageSelected(position: Int) { override fun onPageSelected(position: Int) {
val page = adapter.items.getOrNull(position) val page = adapter.items.getOrNull(position)
if (page != null && currentPage != page) { if (page != null && currentPage != page) {
@ -81,7 +81,7 @@ abstract class PagerViewer(val activity: ReaderActivity) : BaseViewer {
} }
override fun onPageScrollStateChanged(state: Int) { override fun onPageScrollStateChanged(state: Int) {
isIdle = state == androidx.viewpager.widget.ViewPager.SCROLL_STATE_IDLE isIdle = state == ViewPager.SCROLL_STATE_IDLE
} }
}) })
pager.tapListener = { event -> pager.tapListener = { event ->

View File

@ -134,8 +134,8 @@ fun syncChaptersWithSource(db: DatabaseHelper,
db.fixChaptersSourceOrder(sourceChapters).executeAsBlocking() db.fixChaptersSourceOrder(sourceChapters).executeAsBlocking()
// Set this manga as updated since chapters were changed // Set this manga as updated since chapters were changed
val newestChaper = db.getChapters(manga).executeAsBlocking().maxBy { it.date_fetch } val newestChatper = db.getChapters(manga).executeAsBlocking().maxBy { it.date_upload }
val dateFetch = newestChaper?.date_fetch ?: manga.last_update val dateFetch = newestChatper?.date_upload ?: manga.last_update
manga.last_update = dateFetch manga.last_update = dateFetch
db.updateLastUpdated(manga).executeAsBlocking() db.updateLastUpdated(manga).executeAsBlocking()
} }

View File

@ -1,9 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="match_parent" android:layout_height="match_parent">
xmlns:tools="http://schemas.android.com/tools"
xmlns:app="http://schemas.android.com/apk/res-auto">
<androidx.viewpager.widget.ViewPager <androidx.viewpager.widget.ViewPager
android:layout_width="match_parent" android:layout_width="match_parent"

View File

@ -34,21 +34,28 @@
<item <item
android:id="@+id/action_reorganize" android:id="@+id/action_reorganize"
android:title="@string/action_reorganize_by" android:title="@string/action_sort_by"
app:showAsAction="never"> app:showAsAction="never">
<menu> <menu>
<item <group android:id="@+id/reorder_group"
android:id="@+id/action_alpha_asc" android:checkableBehavior="single">
android:title="@string/action_sort_alpha"/> <item
<item android:id="@+id/action_alpha_asc"
android:id="@+id/action_alpha_dsc" android:title="@string/action_sort_alpha"/>
android:title="@string/action_alpha_reverse"/> <item
<item android:id="@+id/action_alpha_dsc"
android:id="@+id/action_update_asc" android:title="@string/action_alpha_reverse"/>
android:title="@string/action_sort_last_updated"/> <item
<item android:id="@+id/action_update_asc"
android:id="@+id/action_update_dsc" android:title="@string/action_sort_last_updated"/>
android:title="@string/action_sort_first_updated"/> <item
android:id="@+id/action_update_dsc"
android:title="@string/action_sort_first_updated"/>
<item
android:id="@+id/action_unread"
android:title="@string/action_filter_unread"/>
</group>
</menu> </menu>
</item> </item>

View File

@ -117,6 +117,7 @@
<string name="action_move_to_top">Move to top</string> <string name="action_move_to_top">Move to top</string>
<string name="action_move_to_bottom">Move to bottom</string> <string name="action_move_to_bottom">Move to bottom</string>
<string name="action_track">Track</string> <string name="action_track">Track</string>
<string name="action_sort_by">Sort category by…</string>
<!-- Operations --> <!-- Operations -->
<string name="deleting">Deleting…</string> <string name="deleting">Deleting…</string>