Library List updates

Empty categories now show
Moving a manga from one category to the next now shows a snackbar with undo
Edit categories now shows total category count
This commit is contained in:
Jay 2020-03-21 23:48:56 -04:00
parent 3e40105232
commit ff948ea4d9
12 changed files with 124 additions and 29 deletions

View File

@ -5,4 +5,14 @@ class LibraryManga : MangaImpl() {
var unread: Int = 0 var unread: Int = 0
var category: Int = 0 var category: Int = 0
fun isBlank() = id == Long.MIN_VALUE
companion object {
fun createBlank(categoryId: Int): LibraryManga = LibraryManga().apply {
title = ""
id = Long.MIN_VALUE
category = categoryId
}
}
} }

View File

@ -40,7 +40,7 @@ class LibraryBadge @JvmOverloads constructor(context: Context, attrs: AttributeS
} }
// Show the bade card if unread or downloads exists // Show the bade card if unread or downloads exists
badge_view.visibility = if (download_text.visibility == View.VISIBLE || unread_text visibility = if (download_text.visibility == View.VISIBLE || unread_text
.visibility != View.GONE) View.VISIBLE else View.GONE .visibility != View.GONE) View.VISIBLE else View.GONE
// Show the angles divider if both unread and downloads exists // Show the angles divider if both unread and downloads exists
@ -63,7 +63,7 @@ class LibraryBadge @JvmOverloads constructor(context: Context, attrs: AttributeS
} }
fun setInLibrary(inLibrary: Boolean) { fun setInLibrary(inLibrary: Boolean) {
badge_view.visibility = if (inLibrary) View.VISIBLE else View.GONE visibility = if (inLibrary) View.VISIBLE else View.GONE
unread_angle.visibility = View.GONE unread_angle.visibility = View.GONE
unread_text.updatePaddingRelative(start = 5.dpToPx) unread_text.updatePaddingRelative(start = 5.dpToPx)
unread_text.visibility = if (inLibrary) View.VISIBLE else View.GONE unread_text.visibility = if (inLibrary) View.VISIBLE else View.GONE

View File

@ -37,7 +37,7 @@ class LibraryItem(
var chapterCount = -1 var chapterCount = -1
override fun getLayoutRes(): Int { override fun getLayoutRes(): Int {
return if (libraryLayout.getOrDefault() == 0) return if (libraryLayout.getOrDefault() == 0 || manga.isBlank())
R.layout.catalogue_list_item R.layout.catalogue_list_item
else else
R.layout.catalogue_grid_item R.layout.catalogue_grid_item
@ -48,7 +48,7 @@ class LibraryItem(
return if (parent is AutofitRecyclerView) { return if (parent is AutofitRecyclerView) {
val libraryLayout = libraryLayout.getOrDefault() val libraryLayout = libraryLayout.getOrDefault()
val isFixedSize = fixedSize.getOrDefault() val isFixedSize = fixedSize.getOrDefault()
if (libraryLayout == 0) { if (libraryLayout == 0 || manga.isBlank()) {
LibraryListHolder(view, adapter as LibraryCategoryAdapter) LibraryListHolder(view, adapter as LibraryCategoryAdapter)
} else { } else {
view.apply { view.apply {
@ -111,7 +111,15 @@ class LibraryItem(
* Returns true if this item is draggable. * Returns true if this item is draggable.
*/ */
override fun isDraggable(): Boolean { override fun isDraggable(): Boolean {
return true return !manga.isBlank()
}
override fun isEnabled(): Boolean {
return !manga.isBlank()
}
override fun isSelectable(): Boolean {
return !manga.isBlank()
} }
/** /**
@ -121,6 +129,8 @@ class LibraryItem(
* @return true if the manga should be included, false otherwise. * @return true if the manga should be included, false otherwise.
*/ */
override fun filter(constraint: String): Boolean { override fun filter(constraint: String): Boolean {
if (manga.isBlank())
return constraint.isEmpty()
val sourceManager by injectLazy<SourceManager>() val sourceManager by injectLazy<SourceManager>()
val sourceName = if (manga.source == 0L) "Local" else val sourceName = if (manga.source == 0L) "Local" else
sourceManager.getOrStub(manga.source).name sourceManager.getOrStub(manga.source).name

View File

@ -22,10 +22,12 @@ import com.afollestad.materialdialogs.checkbox.isCheckPromptChecked
import com.bluelinelabs.conductor.ControllerChangeHandler import com.bluelinelabs.conductor.ControllerChangeHandler
import com.bluelinelabs.conductor.ControllerChangeType import com.bluelinelabs.conductor.ControllerChangeType
import com.google.android.material.bottomsheet.BottomSheetBehavior import com.google.android.material.bottomsheet.BottomSheetBehavior
import com.google.android.material.snackbar.Snackbar
import eu.davidea.flexibleadapter.FlexibleAdapter import eu.davidea.flexibleadapter.FlexibleAdapter
import eu.davidea.flexibleadapter.SelectableAdapter import eu.davidea.flexibleadapter.SelectableAdapter
import eu.davidea.flexibleadapter.items.IFlexible import eu.davidea.flexibleadapter.items.IFlexible
import eu.kanade.tachiyomi.R import eu.kanade.tachiyomi.R
import eu.kanade.tachiyomi.data.database.models.Category
import eu.kanade.tachiyomi.data.database.models.LibraryManga import eu.kanade.tachiyomi.data.database.models.LibraryManga
import eu.kanade.tachiyomi.data.database.models.Manga import eu.kanade.tachiyomi.data.database.models.Manga
import eu.kanade.tachiyomi.data.library.LibraryUpdateService import eu.kanade.tachiyomi.data.library.LibraryUpdateService
@ -291,7 +293,9 @@ class LibraryListController(bundle: Bundle? = null) : LibraryController(bundle),
override fun getSpanSize(position: Int): Int { override fun getSpanSize(position: Int): Int {
if (libraryLayout == 0) return 1 if (libraryLayout == 0) return 1
val item = this@LibraryListController.adapter.getItem(position) val item = this@LibraryListController.adapter.getItem(position)
return if (item is LibraryHeaderItem) recycler.manager.spanCount else 1 return if (item is LibraryHeaderItem) recycler.manager.spanCount
else if (item is LibraryItem && item.manga.isBlank()) recycler.manager.spanCount
else 1
} }
}) })
recycler.setHasFixedSize(true) recycler.setHasFixedSize(true)
@ -493,7 +497,7 @@ class LibraryListController(bundle: Bundle? = null) : LibraryController(bundle),
private fun toggleSelection(position: Int) { private fun toggleSelection(position: Int) {
val item = adapter.getItem(position) as? LibraryItem ?: return val item = adapter.getItem(position) as? LibraryItem ?: return
if (item.manga.isBlank()) return
setSelection(item.manga, !adapter.isSelected(position)) setSelection(item.manga, !adapter.isSelected(position))
invalidateActionMode() invalidateActionMode()
} }
@ -625,14 +629,14 @@ class LibraryListController(bundle: Bundle? = null) : LibraryController(bundle),
return return
} }
if (newHeader?.category?.mangaSort == null) { if (newHeader?.category?.mangaSort == null) {
presenter.moveMangaToCategory(item, newHeader?.category?.id, mangaIds, true) moveMangaToCategory(item.manga, newHeader?.category, mangaIds, true)
} else { } else {
val keepCatSort = preferences.keepCatSort().getOrDefault() val keepCatSort = preferences.keepCatSort().getOrDefault()
if (keepCatSort == 0) { if (keepCatSort == 0) {
MaterialDialog(activity!!).message(R.string.switch_to_dnd) MaterialDialog(activity!!).message(R.string.switch_to_dnd)
.positiveButton(R.string.action_switch) { .positiveButton(R.string.action_switch) {
presenter.moveMangaToCategory( moveMangaToCategory(
item, newHeader.category.id, mangaIds, true item.manga, newHeader.category, mangaIds, true
) )
if (it.isCheckPromptChecked()) preferences.keepCatSort().set(2) if (it.isCheckPromptChecked()) preferences.keepCatSort().set(2)
}.checkBoxPrompt(R.string.remember_choice) {}.negativeButton( }.checkBoxPrompt(R.string.remember_choice) {}.negativeButton(
@ -643,14 +647,14 @@ class LibraryListController(bundle: Bundle? = null) : LibraryController(bundle),
) )
) )
) { ) {
presenter.moveMangaToCategory( moveMangaToCategory(
item, newHeader.category.id, mangaIds, false item.manga, newHeader.category, mangaIds, false
) )
if (it.isCheckPromptChecked()) preferences.keepCatSort().set(1) if (it.isCheckPromptChecked()) preferences.keepCatSort().set(1)
}.cancelOnTouchOutside(false).show() }.cancelOnTouchOutside(false).show()
} else { } else {
presenter.moveMangaToCategory( moveMangaToCategory(
item, newHeader.category.id, mangaIds, keepCatSort == 2 item.manga, newHeader.category, mangaIds, keepCatSort == 2
) )
} }
} }
@ -658,6 +662,27 @@ class LibraryListController(bundle: Bundle? = null) : LibraryController(bundle),
lastItemPosition = null lastItemPosition = null
} }
private fun moveMangaToCategory(
manga: LibraryManga,
category: Category?,
mangaIds: List<Long>,
useDND: Boolean
) {
if (category?.id == null) return
val oldCatId = manga.category
presenter.moveMangaToCategory(manga, category.id, mangaIds, useDND)
snack?.dismiss()
snack = view?.snack(
resources!!.getString(R.string.moved_to_category, category.name)
) {
anchorView = bottom_sheet
setAction(R.string.action_undo) {
manga.category = category.id!!
presenter.moveMangaToCategory(manga, oldCatId, mangaIds, useDND)
}
}
}
override fun updateCategory(catId: Int): Boolean { override fun updateCategory(catId: Int): Boolean {
val category = (adapter.getItem(catId) as? LibraryHeaderItem)?.category ?: return false val category = (adapter.getItem(catId) as? LibraryHeaderItem)?.category ?: return false
val inQueue = LibraryUpdateService.categoryInQueue(category.id) val inQueue = LibraryUpdateService.categoryInQueue(category.id)
@ -669,7 +694,7 @@ class LibraryListController(bundle: Bundle? = null) : LibraryController(bundle),
LibraryUpdateService.isRunning() -> R.string.adding_category_to_queue LibraryUpdateService.isRunning() -> R.string.adding_category_to_queue
else -> R.string.updating_category_x else -> R.string.updating_category_x
}, category.name }, category.name
) ), Snackbar.LENGTH_LONG
) { ) {
anchorView = bottom_sheet anchorView = bottom_sheet
} }

