Skip to content

Commit

Permalink
improvement(browser-columns): allow cards/notes mismatch
Browse files Browse the repository at this point in the history
If a user is editing options and changes 'cards/notes'
then they should be editing columns for their provided selection

If a user saves columns, then reverts the switch, columns
should not be updated

Issue 17780
  • Loading branch information
david-allison committed Jan 13, 2025
1 parent b4e24c1 commit b01dba5
Show file tree
Hide file tree
Showing 5 changed files with 78 additions and 18 deletions.
2 changes: 1 addition & 1 deletion AnkiDroid/src/main/java/com/ichi2/anki/CardBrowser.kt
Original file line number Diff line number Diff line change
Expand Up @@ -414,7 +414,7 @@ open class CardBrowser :
findViewById<ViewGroup>(R.id.browser_column_headings).apply {
setOnLongClickListener {
Timber.d("long press on headings: opening column selection options")
val dialog = BrowserColumnSelectionFragment()
val dialog = BrowserColumnSelectionFragment.createInstance(viewModel.cardsOrNotes)
dialog.show(supportFragmentManager, "browserColumnsDialog")
true
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,8 @@ import android.view.KeyEvent
import android.view.View
import androidx.activity.OnBackPressedCallback
import androidx.annotation.StringRes
import androidx.core.os.BundleCompat
import androidx.core.os.bundleOf
import androidx.fragment.app.DialogFragment
import androidx.fragment.app.activityViewModels
import androidx.recyclerview.widget.ItemTouchHelper
Expand Down Expand Up @@ -76,6 +78,12 @@ class BrowserColumnSelectionFragment : DialogFragment(R.layout.preferences_brows

private lateinit var toolbar: MaterialToolbar

private val cardsOrNotes: CardsOrNotes
get() =
requireNotNull(
BundleCompat.getParcelable(requireArguments(), ARG_MODE, CardsOrNotes::class.java),
)

// this is needed as requireActivity().onBackPressedDispatcher.onBackPressed()
// closes the activity, but an actual back press closes the dialog
private val onCloseDialogCallback =
Expand Down Expand Up @@ -124,7 +132,7 @@ class BrowserColumnSelectionFragment : DialogFragment(R.layout.preferences_brows
}

Timber.d("save columns and close")
if (viewModel.updateColumns(columnAdapter.selected)) {
if (viewModel.updateColumns(columnAdapter.selected, cardsOrNotes)) {
dismiss()
true
} else {
Expand Down Expand Up @@ -174,7 +182,7 @@ class BrowserColumnSelectionFragment : DialogFragment(R.layout.preferences_brows
// Create a RecyclerView with 2 types of elements: [ColumnWithSample] and [ColumnUsage]
// Columns are draggable elements, the usage elements act as headings
// TODO: runBlocking shouldn't be necessary here.
val (displayed, available) = runBlocking { viewModel.previewColumnHeadings() }
val (displayed, available) = runBlocking { viewModel.previewColumnHeadings(cardsOrNotes) }
this.initiallySelectedColumns = displayed.map { it.columnType }

val recyclerViewItems =
Expand Down Expand Up @@ -237,6 +245,19 @@ class BrowserColumnSelectionFragment : DialogFragment(R.layout.preferences_brows
// Although a user can reorder the non-displayed columns, this order is not persisted
private val hasUnsavedChanges: Boolean
get() = initiallySelectedColumns != columnAdapter.selected

companion object {
const val ARG_MODE = "mode"

fun createInstance(cardsOrNotes: CardsOrNotes): BrowserColumnSelectionFragment =
BrowserColumnSelectionFragment().apply {
Timber.d("Building 'Manage columns' dialog for %s mode", cardsOrNotes)
arguments =
bundleOf(
ARG_MODE to cardsOrNotes,
)
}
}
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -516,7 +516,10 @@ class CardBrowserViewModel(
}

@CheckResult
fun updateColumns(columns: List<CardBrowserColumn>): Boolean {
fun updateColumns(
columns: List<CardBrowserColumn>,
cardsOrNotes: CardsOrNotes,
): Boolean {
if (columns.isEmpty()) return false.also { Timber.d("updateColumns: no columns") }
if (availableColumns == columns) return false.also { Timber.d("updateColumns: no changes") }

Expand Down Expand Up @@ -889,8 +892,13 @@ class CardBrowserViewModel(
* (1): An ordered list of columns which is displayed to the user
* (2): A list of columns which are available to display to the user
*/
suspend fun previewColumnHeadings(): Pair<List<ColumnWithSample>, List<ColumnWithSample>> {
val currentColumns = availableColumns
suspend fun previewColumnHeadings(cardsOrNotes: CardsOrNotes): Pair<List<ColumnWithSample>, List<ColumnWithSample>> {
val currentColumns =
when {
// if we match, use the loaded the columns
cardsOrNotes == this.cardsOrNotes -> availableColumns
else -> BrowserColumnCollection.load(sharedPrefs(), cardsOrNotes).columns
}

val columnsWithSample = ColumnWithSample.loadSample(cards.firstOrNull(), cardsOrNotes)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,13 +41,20 @@ class BrowserOptionsDialog : AppCompatDialogFragment() {

private val viewModel: CardBrowserViewModel by activityViewModels()

/** The unsaved value of [CardsOrNotes] */
private val dialogCardsOrNotes: CardsOrNotes
get() {
@IdRes val selectedButtonId =
dialogView.findViewById<RadioGroup>(R.id.select_browser_mode).checkedRadioButtonId
return when (selectedButtonId) {
R.id.select_cards_mode -> CardsOrNotes.CARDS
else -> CardsOrNotes.NOTES
}
}

private val positiveButtonClick = { _: DialogInterface, _: Int ->
@IdRes val selectedButtonId =
dialogView.findViewById<RadioGroup>(R.id.select_browser_mode).checkedRadioButtonId
val newCardsOrNotes =
if (selectedButtonId == R.id.select_cards_mode) CardsOrNotes.CARDS else CardsOrNotes.NOTES
if (cardsOrNotes != newCardsOrNotes) {
viewModel.setCardsOrNotes(newCardsOrNotes)
if (cardsOrNotes != dialogCardsOrNotes) {
viewModel.setCardsOrNotes(dialogCardsOrNotes)
}
val newTruncate = dialogView.findViewById<CheckBox>(R.id.truncate_checkbox).isChecked

Expand Down Expand Up @@ -111,7 +118,7 @@ class BrowserOptionsDialog : AppCompatDialogFragment() {

/** Opens [BrowserColumnSelectionFragment] for the current selection of [CardsOrNotes] */
private fun openColumnManager() {
val dialog = BrowserColumnSelectionFragment()
val dialog = BrowserColumnSelectionFragment.createInstance(viewModel.cardsOrNotes)
dialog.show(requireActivity().supportFragmentManager, "browserColumnsDialog")
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -427,6 +427,29 @@ class CardBrowserViewModelTest : JvmTest() {
}
}

@Test
fun `mode mismatch - changing columns`() {
// a user can use the options to update the inactive column set
// this should cause an update to the backend, but no frontend events should be obtained
runViewModelTest {
flowOfAvailableColumns.test {
ignoreEventsDuringViewModelInit()
assertThat("cardsOrNotes", cardsOrNotes, equalTo(CardsOrNotes.CARDS))

// we're in cards so editing NOTES should not update visible columns
setColumn(1, FSRS_STABILITY, CardsOrNotes.NOTES)
expectNoEvents()

BrowserColumnCollection.load(sharedPrefs(), CardsOrNotes.NOTES).apply {
assertThat("notes is changed", this.columns[1], equalTo(FSRS_STABILITY))
}
BrowserColumnCollection.load(sharedPrefs(), CardsOrNotes.CARDS).apply {
assertThat("cards is unchanged", this.columns[1], not(equalTo(FSRS_STABILITY)))
}
}
}
}

@Test
fun `change card order to NO_SORTING is a no-op if done twice`() =
runViewModelTest {
Expand Down Expand Up @@ -858,7 +881,7 @@ class CardBrowserViewModelTest : JvmTest() {
}
runViewModelTest(notes = 0) {
assertThat("no rows", rowCount, equalTo(0))
this.previewColumnHeadings().apply {
this.previewColumnHeadings(CardsOrNotes.CARDS).apply {
assertTrue("no sample values") { allColumns.all { it.sampleValue == null } }
val (displayed, _) = this
assertThat(
Expand All @@ -871,7 +894,7 @@ class CardBrowserViewModelTest : JvmTest() {

runViewModelNotesTest(notes = 0) {
assertThat("no rows", rowCount, equalTo(0))
this.previewColumnHeadings().apply {
this.previewColumnHeadings(CardsOrNotes.NOTES).apply {
assertTrue("no sample values") { allColumns.all { it.sampleValue == null } }
val (displayed, _) = this
assertThat(
Expand All @@ -892,7 +915,7 @@ class CardBrowserViewModelTest : JvmTest() {
@Test
fun `preview - cards`() {
runViewModelTest(notes = 1) {
for (preview in previewColumnHeadings().allColumns) {
for (preview in previewColumnHeadings(CardsOrNotes.CARDS).allColumns) {
val (expectedLabel, expectedValue) =
when (preview.columnType) {
SFLD -> Pair("Sort Field", "Front")
Expand Down Expand Up @@ -925,7 +948,7 @@ class CardBrowserViewModelTest : JvmTest() {
@Test
fun `preview - notes`() {
runViewModelNotesTest(notes = 1) {
for (preview in previewColumnHeadings().allColumns) {
for (preview in previewColumnHeadings(CardsOrNotes.NOTES).allColumns) {
val (expectedLabel, expectedValue) =
when (preview.columnType) {
SFLD -> Pair("Sort Field", "Front")
Expand Down Expand Up @@ -1131,10 +1154,11 @@ val CardBrowserViewModel.column2
fun CardBrowserViewModel.setColumn(
index: Int,
column: CardBrowserColumn,
cardsOrNotes: CardsOrNotes = this.cardsOrNotes,
): Boolean {
val newColumns = availableColumns.toMutableList()
newColumns[index] = column
return updateColumns(newColumns)
return updateColumns(newColumns, cardsOrNotes)
}

val Pair<List<ColumnWithSample>, List<ColumnWithSample>>.allColumns
Expand Down

0 comments on commit b01dba5

Please sign in to comment.