100 lines
3.1 KiB
Rust
100 lines
3.1 KiB
Rust
// Copyright: Ankitects Pty Ltd and contributors
|
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
use crate::serde::{default_on_invalid, deserialize_int_from_number};
|
|
use crate::{define_newtype, prelude::*};
|
|
use num_enum::TryFromPrimitive;
|
|
use serde::Deserialize;
|
|
use serde_repr::{Deserialize_repr, Serialize_repr};
|
|
use serde_tuple::Serialize_tuple;
|
|
|
|
define_newtype!(RevlogID, i64);
|
|
|
|
#[derive(Serialize_tuple, Deserialize, Debug, Default, PartialEq)]
|
|
pub struct RevlogEntry {
|
|
pub id: TimestampMillis,
|
|
pub cid: CardID,
|
|
pub usn: Usn,
|
|
/// - In the V1 scheduler, 3 represents easy in the learning case.
|
|
/// - 0 represents manual rescheduling.
|
|
#[serde(rename = "ease")]
|
|
pub button_chosen: u8,
|
|
/// Positive values are in days, negative values in seconds.
|
|
#[serde(rename = "ivl", deserialize_with = "deserialize_int_from_number")]
|
|
pub interval: i32,
|
|
/// Positive values are in days, negative values in seconds.
|
|
#[serde(rename = "lastIvl", deserialize_with = "deserialize_int_from_number")]
|
|
pub last_interval: i32,
|
|
/// Card's ease after answering, stored as 10x the %, eg 2500 represents 250%.
|
|
#[serde(rename = "factor", deserialize_with = "deserialize_int_from_number")]
|
|
pub ease_factor: u32,
|
|
/// Amount of milliseconds taken to answer the card.
|
|
#[serde(rename = "time", deserialize_with = "deserialize_int_from_number")]
|
|
pub taken_millis: u32,
|
|
#[serde(rename = "type", default, deserialize_with = "default_on_invalid")]
|
|
pub review_kind: RevlogReviewKind,
|
|
}
|
|
|
|
#[derive(Serialize_repr, Deserialize_repr, Debug, PartialEq, TryFromPrimitive, Clone, Copy)]
|
|
#[repr(u8)]
|
|
pub enum RevlogReviewKind {
|
|
Learning = 0,
|
|
Review = 1,
|
|
Relearning = 2,
|
|
EarlyReview = 3,
|
|
Manual = 4,
|
|
}
|
|
|
|
impl Default for RevlogReviewKind {
|
|
fn default() -> Self {
|
|
RevlogReviewKind::Learning
|
|
}
|
|
}
|
|
|
|
impl RevlogEntry {
|
|
pub(crate) fn interval_secs(&self) -> u32 {
|
|
(if self.interval > 0 {
|
|
self.interval * 86_400
|
|
} else {
|
|
-self.interval
|
|
}) as u32
|
|
}
|
|
}
|
|
|
|
impl Card {
|
|
fn last_interval_for_revlog_todo(&self) -> i32 {
|
|
self.interval as i32
|
|
|
|
// fixme: need to pass in delays for (re)learning
|
|
// if let Some(delay) = self.current_learning_delay_seconds(&[]) {
|
|
// -(delay as i32)
|
|
// } else {
|
|
// self.interval as i32
|
|
// }
|
|
}
|
|
}
|
|
|
|
impl Collection {
|
|
pub(crate) fn log_manually_scheduled_review(
|
|
&mut self,
|
|
card: &Card,
|
|
usn: Usn,
|
|
next_interval: u32,
|
|
) -> Result<()> {
|
|
println!("fixme: learning last_interval");
|
|
// let deck = self.get_deck(card.deck_id)?.ok_or(AnkiError::NotFound)?;
|
|
let entry = RevlogEntry {
|
|
id: TimestampMillis::now(),
|
|
cid: card.id,
|
|
usn,
|
|
button_chosen: 0,
|
|
interval: next_interval as i32,
|
|
last_interval: card.last_interval_for_revlog_todo(),
|
|
ease_factor: card.ease_factor as u32,
|
|
taken_millis: 0,
|
|
review_kind: RevlogReviewKind::Manual,
|
|
};
|
|
self.storage.add_revlog_entry(&entry)
|
|
}
|
|
}
|