View File

@ -7,6 +7,8 @@ import com.bumptech.glide.signature.ObjectKey
import eu.kanade.tachiyomi.R import eu.kanade.tachiyomi.R
import eu.kanade.tachiyomi.data.database.models.MangaImpl import eu.kanade.tachiyomi.data.database.models.MangaImpl
import eu.kanade.tachiyomi.data.glide.GlideApp import eu.kanade.tachiyomi.data.glide.GlideApp
import eu.kanade.tachiyomi.util.view.gone
import eu.kanade.tachiyomi.util.view.visible
import kotlinx.android.synthetic.main.catalogue_list_item.* import kotlinx.android.synthetic.main.catalogue_list_item.*
import kotlinx.android.synthetic.main.catalogue_list_item.view.* import kotlinx.android.synthetic.main.catalogue_list_item.view.*
import kotlinx.android.synthetic.main.unread_download_badge.* import kotlinx.android.synthetic.main.unread_download_badge.*
@ -33,6 +35,17 @@ class LibraryListHolder(
* @param item the manga item to bind. * @param item the manga item to bind.
*/ */
override fun onSetValues(item: LibraryItem) { override fun onSetValues(item: LibraryItem) {
if (item.manga.isBlank()) {
title.text = itemView.context.getString(R.string.category_is_empty)
title.textAlignment = View.TEXT_ALIGNMENT_CENTER
card.gone()
badge_view.gone()
return
}
card.visible()
title.textAlignment = View.TEXT_ALIGNMENT_TEXT_START
// Update the title of the manga. // Update the title of the manga.
title.text = item.manga.title title.text = item.manga.title
setUnreadBadge(badge_view, item) setUnreadBadge(badge_view, item)

View File

@ -445,6 +445,7 @@ class LibraryPresenter(
val showCategories = !preferences.hideCategories().getOrDefault() val showCategories = !preferences.hideCategories().getOrDefault()
val unreadBadgeType = preferences.unreadBadgeType().getOrDefault() val unreadBadgeType = preferences.unreadBadgeType().getOrDefault()
var libraryManga = db.getLibraryMangas().executeAsBlocking() var libraryManga = db.getLibraryMangas().executeAsBlocking()
val singleList = preferences.libraryAsSingleList().getOrDefault()
if (!showCategories) if (!showCategories)
libraryManga = libraryManga.distinctBy { it.id } libraryManga = libraryManga.distinctBy { it.id }
/*val libraryMap = libraryManga.map { manga -> /*val libraryMap = libraryManga.map { manga ->
@ -457,7 +458,7 @@ class LibraryPresenter(
preferences.librarySortingAscending().getOrDefault()) preferences.librarySortingAscending().getOrDefault())
val catItemAll = LibraryHeaderItem({ categoryAll }, -1) val catItemAll = LibraryHeaderItem({ categoryAll }, -1)
val libraryMap = val libraryMap =
if (!preferences.libraryAsSingleList().getOrDefault()) { if (!singleList) {
libraryManga.map { manga -> libraryManga.map { manga ->
LibraryItem(manga, libraryLayout, preferences.uniformGrid(), null).apply { unreadType = LibraryItem(manga, libraryLayout, preferences.uniformGrid(), null).apply { unreadType =
unreadBadgeType } unreadBadgeType }
@ -482,10 +483,27 @@ class LibraryPresenter(
cat to it cat to it
// LibraryItem(manga, libraryLayout).apply { unreadType = unreadBadgeType } // LibraryItem(manga, libraryLayout).apply { unreadType = unreadBadgeType }
}.toMap() }.toMap()
} }.toMutableMap()
if (libraryMap.containsKey(0)) if (libraryMap.containsKey(0))
categories.add(0, createDefaultCategory()) categories.add(0, createDefaultCategory())
if (showCategories) {
categories.forEach { category ->
if (!libraryMap.containsKey(category.id)) {
val headerItem =
LibraryHeaderItem({ getCategory(category.id!!) }, category.id!!)
libraryMap[category.id!!] = listOf(
LibraryItem(
LibraryManga.createBlank(category.id!!),
libraryLayout,
preferences.uniformGrid(),
headerItem
)
)
}
}
}
if (categories.size == 1 && showCategories) if (categories.size == 1 && showCategories)
categories.first().name = context.getString(R.string.label_library) categories.first().name = context.getString(R.string.label_library)
@ -831,13 +849,17 @@ class LibraryPresenter(
} }
} }
fun moveMangaToCategory(item: LibraryItem, catId: Int?, mangaIds: List<Long>, useDND: Boolean) { fun moveMangaToCategory(
manga: LibraryManga,
catId: Int?,
mangaIds: List<Long>,
useDND: Boolean
) {
GlobalScope.launch(Dispatchers.IO) { GlobalScope.launch(Dispatchers.IO) {
val categoryId = catId ?: return@launch val categoryId = catId ?: return@launch
val category = categories.find { catId == it.id } ?: return@launch val category = categories.find { catId == it.id } ?: return@launch
val manga = item.manga
val mangaMap = currentMangaMap?.toMutableMap() ?: return@launch /*val mangaMap = currentMangaMap?.toMutableMap() ?: return@launch
val oldCatId = item.manga.category val oldCatId = item.manga.category
val oldCatMap = mangaMap[manga.category]?.toMutableList() ?: return@launch val oldCatMap = mangaMap[manga.category]?.toMutableList() ?: return@launch
val newCatMap = mangaMap[catId]?.toMutableList() ?: return@launch val newCatMap = mangaMap[catId]?.toMutableList() ?: return@launch
@ -845,13 +867,14 @@ class LibraryPresenter(
newCatMap.add(item) newCatMap.add(item)
mangaMap[oldCatId] = oldCatMap mangaMap[oldCatId] = oldCatMap
mangaMap[catId] = newCatMap mangaMap[catId] = newCatMap
currentMangaMap = mangaMap currentMangaMap = mangaMap*/
item.manga.category = categoryId val oldCatId = manga.category
manga.category = categoryId
val mc = ArrayList<MangaCategory>() val mc = ArrayList<MangaCategory>()
val categories = val categories = db.getCategoriesForManga(manga).executeAsBlocking()
db.getCategoriesForManga(manga).executeAsBlocking().filter { it.id != oldCatId } + listOf(category) .filter { it.id != oldCatId } + listOf(category)
for (cat in categories) { for (cat in categories) {
mc.add(MangaCategory.create(manga, cat)) mc.add(MangaCategory.create(manga, cat))
@ -864,7 +887,8 @@ class LibraryPresenter(
val ids = mangaIds.toMutableList() val ids = mangaIds.toMutableList()
if (!ids.contains(manga.id!!)) ids.add(manga.id!!) if (!ids.contains(manga.id!!)) ids.add(manga.id!!)
category.mangaOrder = ids category.mangaOrder = ids
if (category.id == 0) preferences.defaultMangaOrder().set(mangaIds.joinToString("/")) if (category.id == 0) preferences.defaultMangaOrder()
.set(mangaIds.joinToString("/"))
else db.insertCategory(category).executeAsBlocking() else db.insertCategory(category).executeAsBlocking()
} }
getLibrary() getLibrary()

View File

@ -98,6 +98,8 @@ import eu.kanade.tachiyomi.util.view.getText
import eu.kanade.tachiyomi.util.view.snack import eu.kanade.tachiyomi.util.view.snack
import eu.kanade.tachiyomi.util.view.updateLayoutParams import eu.kanade.tachiyomi.util.view.updateLayoutParams
import eu.kanade.tachiyomi.util.view.updatePaddingRelative import eu.kanade.tachiyomi.util.view.updatePaddingRelative
import java.io.File
import java.io.IOException
import jp.wasabeef.glide.transformations.CropSquareTransformation import jp.wasabeef.glide.transformations.CropSquareTransformation
import jp.wasabeef.glide.transformations.MaskTransformation import jp.wasabeef.glide.transformations.MaskTransformation
import kotlinx.android.synthetic.main.main_activity.* import kotlinx.android.synthetic.main.main_activity.*
@ -106,8 +108,6 @@ import kotlinx.android.synthetic.main.manga_header_item.*
import timber.log.Timber import timber.log.Timber
import uy.kohesive.injekt.Injekt import uy.kohesive.injekt.Injekt
import uy.kohesive.injekt.api.get import uy.kohesive.injekt.api.get
import java.io.File
import java.io.IOException
class MangaDetailsController : BaseController, class MangaDetailsController : BaseController,
FlexibleAdapter.OnItemClickListener, FlexibleAdapter.OnItemClickListener,

View File

@ -51,6 +51,8 @@ class SettingsLibraryController : SettingsController() {
titleRes = R.string.pref_category_library_categories titleRes = R.string.pref_category_library_categories
preference { preference {
titleRes = R.string.action_edit_categories titleRes = R.string.action_edit_categories
val catCount = db.getCategories().executeAsBlocking().size
summary = context.resources.getQuantityString(R.plurals.category, catCount, catCount)
onClick { router.pushController(CategoryController().withFadeTransaction()) } onClick { router.pushController(CategoryController().withFadeTransaction()) }
} }
intListPreference(activity) { intListPreference(activity) {

View File

@ -72,7 +72,7 @@ class SettingsMainController : SettingsController() {
onClick { navigateTo(SettingsAdvancedController()) } onClick { navigateTo(SettingsAdvancedController()) }
} }
preference { preference {
iconRes = R.drawable.ic_help_black_24dp iconRes = R.drawable.ic_info_black_24dp
iconTint = tintColor iconTint = tintColor
titleRes = R.string.pref_category_about titleRes = R.string.pref_category_about
onClick { navigateTo(SettingsAboutController()) } onClick { navigateTo(SettingsAboutController()) }

View File

@ -0,0 +1,5 @@
<vector android:height="24dp" android:tint="#000000"
android:viewportHeight="24.0" android:viewportWidth="24.0"
android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
<path android:fillColor="#FF000000" android:pathData="M12,2C6.48,2 2,6.48 2,12s4.48,10 10,10 10,-4.48 10,-10S17.52,2 12,2zM13,17h-2v-6h2v6zM13,9h-2L11,7h2v2z"/>
</vector>

View File

@ -77,7 +77,7 @@
<com.google.android.material.textview.MaterialTextView <com.google.android.material.textview.MaterialTextView
android:id="@+id/title" android:id="@+id/title"
style="@style/TextAppearance.Regular.Body1.SemiBold" style="@style/TextAppearance.Regular.Body1.SemiBold"
android:layout_width="wrap_content" android:layout_width="0dp"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_marginStart="12dp" android:layout_marginStart="12dp"
android:layout_marginTop="2dp" android:layout_marginTop="2dp"

View File

@ -197,6 +197,10 @@
<string name="landscape">Landscape</string> <string name="landscape">Landscape</string>
<string name="default_columns">Default</string> <string name="default_columns">Default</string>
<string name="create_new_category">Create new category</string> <string name="create_new_category">Create new category</string>
<plurals name="category">
<item quantity="one">%d category</item>
<item quantity="other">%d categories</item>
</plurals>
<string name="pref_category_library_update">Updates</string> <string name="pref_category_library_update">Updates</string>
<string name="pref_library_update_interval">Library update frequency</string> <string name="pref_library_update_interval">Library update frequency</string>
@ -442,6 +446,8 @@
<string name="updating_category_x">Updating %1$s</string> <string name="updating_category_x">Updating %1$s</string>
<string name="adding_category_to_queue">Adding %1$s to update queue</string> <string name="adding_category_to_queue">Adding %1$s to update queue</string>
<string name="category_already_in_queue">%1$s is already in queue</string> <string name="category_already_in_queue">%1$s is already in queue</string>
<string name="moved_to_category">Moved to %1$s</string>
<string name="category_is_empty">Category is empty</string>
<string name="local_source_badge">Local</string> <string name="local_source_badge">Local</string>
<string name="confirm_manga_deletion">Remove from library?</string> <string name="confirm_manga_deletion">Remove from library?</string>
<string name="switch_to_dnd">Switch to Drag &amp; Drop mode?</string> <string name="switch_to_dnd">Switch to Drag &amp; Drop mode?</string>