ts-fsrs 3.1.0-beta1 → 3.1.0-beta2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -21,7 +21,7 @@ cp .env.local.example .env.local
21
21
  # Example
22
22
 
23
23
  ```typescript
24
- import {createEmptyCard, formatDate, fsrs, generatorParameters, Rating} from 'ts-fsrs';
24
+ import {createEmptyCard, formatDate, fsrs, generatorParameters, Rating, Grades} from 'ts-fsrs';
25
25
 
26
26
  const params = generatorParameters({ enable_fuzz: true });
27
27
  const f = fsrs(params);
@@ -30,28 +30,25 @@ const now = new Date('2022-2-2 10:00:00');// new Date();
30
30
  const scheduling_cards = f.repeat(card, now);
31
31
 
32
32
  // console.log(scheduling_cards);
33
- Object.keys(Rating)
34
- .filter(key => !isNaN(Number(key)))
35
- .map(key => Number(key) as Rating) // [Rating.Again, Rating.Hard, Rating.Good, Rating.Easy]
36
- .forEach(grade => {
37
- const { log, card } = scheduling_cards[grade];
38
- console.group(`${Rating[grade]}`);
39
- console.table({
40
- [`card_${Rating[grade]}`]: {
41
- ...card,
42
- due: formatDate(card.due),
43
- last_review: formatDate(card.last_review as Date),
44
- },
45
- });
46
- console.table({
47
- [`log_${Rating[grade]}`]: {
48
- ...log,
49
- review: formatDate(log.review),
50
- },
51
- });
52
- console.groupEnd();
53
- console.log('----------------------------------------------------------------');
33
+ Grades.forEach(grade => { // [Rating.Again, Rating.Hard, Rating.Good, Rating.Easy]
34
+ const { log, card } = scheduling_cards[grade];
35
+ console.group(`${Rating[grade]}`);
36
+ console.table({
37
+ [`card_${Rating[grade]}`]: {
38
+ ...card,
39
+ due: formatDate(card.due),
40
+ last_review: formatDate(card.last_review as Date),
41
+ },
54
42
  });
43
+ console.table({
44
+ [`log_${Rating[grade]}`]: {
45
+ ...log,
46
+ review: formatDate(log.review),
47
+ },
48
+ });
49
+ console.groupEnd();
50
+ console.log('----------------------------------------------------------------');
51
+ });
55
52
  ```
56
53
 
