mirror of
https://github.com/ankitects/anki.git
synced 2025-09-18 14:02:21 -04:00
194 lines
6.6 KiB
Rust
194 lines
6.6 KiB
Rust
// Copyright: Ankitects Pty Ltd and contributors
|
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
use super::timing::SchedTimingToday;
|
|
use crate::{
|
|
backend_proto::{
|
|
bury_or_suspend_cards_request::Mode as BuryOrSuspendMode,
|
|
unbury_deck_request::Mode as UnburyDeckMode,
|
|
},
|
|
card::CardQueue,
|
|
config::SchedulerVersion,
|
|
prelude::*,
|
|
search::{SearchNode, SortMode, StateKind},
|
|
};
|
|
|
|
impl Card {
|
|
/// True if card was buried/suspended prior to the call.
|
|
pub(crate) fn restore_queue_after_bury_or_suspend(&mut self) -> bool {
|
|
if !matches!(
|
|
self.queue,
|
|
CardQueue::Suspended | CardQueue::SchedBuried | CardQueue::UserBuried
|
|
) {
|
|
false
|
|
} else {
|
|
self.restore_queue_from_type();
|
|
true
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Collection {
|
|
pub(crate) fn unbury_if_day_rolled_over(&mut self, timing: SchedTimingToday) -> Result<()> {
|
|
let last_unburied = self.get_last_unburied_day();
|
|
let today = timing.days_elapsed;
|
|
if last_unburied < today || (today + 7) < last_unburied {
|
|
self.unbury_on_day_rollover()?;
|
|
self.set_last_unburied_day(today)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Unbury cards from the previous day.
|
|
/// Done automatically, and does not mark the cards as modified.
|
|
fn unbury_on_day_rollover(&mut self) -> Result<()> {
|
|
self.search_cards_into_table("is:buried", SortMode::NoOrder)?;
|
|
self.storage.for_each_card_in_search(|mut card| {
|
|
card.restore_queue_after_bury_or_suspend();
|
|
self.storage.update_card(&card)
|
|
})?;
|
|
self.storage.clear_searched_cards_table()
|
|
}
|
|
|
|
/// Unsuspend/unbury cards in search table, and clear it.
|
|
/// Marks the cards as modified.
|
|
fn unsuspend_or_unbury_searched_cards(&mut self) -> Result<()> {
|
|
let usn = self.usn()?;
|
|
for original in self.storage.all_searched_cards()? {
|
|
let mut card = original.clone();
|
|
if card.restore_queue_after_bury_or_suspend() {
|
|
self.update_card_inner(&mut card, original, usn)?;
|
|
}
|
|
}
|
|
self.storage.clear_searched_cards_table()
|
|
}
|
|
|
|
pub fn unbury_or_unsuspend_cards(&mut self, cids: &[CardId]) -> Result<OpOutput<()>> {
|
|
self.transact(Op::UnburyUnsuspend, |col| {
|
|
col.storage.set_search_table_to_card_ids(cids, false)?;
|
|
col.unsuspend_or_unbury_searched_cards()
|
|
})
|
|
}
|
|
|
|
pub fn unbury_deck(&mut self, deck_id: DeckId, mode: UnburyDeckMode) -> Result<OpOutput<()>> {
|
|
let state = match mode {
|
|
UnburyDeckMode::All => StateKind::Buried,
|
|
UnburyDeckMode::UserOnly => StateKind::UserBuried,
|
|
UnburyDeckMode::SchedOnly => StateKind::SchedBuried,
|
|
};
|
|
self.transact(Op::UnburyUnsuspend, |col| {
|
|
col.search_cards_into_table(
|
|
match_all![SearchNode::DeckIdWithChildren(deck_id), state],
|
|
SortMode::NoOrder,
|
|
)?;
|
|
col.unsuspend_or_unbury_searched_cards()
|
|
})
|
|
}
|
|
|
|
/// Bury/suspend cards in search table, and clear it.
|
|
/// Marks the cards as modified.
|
|
fn bury_or_suspend_searched_cards(&mut self, mode: BuryOrSuspendMode) -> Result<usize> {
|
|
let mut count = 0;
|
|
let usn = self.usn()?;
|
|
let sched = self.scheduler_version();
|
|
let desired_queue = match mode {
|
|
BuryOrSuspendMode::Suspend => CardQueue::Suspended,
|
|
BuryOrSuspendMode::BurySched => CardQueue::SchedBuried,
|
|
BuryOrSuspendMode::BuryUser => {
|
|
if sched == SchedulerVersion::V1 {
|
|
// v1 scheduler only had one bury type
|
|
CardQueue::SchedBuried
|
|
} else {
|
|
CardQueue::UserBuried
|
|
}
|
|
}
|
|
};
|
|
|
|
for original in self.storage.all_searched_cards()? {
|
|
let mut card = original.clone();
|
|
if card.queue != desired_queue {
|
|
// do not bury suspended cards as that would unsuspend them
|
|
if card.queue != CardQueue::Suspended {
|
|
if sched == SchedulerVersion::V1 {
|
|
card.remove_from_filtered_deck_restoring_queue(sched);
|
|
card.remove_from_learning();
|
|
}
|
|
card.queue = desired_queue;
|
|
count += 1;
|
|
self.update_card_inner(&mut card, original, usn)?;
|
|
}
|
|
}
|
|
}
|
|
|
|
self.storage.clear_searched_cards_table()?;
|
|
|
|
Ok(count)
|
|
}
|
|
|
|
pub fn bury_or_suspend_cards(
|
|
&mut self,
|
|
cids: &[CardId],
|
|
mode: BuryOrSuspendMode,
|
|
) -> Result<OpOutput<usize>> {
|
|
let op = match mode {
|
|
BuryOrSuspendMode::Suspend => Op::Suspend,
|
|
BuryOrSuspendMode::BurySched | BuryOrSuspendMode::BuryUser => Op::Bury,
|
|
};
|
|
self.transact(op, |col| {
|
|
col.storage.set_search_table_to_card_ids(cids, false)?;
|
|
col.bury_or_suspend_searched_cards(mode)
|
|
})
|
|
}
|
|
|
|
pub(crate) fn bury_siblings(
|
|
&mut self,
|
|
cid: CardId,
|
|
nid: NoteId,
|
|
include_new: bool,
|
|
include_reviews: bool,
|
|
) -> Result<usize> {
|
|
self.storage
|
|
.search_siblings_for_bury(cid, nid, include_new, include_reviews)?;
|
|
self.bury_or_suspend_searched_cards(BuryOrSuspendMode::BurySched)
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use crate::{
|
|
card::{Card, CardQueue},
|
|
collection::{open_test_collection, Collection},
|
|
search::{SortMode, StateKind},
|
|
};
|
|
|
|
#[test]
|
|
fn unbury() {
|
|
let mut col = open_test_collection();
|
|
let mut card = Card {
|
|
queue: CardQueue::UserBuried,
|
|
..Default::default()
|
|
};
|
|
col.add_card(&mut card).unwrap();
|
|
let assert_count = |col: &mut Collection, cnt| {
|
|
assert_eq!(
|
|
col.search_cards(StateKind::Buried, SortMode::NoOrder)
|
|
.unwrap()
|
|
.len(),
|
|
cnt
|
|
);
|
|
};
|
|
assert_count(&mut col, 1);
|
|
// day 0, last unburied 0, so no change
|
|
let timing = col.timing_today().unwrap();
|
|
col.unbury_if_day_rolled_over(timing).unwrap();
|
|
assert_count(&mut col, 1);
|
|
// move creation time back and it should succeed
|
|
let mut stamp = col.storage.creation_stamp().unwrap();
|
|
stamp.0 -= 86_400;
|
|
col.set_creation_stamp(stamp).unwrap();
|
|
let timing = col.timing_today().unwrap();
|
|
col.unbury_if_day_rolled_over(timing).unwrap();
|
|
assert_count(&mut col, 0);
|
|
}
|
|
}
|