57
54
  > More examples refer to the [Example](https://github.com/ishiko732/ts-fsrs/blob/master/example/index.ts)
@@ -1,5 +1,5 @@
1
1
  import { SchedulingCard } from "./index";
2
- import { FSRSParameters, Rating } from "./models";
2
+ import { FSRSParameters, Grade } from "./models";
3
3
  import type { int } from "./type";
4
4
  export declare class FSRSAlgorithm {
5
5
  protected param: FSRSParameters;
@@ -23,16 +23,16 @@ export declare class FSRSAlgorithm {
23
23
  * @param g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
24
24
  * @return Stability (interval when R=90%)
25
25
  */
26
- init_stability(g: number): number;
26
+ init_stability(g: Grade): number;
27
27
  /**
28
28
  * The formula used is :
29
29
  * $$D_0(G) = w_4 - (G-3) \cdot w_5$$
30
30
  * $$\min \{\max \{D_0(G),1\},10\}$$
31
31
  * where the D_0(3)=w_4 when the first rating is good.
32
- * @param {number} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
32
+ * @param {Grade} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
33
33
  * @return {number} Difficulty D \in [1,10]
34
34
  */
35
- init_difficulty(g: number): number;
35
+ init_difficulty(g: Grade): number;
36
36
  /**
37
37
  * If fuzzing is disabled or ivl is less than 2.5, it returns the original interval.
38
38
  * @param {number} ivl - The interval to be fuzzed.
@@ -51,10 +51,10 @@ export declare class FSRSAlgorithm {
51
51
  * $$next_d = D - w_6 \cdot (R - 2)$$
52
52
  * $$D^\prime(D,R) = w_5 \cdot D_0(2) +(1 - w_5) \cdot next_d$$
53
53
  * @param {number} d Difficulty D \in [1,10]
54
- * @param {Rating} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
54
+ * @param {Grade} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
55
55
  * @return {number} next_D
56
56
  */
57
- next_difficulty(d: number, g: number): number;
57
+ next_difficulty(d: number, g: Grade): number;
58
58
  /**
59
59
  * The formula used is :
60
60
  * $$\min \{\max \{D_0,1\},10\}$$
@@ -75,10 +75,10 @@ export declare class FSRSAlgorithm {
75
75
  * @param {number} d Difficulty D \in [1,10]
76
76
  * @param {number} s Stability (interval when R=90%)
77
77
  * @param {number} r Retrievability (probability of recall)
78
- * @param {Rating} g Grade (Rating[0.again,1.hard,2.good,3.easy])
78
+ * @param {Grade} g Grade (Rating[0.again,1.hard,2.good,3.easy])
79
79
  * @return {number} S^\prime_r new stability after recall
80
80
  */
81
- next_recall_stability(d: number, s: number, r: number, g: Rating): number;
81
+ next_recall_stability(d: number, s: number, r: number, g: Grade): number;
82
82
  /**
83
83
  * The formula used is :
84
84
  * $$S^\prime_f(D,S,R) = w_11\cdot D^{-w_{12}}\cdot ((S+1)^{w_{13}}-1) \cdot e^{w_{14}\cdot(1-R)}.$$
package/dist/algorithm.js CHANGED
@@ -61,7 +61,7 @@ class FSRSAlgorithm {
61
61
  * $$D_0(G) = w_4 - (G-3) \cdot w_5$$
62
62
  * $$\min \{\max \{D_0(G),1\},10\}$$
63
63
  * where the D_0(3)=w_4 when the first rating is good.
64
- * @param {number} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
64
+ * @param {Grade} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
65
65
  * @return {number} Difficulty D \in [1,10]
66
66
  */
67
67
  init_difficulty(g) {
@@ -97,7 +97,7 @@ class FSRSAlgorithm {
97
97
  * $$next_d = D - w_6 \cdot (R - 2)$$
98
98
  * $$D^\prime(D,R) = w_5 \cdot D_0(2) +(1 - w_5) \cdot next_d$$
99
99
  * @param {number} d Difficulty D \in [1,10]
100
- * @param {Rating} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
100
+ * @param {Grade} g Grade (rating at Anki) [1.again,2.hard,3.good,4.easy]
101
101
  * @return {number} next_D
102
102
  */
103
103
  next_difficulty(d, g) {
@@ -128,7 +128,7 @@ class FSRSAlgorithm {
128
128
  * @param {number} d Difficulty D \in [1,10]
129
129
  * @param {number} s Stability (interval when R=90%)
130
130
  * @param {number} r Retrievability (probability of recall)
131
- * @param {Rating} g Grade (Rating[0.again,1.hard,2.good,3.easy])
131
+ * @param {Grade} g Grade (Rating[0.again,1.hard,2.good,3.easy])
132
132
  * @return {number} S^\prime_r new stability after recall
133
133
  */
134
134
  next_recall_stability(d, s, r, g) {
package/dist/default.js CHANGED
@@ -29,7 +29,7 @@ exports.default_w = exports.envParams.FSRS_W || [
29
29
  0.34, 1.26, 0.29, 2.61,
30
30
  ];
31
31
  exports.default_enable_fuzz = exports.envParams.FSRS_ENABLE_FUZZ || false;
32
- exports.FSRSVersion = "3.1.0-beta1";
32
+ exports.FSRSVersion = "3.1.0-beta2";
33
33
  const generatorParameters = (props) => {
34
34
  return {
35
35
  request_retention: props?.request_retention || exports.default_request_retention,
package/dist/fsrs.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import { Card, CardInput, DateInput, FSRSParameters, RecordLog, ReviewLogInput } from "./models";
1
+ import { Card, CardInput, DateInput, FSRSParameters, RecordLog, RecordLogItem, ReviewLogInput } from "./models";
2
2
  import { FSRSAlgorithm } from "./algorithm";
3
3
  export declare class FSRS extends FSRSAlgorithm {
4
4
  constructor(param: Partial<FSRSParameters>);
@@ -8,4 +8,5 @@ export declare class FSRS extends FSRSAlgorithm {
8
8
  repeat: (card: CardInput, now: DateInput) => RecordLog;
9
9
  get_retrievability: (card: Card, now: Date) => undefined | string;
10
10
  rollback: (card: CardInput, log: ReviewLogInput) => Card;
11
+ forget: (card: CardInput, now: DateInput, reset_count?: boolean) => RecordLogItem;
11
12
  }
package/dist/fsrs.js CHANGED
@@ -80,6 +80,9 @@ class FSRS extends algorithm_1.FSRSAlgorithm {
80
80
  rollback = (card, log) => {
81
81
  card = this.preProcessCard(card);
82
82
  log = this.preProcessLog(log);
83
+ if (log.rating === models_1.Rating.Manual) {
84
+ throw new Error("Cannot rollback a manual rating");
85
+ }
83
86
  let last_due, last_review, last_lapses;
84
87
  switch (log.state) {
85
88
  case models_1.State.New:
@@ -110,5 +113,32 @@ class FSRS extends algorithm_1.FSRSAlgorithm {
110
113
  last_review: last_review,
111
114
  };
112
115
  };
116
+ forget = (card, now, reset_count = false) => {
117
+ card = this.preProcessCard(card);
118
+ now = this.preProcessDate(now);
119
+ const forget_log = {
120
+ rating: models_1.Rating.Manual,
121
+ state: card.state,
122
+ due: card.due,
123
+ stability: card.stability,
124
+ difficulty: card.difficulty,
125
+ elapsed_days: card.elapsed_days,
126
+ scheduled_days: card.scheduled_days,
127
+ review: now,
128
+ };
129
+ const forget_card = {
130
+ ...card,
131
+ due: now,
132
+ stability: 0,
133
+ difficulty: 0,
134
+ elapsed_days: 0,
135
+ scheduled_days: 0,
136
+ reps: reset_count ? 0 : card.reps,
137
+ lapses: reset_count ? 0 : card.lapses,
138
+ state: models_1.State.New,
139
+ last_review: card.last_review,
140
+ };
141
+ return { card: forget_card, log: forget_log };
142
+ };
113
143
  }
114
144
  exports.FSRS = FSRS;
package/dist/help.d.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  import type { int, unit } from "./type";
2
- import { Rating, State } from "./models";
2
+ import { Grade, Rating, State } from "./models";
3
3
  declare global {
4
4
  export interface Date {
5
5
  scheduler(t: int, isDay?: boolean): Date;
@@ -22,3 +22,4 @@ export declare function show_diff_message(due: Date, last_review: Date, unit?: b
22
22
  export declare function fixDate(value: unknown): Date;
23
23
  export declare function fixState(value: unknown): State;
24
24
  export declare function fixRating(value: unknown): Rating;
25
+ export declare const Grades: Grade[];
package/dist/help.js CHANGED
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.fixRating = exports.fixState = exports.fixDate = exports.show_diff_message = exports.formatDate = exports.date_diff = exports.date_scheduler = void 0;
3
+ exports.Grades = exports.fixRating = exports.fixState = exports.fixDate = exports.show_diff_message = exports.formatDate = exports.date_diff = exports.date_scheduler = void 0;
4
4
  const models_1 = require("./models");
5
5
  Date.prototype.scheduler = function (t, isDay) {
6
6
  return date_scheduler(this, t, isDay);
@@ -123,3 +123,9 @@ function fixRating(value) {
123
123
  throw new Error(`Invalid rating:[${value}]`);
124
124
  }
125
125
  exports.fixRating = fixRating;
126
+ exports.Grades = [
127
+ models_1.Rating.Again,
128
+ models_1.Rating.Hard,
129
+ models_1.Rating.Good,
130
+ models_1.Rating.Easy,
131
+ ];
package/dist/index.d.ts CHANGED
@@ -1,7 +1,7 @@
1
1
  export { SchedulingCard } from "./scheduler";
2
2
  export { default_request_retention, default_maximum_interval, default_w, default_enable_fuzz, FSRSVersion, generatorParameters, createEmptyCard, fsrs, envParams, } from "./default";
3
- export { date_scheduler, date_diff, formatDate, show_diff_message, } from "./help";
3
+ export { date_scheduler, date_diff, formatDate, show_diff_message, Grades, } from "./help";
4
4
  export type { int, double } from "./type";
5
- export type { FSRSParameters, Card, ReviewLog, RecordLog, RecordLogItem, StateType, RatingType, CardInput, DateInput, } from "./models";
5
+ export type { FSRSParameters, Card, ReviewLog, RecordLog, RecordLogItem, StateType, RatingType, Grade, CardInput, DateInput, } from "./models";
6
6
  export { State, Rating } from "./models";
7
7
  export { FSRS } from "./fsrs";
package/dist/index.js CHANGED
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.FSRS = exports.Rating = exports.State = exports.show_diff_message = exports.formatDate = exports.date_diff = exports.date_scheduler = exports.envParams = exports.fsrs = exports.createEmptyCard = exports.generatorParameters = exports.FSRSVersion = exports.default_enable_fuzz = exports.default_w = exports.default_maximum_interval = exports.default_request_retention = exports.SchedulingCard = void 0;
3
+ exports.FSRS = exports.Rating = exports.State = exports.Grades = exports.show_diff_message = exports.formatDate = exports.date_diff = exports.date_scheduler = exports.envParams = exports.fsrs = exports.createEmptyCard = exports.generatorParameters = exports.FSRSVersion = exports.default_enable_fuzz = exports.default_w = exports.default_maximum_interval = exports.default_request_retention = exports.SchedulingCard = void 0;
4
4
  var scheduler_1 = require("./scheduler");
5
5
  Object.defineProperty(exports, "SchedulingCard", { enumerable: true, get: function () { return scheduler_1.SchedulingCard; } });
6
6
  var default_1 = require("./default");
@@ -18,6 +18,7 @@ Object.defineProperty(exports, "date_scheduler", { enumerable: true, get: functi
18
18
  Object.defineProperty(exports, "date_diff", { enumerable: true, get: function () { return help_1.date_diff; } });
19
19
  Object.defineProperty(exports, "formatDate", { enumerable: true, get: function () { return help_1.formatDate; } });
20
20
  Object.defineProperty(exports, "show_diff_message", { enumerable: true, get: function () { return help_1.show_diff_message; } });
21
+ Object.defineProperty(exports, "Grades", { enumerable: true, get: function () { return help_1.Grades; } });
21
22
  var models_1 = require("./models");
22
23
  Object.defineProperty(exports, "State", { enumerable: true, get: function () { return models_1.State; } });
23
24
  Object.defineProperty(exports, "Rating", { enumerable: true, get: function () { return models_1.Rating; } });
package/dist/models.d.ts CHANGED
@@ -7,11 +7,14 @@ export declare enum State {
7
7
  }
8
8
  export type RatingType = "Again" | "Hard" | "Good" | "Easy";
9
9
  export declare enum Rating {
10
+ Manual = 0,
10
11
  Again = 1,
11
12
  Hard = 2,
12
13
  Good = 3,
13
14
  Easy = 4
14
15
  }
16
+ type ExcludeManual<T> = Exclude<T, Rating.Manual>;
17
+ export type Grade = ExcludeManual<Rating>;
15
18
  export interface ReviewLog {
16
19
  rating: Rating;
17
20
  state: State;
@@ -27,7 +30,7 @@ export type RecordLogItem = {
27
30
  log: ReviewLog;
28
31
  };
29
32
  export type RecordLog = {
30
- [key in Rating]: RecordLogItem;
33
+ [key in Grade]: RecordLogItem;
31
34
  };
32
35
  export interface Card {
33
36
  due: Date;
@@ -54,3 +57,4 @@ export interface FSRSParameters {
54
57
  w: number[];
55
58
  enable_fuzz: boolean;
56
59
  }
60
+ export {};
package/dist/models.js CHANGED
@@ -10,6 +10,7 @@ var State;
10
10
  })(State || (exports.State = State = {}));
11
11
  var Rating;
12
12
  (function (Rating) {
13
+ Rating[Rating["Manual"] = 0] = "Manual";
13
14
  Rating[Rating["Again"] = 1] = "Again";
14
15
  Rating[Rating["Hard"] = 2] = "Hard";
15
16
  Rating[Rating["Good"] = 3] = "Good";
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "ts-fsrs",
3
- "version": "3.1.0-beta1",
3
+ "version": "3.1.0-beta2",
4
4
  "description": "ts-fsrs is a TypeScript package used to implement the Free Spaced Repetition Scheduler (FSRS) algorithm. It helps developers apply FSRS to their flashcard applications, thereby improving the user learning experience.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",