feat(vocab): implement SRS features and enhance vocabulary management
All checks were successful
Deploy to production / deploy (push) Successful in 2m49s
All checks were successful
Deploy to production / deploy (push) Successful in 2m49s
- Added new endpoints in vocabController for retrieving SRS due items and reviewing SRS items, improving spaced repetition support. - Updated vocabService to handle SRS item creation and scheduling, ensuring effective tracking of vocabulary exposure. - Enhanced vocabRouter with new routes for SRS functionalities, facilitating user interaction with spaced repetition features. - Modified VocabPracticeDialog and VocabCourseView to integrate SRS due items, providing users with timely review opportunities. - Updated translations and UI elements to reflect new SRS features, enhancing user experience and accessibility.
This commit is contained in:
@@ -40,6 +40,13 @@ class VocabController {
|
|||||||
this.getVocabDistractorPool = this._wrapWithUser((userId, req) =>
|
this.getVocabDistractorPool = this._wrapWithUser((userId, req) =>
|
||||||
this.service.getVocabDistractorPool(userId, req.params.courseId, req.query.beforeLessonId)
|
this.service.getVocabDistractorPool(userId, req.params.courseId, req.query.beforeLessonId)
|
||||||
);
|
);
|
||||||
|
this.getCourseSrsDue = this._wrapWithUser((userId, req) =>
|
||||||
|
this.service.getCourseSrsDue(userId, req.params.courseId, req.query)
|
||||||
|
);
|
||||||
|
this.reviewSrsItem = this._wrapWithUser((userId, req) =>
|
||||||
|
this.service.reviewSrsItem(userId, req.body),
|
||||||
|
{ successStatus: 201 }
|
||||||
|
);
|
||||||
this.getCourseByShareCode = this._wrapWithUser((userId, req) => this.service.getCourseByShareCode(userId, req.body.shareCode));
|
this.getCourseByShareCode = this._wrapWithUser((userId, req) => this.service.getCourseByShareCode(userId, req.body.shareCode));
|
||||||
this.updateCourse = this._wrapWithUser((userId, req) => this.service.updateCourse(userId, req.params.courseId, req.body));
|
this.updateCourse = this._wrapWithUser((userId, req) => this.service.updateCourse(userId, req.params.courseId, req.body));
|
||||||
this.deleteCourse = this._wrapWithUser((userId, req) => this.service.deleteCourse(userId, req.params.courseId));
|
this.deleteCourse = this._wrapWithUser((userId, req) => this.service.deleteCourse(userId, req.params.courseId));
|
||||||
|
|||||||
49
backend/migrations/20260417000000-add-vocab-srs-item.cjs
Normal file
49
backend/migrations/20260417000000-add-vocab-srs-item.cjs
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
async up(queryInterface) {
|
||||||
|
await queryInterface.sequelize.query(`
|
||||||
|
CREATE TABLE IF NOT EXISTS community.vocab_srs_item (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
user_id INTEGER NOT NULL REFERENCES community."user"(id) ON DELETE CASCADE,
|
||||||
|
course_id INTEGER NOT NULL REFERENCES community.vocab_course(id) ON DELETE CASCADE,
|
||||||
|
lesson_id INTEGER NULL REFERENCES community.vocab_course_lesson(id) ON DELETE SET NULL,
|
||||||
|
item_key VARCHAR(80) NOT NULL,
|
||||||
|
learning TEXT NOT NULL,
|
||||||
|
reference TEXT NOT NULL,
|
||||||
|
direction VARCHAR(8) NOT NULL DEFAULT 'BOTH',
|
||||||
|
stage INTEGER NOT NULL DEFAULT 0,
|
||||||
|
interval_days INTEGER NOT NULL DEFAULT 0,
|
||||||
|
last_reviewed_at TIMESTAMP WITH TIME ZONE NULL,
|
||||||
|
next_due_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
|
||||||
|
correct_count INTEGER NOT NULL DEFAULT 0,
|
||||||
|
wrong_count INTEGER NOT NULL DEFAULT 0,
|
||||||
|
lapse_count INTEGER NOT NULL DEFAULT 0,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
|
||||||
|
CONSTRAINT vocab_srs_item_user_key_unique UNIQUE (user_id, item_key)
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_vocab_srs_item_due
|
||||||
|
ON community.vocab_srs_item (user_id, course_id, next_due_at);
|
||||||
|
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_vocab_srs_item_lesson
|
||||||
|
ON community.vocab_srs_item (user_id, course_id, lesson_id);
|
||||||
|
|
||||||
|
COMMENT ON TABLE community.vocab_srs_item IS
|
||||||
|
'Nutzerbezogener SRS-Fortschritt pro Vokabel/Phrase aus Sprachkursen.';
|
||||||
|
COMMENT ON COLUMN community.vocab_srs_item.item_key IS
|
||||||
|
'Stabiler deterministischer Schlüssel aus Kurs, Lektion und normalisiertem Begriffspaar.';
|
||||||
|
COMMENT ON COLUMN community.vocab_srs_item.stage IS
|
||||||
|
'SRS-Stufe. Höhere Stufen bedeuten längere Wiederholungsintervalle.';
|
||||||
|
COMMENT ON COLUMN community.vocab_srs_item.next_due_at IS
|
||||||
|
'Zeitpunkt, zu dem das Item wieder fällig ist.';
|
||||||
|
`);
|
||||||
|
},
|
||||||
|
|
||||||
|
async down(queryInterface) {
|
||||||
|
await queryInterface.sequelize.query(`
|
||||||
|
DROP TABLE IF EXISTS community.vocab_srs_item;
|
||||||
|
`);
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -126,6 +126,7 @@ import VocabCourseProgress from './community/vocab_course_progress.js';
|
|||||||
import VocabGrammarExerciseType from './community/vocab_grammar_exercise_type.js';
|
import VocabGrammarExerciseType from './community/vocab_grammar_exercise_type.js';
|
||||||
import VocabGrammarExercise from './community/vocab_grammar_exercise.js';
|
import VocabGrammarExercise from './community/vocab_grammar_exercise.js';
|
||||||
import VocabGrammarExerciseProgress from './community/vocab_grammar_exercise_progress.js';
|
import VocabGrammarExerciseProgress from './community/vocab_grammar_exercise_progress.js';
|
||||||
|
import VocabSrsItem from './community/vocab_srs_item.js';
|
||||||
import CalendarEvent from './community/calendar_event.js';
|
import CalendarEvent from './community/calendar_event.js';
|
||||||
import Campaign from './match3/campaign.js';
|
import Campaign from './match3/campaign.js';
|
||||||
import Match3Level from './match3/level.js';
|
import Match3Level from './match3/level.js';
|
||||||
@@ -1176,6 +1177,13 @@ export default function setupAssociations() {
|
|||||||
User.hasMany(VocabGrammarExerciseProgress, { foreignKey: 'userId', as: 'grammarExerciseProgress' });
|
User.hasMany(VocabGrammarExerciseProgress, { foreignKey: 'userId', as: 'grammarExerciseProgress' });
|
||||||
VocabGrammarExerciseProgress.belongsTo(VocabGrammarExercise, { foreignKey: 'exerciseId', as: 'exercise' });
|
VocabGrammarExerciseProgress.belongsTo(VocabGrammarExercise, { foreignKey: 'exerciseId', as: 'exercise' });
|
||||||
VocabGrammarExercise.hasMany(VocabGrammarExerciseProgress, { foreignKey: 'exerciseId', as: 'progress' });
|
VocabGrammarExercise.hasMany(VocabGrammarExerciseProgress, { foreignKey: 'exerciseId', as: 'progress' });
|
||||||
|
|
||||||
|
VocabSrsItem.belongsTo(User, { foreignKey: 'userId', as: 'user' });
|
||||||
|
User.hasMany(VocabSrsItem, { foreignKey: 'userId', as: 'vocabSrsItems' });
|
||||||
|
VocabSrsItem.belongsTo(VocabCourse, { foreignKey: 'courseId', as: 'course' });
|
||||||
|
VocabCourse.hasMany(VocabSrsItem, { foreignKey: 'courseId', as: 'srsItems' });
|
||||||
|
VocabSrsItem.belongsTo(VocabCourseLesson, { foreignKey: 'lessonId', as: 'lesson' });
|
||||||
|
VocabCourseLesson.hasMany(VocabSrsItem, { foreignKey: 'lessonId', as: 'srsItems' });
|
||||||
|
|
||||||
// Calendar associations
|
// Calendar associations
|
||||||
CalendarEvent.belongsTo(User, { foreignKey: 'userId', as: 'user' });
|
CalendarEvent.belongsTo(User, { foreignKey: 'userId', as: 'user' });
|
||||||
|
|||||||
94
backend/models/community/vocab_srs_item.js
Normal file
94
backend/models/community/vocab_srs_item.js
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
import { Model, DataTypes } from 'sequelize';
|
||||||
|
import { sequelize } from '../../utils/sequelize.js';
|
||||||
|
|
||||||
|
class VocabSrsItem extends Model {}
|
||||||
|
|
||||||
|
VocabSrsItem.init({
|
||||||
|
id: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
primaryKey: true,
|
||||||
|
autoIncrement: true
|
||||||
|
},
|
||||||
|
userId: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
field: 'user_id'
|
||||||
|
},
|
||||||
|
courseId: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
field: 'course_id'
|
||||||
|
},
|
||||||
|
lessonId: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: true,
|
||||||
|
field: 'lesson_id'
|
||||||
|
},
|
||||||
|
itemKey: {
|
||||||
|
type: DataTypes.STRING(80),
|
||||||
|
allowNull: false,
|
||||||
|
field: 'item_key'
|
||||||
|
},
|
||||||
|
learning: {
|
||||||
|
type: DataTypes.TEXT,
|
||||||
|
allowNull: false
|
||||||
|
},
|
||||||
|
reference: {
|
||||||
|
type: DataTypes.TEXT,
|
||||||
|
allowNull: false
|
||||||
|
},
|
||||||
|
direction: {
|
||||||
|
type: DataTypes.STRING(8),
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 'BOTH'
|
||||||
|
},
|
||||||
|
stage: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 0
|
||||||
|
},
|
||||||
|
intervalDays: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 0,
|
||||||
|
field: 'interval_days'
|
||||||
|
},
|
||||||
|
lastReviewedAt: {
|
||||||
|
type: DataTypes.DATE,
|
||||||
|
allowNull: true,
|
||||||
|
field: 'last_reviewed_at'
|
||||||
|
},
|
||||||
|
nextDueAt: {
|
||||||
|
type: DataTypes.DATE,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: DataTypes.NOW,
|
||||||
|
field: 'next_due_at'
|
||||||
|
},
|
||||||
|
correctCount: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 0,
|
||||||
|
field: 'correct_count'
|
||||||
|
},
|
||||||
|
wrongCount: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 0,
|
||||||
|
field: 'wrong_count'
|
||||||
|
},
|
||||||
|
lapseCount: {
|
||||||
|
type: DataTypes.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 0,
|
||||||
|
field: 'lapse_count'
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
sequelize,
|
||||||
|
modelName: 'VocabSrsItem',
|
||||||
|
tableName: 'vocab_srs_item',
|
||||||
|
schema: 'community',
|
||||||
|
timestamps: true,
|
||||||
|
underscored: true
|
||||||
|
});
|
||||||
|
|
||||||
|
export default VocabSrsItem;
|
||||||
@@ -156,6 +156,7 @@ import VocabCourseProgress from './community/vocab_course_progress.js';
|
|||||||
import VocabGrammarExerciseType from './community/vocab_grammar_exercise_type.js';
|
import VocabGrammarExerciseType from './community/vocab_grammar_exercise_type.js';
|
||||||
import VocabGrammarExercise from './community/vocab_grammar_exercise.js';
|
import VocabGrammarExercise from './community/vocab_grammar_exercise.js';
|
||||||
import VocabGrammarExerciseProgress from './community/vocab_grammar_exercise_progress.js';
|
import VocabGrammarExerciseProgress from './community/vocab_grammar_exercise_progress.js';
|
||||||
|
import VocabSrsItem from './community/vocab_srs_item.js';
|
||||||
import CalendarEvent from './community/calendar_event.js';
|
import CalendarEvent from './community/calendar_event.js';
|
||||||
|
|
||||||
const models = {
|
const models = {
|
||||||
@@ -318,6 +319,7 @@ const models = {
|
|||||||
VocabGrammarExerciseType,
|
VocabGrammarExerciseType,
|
||||||
VocabGrammarExercise,
|
VocabGrammarExercise,
|
||||||
VocabGrammarExerciseProgress,
|
VocabGrammarExerciseProgress,
|
||||||
|
VocabSrsItem,
|
||||||
|
|
||||||
// Calendar
|
// Calendar
|
||||||
CalendarEvent,
|
CalendarEvent,
|
||||||
|
|||||||
@@ -35,9 +35,11 @@ router.post('/courses/find-by-code', vocabController.getCourseByShareCode);
|
|||||||
router.get('/courses/:courseId/completed-lesson-vocabs', vocabController.getCompletedLessonVocabPool);
|
router.get('/courses/:courseId/completed-lesson-vocabs', vocabController.getCompletedLessonVocabPool);
|
||||||
router.get('/courses/:courseId/dictionary', vocabController.getCourseDictionary);
|
router.get('/courses/:courseId/dictionary', vocabController.getCourseDictionary);
|
||||||
router.get('/courses/:courseId/distractor-pool', vocabController.getVocabDistractorPool);
|
router.get('/courses/:courseId/distractor-pool', vocabController.getVocabDistractorPool);
|
||||||
|
router.get('/courses/:courseId/srs/due', vocabController.getCourseSrsDue);
|
||||||
router.get('/courses/:courseId', vocabController.getCourse);
|
router.get('/courses/:courseId', vocabController.getCourse);
|
||||||
router.put('/courses/:courseId', vocabController.updateCourse);
|
router.put('/courses/:courseId', vocabController.updateCourse);
|
||||||
router.delete('/courses/:courseId', vocabController.deleteCourse);
|
router.delete('/courses/:courseId', vocabController.deleteCourse);
|
||||||
|
router.post('/srs/review', vocabController.reviewSrsItem);
|
||||||
|
|
||||||
// Lessons
|
// Lessons
|
||||||
router.post('/courses/:courseId/lessons', vocabController.addLessonToCourse);
|
router.post('/courses/:courseId/lessons', vocabController.addLessonToCourse);
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import VocabCourseProgress from '../models/community/vocab_course_progress.js';
|
|||||||
import VocabGrammarExerciseType from '../models/community/vocab_grammar_exercise_type.js';
|
import VocabGrammarExerciseType from '../models/community/vocab_grammar_exercise_type.js';
|
||||||
import VocabGrammarExercise from '../models/community/vocab_grammar_exercise.js';
|
import VocabGrammarExercise from '../models/community/vocab_grammar_exercise.js';
|
||||||
import VocabGrammarExerciseProgress from '../models/community/vocab_grammar_exercise_progress.js';
|
import VocabGrammarExerciseProgress from '../models/community/vocab_grammar_exercise_progress.js';
|
||||||
|
import VocabSrsItem from '../models/community/vocab_srs_item.js';
|
||||||
import UserParamType from '../models/type/user_param.js';
|
import UserParamType from '../models/type/user_param.js';
|
||||||
import UserParam from '../models/community/user_param.js';
|
import UserParam from '../models/community/user_param.js';
|
||||||
import { sequelize } from '../utils/sequelize.js';
|
import { sequelize } from '../utils/sequelize.js';
|
||||||
@@ -15,6 +16,144 @@ import { Op } from 'sequelize';
|
|||||||
import { BISAYA_PHASE1_DIDACTICS, BISAYA_DIDACTICS_FRAGMENTS } from '../scripts/bisaya-course-phase1.js';
|
import { BISAYA_PHASE1_DIDACTICS, BISAYA_DIDACTICS_FRAGMENTS } from '../scripts/bisaya-course-phase1.js';
|
||||||
|
|
||||||
export default class VocabService {
|
export default class VocabService {
|
||||||
|
_normalizeSrsText(value) {
|
||||||
|
return String(value || '')
|
||||||
|
.trim()
|
||||||
|
.toLowerCase()
|
||||||
|
.normalize('NFKC')
|
||||||
|
.replace(/[\p{P}\p{S}]+/gu, ' ')
|
||||||
|
.replace(/\s+/g, ' ')
|
||||||
|
.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
_buildSrsItemKey({ courseId, lessonId = null, learning, reference, direction = 'BOTH' }) {
|
||||||
|
const raw = [
|
||||||
|
Number(courseId) || 0,
|
||||||
|
lessonId == null ? 'course' : Number(lessonId) || 0,
|
||||||
|
String(direction || 'BOTH').toUpperCase(),
|
||||||
|
this._normalizeSrsText(learning),
|
||||||
|
this._normalizeSrsText(reference)
|
||||||
|
].join('|');
|
||||||
|
return crypto.createHash('sha1').update(raw).digest('hex');
|
||||||
|
}
|
||||||
|
|
||||||
|
_decorateSrsVocabs(vocabs = [], { courseId, lessonId = null } = {}) {
|
||||||
|
return (Array.isArray(vocabs) ? vocabs : [])
|
||||||
|
.map((entry) => {
|
||||||
|
const learning = String(entry?.learning || '').trim();
|
||||||
|
const reference = String(entry?.reference || '').trim();
|
||||||
|
if (!learning || !reference || this._normalizeSrsText(learning) === this._normalizeSrsText(reference)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const direction = String(entry?.direction || 'BOTH').toUpperCase();
|
||||||
|
const itemKey = this._buildSrsItemKey({ courseId, lessonId, learning, reference, direction });
|
||||||
|
return {
|
||||||
|
...entry,
|
||||||
|
id: entry?.id || itemKey,
|
||||||
|
itemKey,
|
||||||
|
courseId: Number(courseId) || null,
|
||||||
|
lessonId: lessonId == null ? null : Number(lessonId),
|
||||||
|
learning,
|
||||||
|
reference,
|
||||||
|
direction
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
_calculateSrsSchedule(item, { correct, rating = null } = {}) {
|
||||||
|
const now = new Date();
|
||||||
|
const previousStage = Math.max(0, Number(item?.stage) || 0);
|
||||||
|
const previousInterval = Math.max(0, Number(item?.intervalDays) || 0);
|
||||||
|
const normalizedRating = String(rating || '').toLowerCase();
|
||||||
|
const isCorrect = Boolean(correct) && normalizedRating !== 'again';
|
||||||
|
|
||||||
|
if (!isCorrect) {
|
||||||
|
return {
|
||||||
|
stage: Math.max(0, previousStage - 1),
|
||||||
|
intervalDays: 0,
|
||||||
|
nextDueAt: new Date(now.getTime() + 10 * 60 * 1000),
|
||||||
|
lapseDelta: 1
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const intervals = [0, 1, 3, 7, 14, 30, 60, 120, 240];
|
||||||
|
let nextStage = Math.min(intervals.length - 1, previousStage + 1);
|
||||||
|
|
||||||
|
if (normalizedRating === 'hard') {
|
||||||
|
nextStage = Math.max(1, previousStage);
|
||||||
|
}
|
||||||
|
if (normalizedRating === 'easy') {
|
||||||
|
nextStage = Math.min(intervals.length - 1, previousStage + 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
let intervalDays = intervals[nextStage] ?? Math.max(1, previousInterval * 2);
|
||||||
|
if (normalizedRating === 'hard') {
|
||||||
|
intervalDays = Math.max(1, Math.ceil(Math.max(previousInterval, 1) * 1.2));
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
stage: nextStage,
|
||||||
|
intervalDays,
|
||||||
|
nextDueAt: new Date(now.getTime() + intervalDays * 24 * 60 * 60 * 1000),
|
||||||
|
lapseDelta: 0
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async _ensureSrsItems(userId, { courseId, lessonId = null, vocabs = [] } = {}) {
|
||||||
|
const decorated = this._decorateSrsVocabs(vocabs, { courseId, lessonId });
|
||||||
|
if (!decorated.length) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const existing = await VocabSrsItem.findAll({
|
||||||
|
where: {
|
||||||
|
userId,
|
||||||
|
itemKey: {
|
||||||
|
[Op.in]: decorated.map((entry) => entry.itemKey)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const existingByKey = new Map(existing.map((entry) => [entry.itemKey, entry]));
|
||||||
|
const now = new Date();
|
||||||
|
|
||||||
|
const createdItems = [];
|
||||||
|
for (const entry of decorated) {
|
||||||
|
if (existingByKey.has(entry.itemKey)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const created = await VocabSrsItem.create({
|
||||||
|
userId,
|
||||||
|
courseId: Number(courseId),
|
||||||
|
lessonId: entry.lessonId,
|
||||||
|
itemKey: entry.itemKey,
|
||||||
|
learning: entry.learning,
|
||||||
|
reference: entry.reference,
|
||||||
|
direction: entry.direction,
|
||||||
|
nextDueAt: now
|
||||||
|
});
|
||||||
|
createdItems.push(created);
|
||||||
|
existingByKey.set(entry.itemKey, created);
|
||||||
|
}
|
||||||
|
|
||||||
|
return decorated.map((entry) => {
|
||||||
|
const item = existingByKey.get(entry.itemKey);
|
||||||
|
return {
|
||||||
|
...entry,
|
||||||
|
srs: item ? {
|
||||||
|
stage: item.stage,
|
||||||
|
intervalDays: item.intervalDays,
|
||||||
|
lastReviewedAt: this._normalizeIsoDate(item.lastReviewedAt),
|
||||||
|
nextDueAt: this._normalizeIsoDate(item.nextDueAt),
|
||||||
|
correctCount: item.correctCount,
|
||||||
|
wrongCount: item.wrongCount,
|
||||||
|
lapseCount: item.lapseCount,
|
||||||
|
isNew: createdItems.some((created) => created.itemKey === entry.itemKey)
|
||||||
|
} : null
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
_normalizeIsoDate(value) {
|
_normalizeIsoDate(value) {
|
||||||
if (!value) {
|
if (!value) {
|
||||||
return '';
|
return '';
|
||||||
@@ -1502,6 +1641,11 @@ export default class VocabService {
|
|||||||
if (!entry?.learning || !entry?.reference) return;
|
if (!entry?.learning || !entry?.reference) return;
|
||||||
mergedVocabs.set(`${entry.learning}-${entry.reference}`, entry);
|
mergedVocabs.set(`${entry.learning}-${entry.reference}`, entry);
|
||||||
});
|
});
|
||||||
|
const vocabs = await this._ensureSrsItems(user.id, {
|
||||||
|
courseId: lesson.courseId,
|
||||||
|
lessonId: lesson.id,
|
||||||
|
vocabs: Array.from(mergedVocabs.values())
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
lesson: {
|
lesson: {
|
||||||
@@ -1510,7 +1654,7 @@ export default class VocabService {
|
|||||||
courseId: lesson.courseId,
|
courseId: lesson.courseId,
|
||||||
courseTitle: lesson.course.title
|
courseTitle: lesson.course.title
|
||||||
},
|
},
|
||||||
vocabs: Array.from(mergedVocabs.values())
|
vocabs
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1612,9 +1756,167 @@ export default class VocabService {
|
|||||||
mergedVocabs.set(`${entry.learning}-${entry.reference}`, entry);
|
mergedVocabs.set(`${entry.learning}-${entry.reference}`, entry);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const vocabs = await this._ensureSrsItems(user.id, {
|
||||||
|
courseId: course.id,
|
||||||
|
lessonId: null,
|
||||||
|
vocabs: Array.from(mergedVocabs.values())
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
courseId: course.id,
|
courseId: course.id,
|
||||||
vocabs: Array.from(mergedVocabs.values())
|
vocabs
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async getCourseSrsDue(hashedUserId, courseId, query = {}) {
|
||||||
|
const user = await this._getUserByHashedId(hashedUserId);
|
||||||
|
const course = await VocabCourse.findByPk(courseId);
|
||||||
|
|
||||||
|
if (!course) {
|
||||||
|
const err = new Error('Course not found');
|
||||||
|
err.status = 404;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (course.ownerUserId !== user.id && !course.isPublic) {
|
||||||
|
const err = new Error('Access denied');
|
||||||
|
err.status = 403;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
const limit = this._clampInteger(query?.limit, { min: 1, max: 100, fallback: 30 });
|
||||||
|
const now = new Date();
|
||||||
|
const rows = await VocabSrsItem.findAll({
|
||||||
|
where: {
|
||||||
|
userId: user.id,
|
||||||
|
courseId: Number(course.id),
|
||||||
|
nextDueAt: {
|
||||||
|
[Op.lte]: now
|
||||||
|
}
|
||||||
|
},
|
||||||
|
order: [
|
||||||
|
['nextDueAt', 'ASC'],
|
||||||
|
['wrongCount', 'DESC'],
|
||||||
|
['stage', 'ASC']
|
||||||
|
],
|
||||||
|
limit
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
courseId: course.id,
|
||||||
|
dueAt: now.toISOString(),
|
||||||
|
count: rows.length,
|
||||||
|
items: rows.map((item) => ({
|
||||||
|
itemKey: item.itemKey,
|
||||||
|
courseId: item.courseId,
|
||||||
|
lessonId: item.lessonId,
|
||||||
|
learning: item.learning,
|
||||||
|
reference: item.reference,
|
||||||
|
direction: item.direction,
|
||||||
|
stage: item.stage,
|
||||||
|
intervalDays: item.intervalDays,
|
||||||
|
lastReviewedAt: this._normalizeIsoDate(item.lastReviewedAt),
|
||||||
|
nextDueAt: this._normalizeIsoDate(item.nextDueAt),
|
||||||
|
correctCount: item.correctCount,
|
||||||
|
wrongCount: item.wrongCount,
|
||||||
|
lapseCount: item.lapseCount
|
||||||
|
}))
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async reviewSrsItem(hashedUserId, payload = {}) {
|
||||||
|
const user = await this._getUserByHashedId(hashedUserId);
|
||||||
|
const courseId = this._clampInteger(payload?.courseId, { min: 1, max: 1_000_000, fallback: 0 });
|
||||||
|
if (!courseId) {
|
||||||
|
const err = new Error('Missing course id');
|
||||||
|
err.status = 400;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
const course = await VocabCourse.findByPk(courseId);
|
||||||
|
if (!course) {
|
||||||
|
const err = new Error('Course not found');
|
||||||
|
err.status = 404;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
if (course.ownerUserId !== user.id && !course.isPublic) {
|
||||||
|
const err = new Error('Access denied');
|
||||||
|
err.status = 403;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
const learning = this._sanitizeShortString(payload?.learning, 1200);
|
||||||
|
const reference = this._sanitizeShortString(payload?.reference, 1200);
|
||||||
|
if (!learning || !reference) {
|
||||||
|
const err = new Error('Missing SRS item text');
|
||||||
|
err.status = 400;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
const lessonId = payload?.lessonId == null
|
||||||
|
? null
|
||||||
|
: this._clampInteger(payload.lessonId, { min: 1, max: 1_000_000, fallback: 0 }) || null;
|
||||||
|
const direction = String(payload?.direction || 'BOTH').toUpperCase().slice(0, 8);
|
||||||
|
const itemKey = this._sanitizeShortString(payload?.itemKey, 80)
|
||||||
|
|| this._buildSrsItemKey({ courseId, lessonId, learning, reference, direction });
|
||||||
|
|
||||||
|
const [item] = await VocabSrsItem.findOrCreate({
|
||||||
|
where: {
|
||||||
|
userId: user.id,
|
||||||
|
itemKey
|
||||||
|
},
|
||||||
|
defaults: {
|
||||||
|
userId: user.id,
|
||||||
|
courseId,
|
||||||
|
lessonId,
|
||||||
|
itemKey,
|
||||||
|
learning,
|
||||||
|
reference,
|
||||||
|
direction,
|
||||||
|
nextDueAt: new Date()
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (
|
||||||
|
item.learning !== learning ||
|
||||||
|
item.reference !== reference ||
|
||||||
|
item.direction !== direction ||
|
||||||
|
item.lessonId !== lessonId
|
||||||
|
) {
|
||||||
|
item.learning = learning;
|
||||||
|
item.reference = reference;
|
||||||
|
item.direction = direction;
|
||||||
|
item.lessonId = lessonId;
|
||||||
|
}
|
||||||
|
|
||||||
|
const correct = Boolean(payload?.correct);
|
||||||
|
const schedule = this._calculateSrsSchedule(item, {
|
||||||
|
correct,
|
||||||
|
rating: payload?.rating
|
||||||
|
});
|
||||||
|
|
||||||
|
item.stage = schedule.stage;
|
||||||
|
item.intervalDays = schedule.intervalDays;
|
||||||
|
item.lastReviewedAt = new Date();
|
||||||
|
item.nextDueAt = schedule.nextDueAt;
|
||||||
|
if (correct) {
|
||||||
|
item.correctCount += 1;
|
||||||
|
} else {
|
||||||
|
item.wrongCount += 1;
|
||||||
|
item.lapseCount += schedule.lapseDelta;
|
||||||
|
}
|
||||||
|
await item.save();
|
||||||
|
|
||||||
|
return {
|
||||||
|
itemKey: item.itemKey,
|
||||||
|
correct,
|
||||||
|
stage: item.stage,
|
||||||
|
intervalDays: item.intervalDays,
|
||||||
|
lastReviewedAt: this._normalizeIsoDate(item.lastReviewedAt),
|
||||||
|
nextDueAt: this._normalizeIsoDate(item.nextDueAt),
|
||||||
|
correctCount: item.correctCount,
|
||||||
|
wrongCount: item.wrongCount,
|
||||||
|
lapseCount: item.lapseCount
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
35
backend/sql/add_vocab_srs_item.sql
Normal file
35
backend/sql/add_vocab_srs_item.sql
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS community.vocab_srs_item (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
user_id INTEGER NOT NULL REFERENCES community."user"(id) ON DELETE CASCADE,
|
||||||
|
course_id INTEGER NOT NULL REFERENCES community.vocab_course(id) ON DELETE CASCADE,
|
||||||
|
lesson_id INTEGER NULL REFERENCES community.vocab_course_lesson(id) ON DELETE SET NULL,
|
||||||
|
item_key VARCHAR(80) NOT NULL,
|
||||||
|
learning TEXT NOT NULL,
|
||||||
|
reference TEXT NOT NULL,
|
||||||
|
direction VARCHAR(8) NOT NULL DEFAULT 'BOTH',
|
||||||
|
stage INTEGER NOT NULL DEFAULT 0,
|
||||||
|
interval_days INTEGER NOT NULL DEFAULT 0,
|
||||||
|
last_reviewed_at TIMESTAMP WITH TIME ZONE NULL,
|
||||||
|
next_due_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
|
||||||
|
correct_count INTEGER NOT NULL DEFAULT 0,
|
||||||
|
wrong_count INTEGER NOT NULL DEFAULT 0,
|
||||||
|
lapse_count INTEGER NOT NULL DEFAULT 0,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
|
||||||
|
CONSTRAINT vocab_srs_item_user_key_unique UNIQUE (user_id, item_key)
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_vocab_srs_item_due
|
||||||
|
ON community.vocab_srs_item (user_id, course_id, next_due_at);
|
||||||
|
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_vocab_srs_item_lesson
|
||||||
|
ON community.vocab_srs_item (user_id, course_id, lesson_id);
|
||||||
|
|
||||||
|
COMMENT ON TABLE community.vocab_srs_item IS
|
||||||
|
'Nutzerbezogener SRS-Fortschritt pro Vokabel/Phrase aus Sprachkursen.';
|
||||||
|
COMMENT ON COLUMN community.vocab_srs_item.item_key IS
|
||||||
|
'Stabiler deterministischer Schlüssel aus Kurs, Lektion und normalisiertem Begriffspaar.';
|
||||||
|
COMMENT ON COLUMN community.vocab_srs_item.stage IS
|
||||||
|
'SRS-Stufe. Höhere Stufen bedeuten längere Wiederholungsintervalle.';
|
||||||
|
COMMENT ON COLUMN community.vocab_srs_item.next_due_at IS
|
||||||
|
'Zeitpunkt, zu dem das Item wieder fällig ist.';
|
||||||
@@ -15,7 +15,7 @@
|
|||||||
<div class="left">
|
<div class="left">
|
||||||
<div class="opts">
|
<div class="opts">
|
||||||
<label class="chk">
|
<label class="chk">
|
||||||
<input type="checkbox" v-model="allVocabs" @change="reloadPool" />
|
<input type="checkbox" v-model="allVocabs" :disabled="srsMode" @change="reloadPool" />
|
||||||
{{ $t('socialnetwork.vocab.practice.allVocabs') }}
|
{{ $t('socialnetwork.vocab.practice.allVocabs') }}
|
||||||
</label>
|
</label>
|
||||||
<label class="chk">
|
<label class="chk">
|
||||||
@@ -116,6 +116,8 @@ export default {
|
|||||||
onClose: null,
|
onClose: null,
|
||||||
loading: false,
|
loading: false,
|
||||||
allVocabs: false,
|
allVocabs: false,
|
||||||
|
srsMode: false,
|
||||||
|
initialPool: null,
|
||||||
simpleMode: false,
|
simpleMode: false,
|
||||||
pool: [],
|
pool: [],
|
||||||
|
|
||||||
@@ -170,13 +172,15 @@ export default {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
open({ languageId, chapterId, lessonId, courseId, onClose = null }) {
|
open({ languageId, chapterId, lessonId, courseId, initialPool = null, srsMode = false, onClose = null }) {
|
||||||
if (this.autoAdvanceTimer) {
|
if (this.autoAdvanceTimer) {
|
||||||
clearTimeout(this.autoAdvanceTimer);
|
clearTimeout(this.autoAdvanceTimer);
|
||||||
this.autoAdvanceTimer = null;
|
this.autoAdvanceTimer = null;
|
||||||
}
|
}
|
||||||
this.openParams = { languageId, chapterId, lessonId, courseId };
|
this.openParams = { languageId, chapterId, lessonId, courseId };
|
||||||
this.onClose = typeof onClose === 'function' ? onClose : null;
|
this.onClose = typeof onClose === 'function' ? onClose : null;
|
||||||
|
this.srsMode = Boolean(srsMode);
|
||||||
|
this.initialPool = Array.isArray(initialPool) ? initialPool : null;
|
||||||
this.allVocabs = false;
|
this.allVocabs = false;
|
||||||
this.simpleMode = false;
|
this.simpleMode = false;
|
||||||
this.correctCount = 0;
|
this.correctCount = 0;
|
||||||
@@ -251,7 +255,7 @@ export default {
|
|||||||
seen.add(key);
|
seen.add(key);
|
||||||
return {
|
return {
|
||||||
...item,
|
...item,
|
||||||
id: item?.id || item?.key || `${key}|${index}`,
|
id: item?.id || item?.itemKey || item?.key || `${key}|${index}`,
|
||||||
learning,
|
learning,
|
||||||
reference
|
reference
|
||||||
};
|
};
|
||||||
@@ -298,6 +302,12 @@ export default {
|
|||||||
},
|
},
|
||||||
async reloadPool() {
|
async reloadPool() {
|
||||||
if (!this.openParams) return;
|
if (!this.openParams) return;
|
||||||
|
if (this.initialPool) {
|
||||||
|
this.loading = false;
|
||||||
|
this.pool = this.normalizePool(this.initialPool);
|
||||||
|
this.next();
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.loading = true;
|
this.loading = true;
|
||||||
try {
|
try {
|
||||||
let res;
|
let res;
|
||||||
@@ -423,11 +433,28 @@ export default {
|
|||||||
// ignore autoplay issues
|
// ignore autoplay issues
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
reportSrsReview(isCorrect) {
|
||||||
|
if (!this.current || !this.openParams?.courseId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
apiClient.post('/api/vocab/srs/review', {
|
||||||
|
courseId: this.openParams.courseId || this.current.courseId,
|
||||||
|
lessonId: this.openParams.lessonId || this.current.lessonId || null,
|
||||||
|
itemKey: this.current.itemKey || null,
|
||||||
|
learning: this.current.learning,
|
||||||
|
reference: this.current.reference,
|
||||||
|
direction: this.direction,
|
||||||
|
correct: Boolean(isCorrect)
|
||||||
|
}).catch((error) => {
|
||||||
|
console.warn('[VocabPracticeDialog] SRS review could not be saved:', error);
|
||||||
|
});
|
||||||
|
},
|
||||||
markResult(isCorrect) {
|
markResult(isCorrect) {
|
||||||
this.answered = true;
|
this.answered = true;
|
||||||
this.lastCorrect = isCorrect;
|
this.lastCorrect = isCorrect;
|
||||||
if (isCorrect) this.correctCount += 1;
|
if (isCorrect) this.correctCount += 1;
|
||||||
else this.wrongCount += 1;
|
else this.wrongCount += 1;
|
||||||
|
this.reportSrsReview(isCorrect);
|
||||||
|
|
||||||
const id = this.current?.id;
|
const id = this.current?.id;
|
||||||
if (!id) return;
|
if (!id) return;
|
||||||
|
|||||||
@@ -726,7 +726,12 @@
|
|||||||
"quickReviewPromptMeaning": "What does \"{term}\" mean?",
|
"quickReviewPromptMeaning": "What does \"{term}\" mean?",
|
||||||
"quickReviewPromptTarget": "Type sa target pinulongan: \"{term}\"",
|
"quickReviewPromptTarget": "Type sa target pinulongan: \"{term}\"",
|
||||||
"quickReviewAcknowledge": "Read, continue",
|
"quickReviewAcknowledge": "Read, continue",
|
||||||
"courseTodayPlanIntroNoDueReview": "Walay angay nga mubo nga balik-balik karon. Makita nimo ang sunod nga makatarunganon nga lakang sa block (limitado sa kabug-aton), unya ang intensive kung naay. Ang mubo nga balik-balik mobalik sa 1/3/7 ka adlaw."
|
"courseTodayPlanIntroNoDueReview": "Walay angay nga mubo nga balik-balik karon. Makita nimo ang sunod nga makatarunganon nga lakang sa block (limitado sa kabug-aton), unya ang intensive kung naay. Ang mubo nga balik-balik mobalik sa 1/3/7 ka adlaw.",
|
||||||
|
"srsDueStat": "SRS angay: {count}",
|
||||||
|
"srsEyebrow": "Dugay nga memorya",
|
||||||
|
"srsTitle": "{count} ka termino ang angay karon",
|
||||||
|
"srsIntro": "Kini nga balik-balik gikan sa SRS nga plano sa matag pulong. Una kini kaysa bag-ong materyal kay nagpalig-on kini sa mga pulong nga hapit malimtan.",
|
||||||
|
"srsStart": "Sugdi ang daily review"
|
||||||
},
|
},
|
||||||
"title": "Trainer sa bokabularyo",
|
"title": "Trainer sa bokabularyo",
|
||||||
"description": "Paghimo og pinulongans (or subscribe aron them) ug share them uban sa friends.",
|
"description": "Paghimo og pinulongans (or subscribe aron them) ug share them uban sa friends.",
|
||||||
|
|||||||
@@ -424,8 +424,7 @@
|
|||||||
"subscribe": "Abonnieren",
|
"subscribe": "Abonnieren",
|
||||||
"subscribeSuccess": "Abo erfolgreich. Menü wird aktualisiert.",
|
"subscribeSuccess": "Abo erfolgreich. Menü wird aktualisiert.",
|
||||||
"subscribeError": "Abo fehlgeschlagen. Code ungültig oder kein Zugriff.",
|
"subscribeError": "Abo fehlgeschlagen. Code ungültig oder kein Zugriff.",
|
||||||
"trainerPlaceholder": "Trainer-Funktionen (Vokabeln/Abfragen) kommen als nächster Schritt."
|
"trainerPlaceholder": "Trainer-Funktionen (Vokabeln/Abfragen) kommen als nächster Schritt.",
|
||||||
,
|
|
||||||
"chapters": "Kapitel",
|
"chapters": "Kapitel",
|
||||||
"newChapter": "Neues Kapitel",
|
"newChapter": "Neues Kapitel",
|
||||||
"createChapter": "Kapitel anlegen",
|
"createChapter": "Kapitel anlegen",
|
||||||
@@ -437,8 +436,7 @@
|
|||||||
"referenceWord": "Referenz",
|
"referenceWord": "Referenz",
|
||||||
"add": "Hinzufügen",
|
"add": "Hinzufügen",
|
||||||
"addVocabError": "Konnte Vokabel nicht hinzufügen.",
|
"addVocabError": "Konnte Vokabel nicht hinzufügen.",
|
||||||
"noVocabs": "In diesem Kapitel sind noch keine Vokabeln."
|
"noVocabs": "In diesem Kapitel sind noch keine Vokabeln.",
|
||||||
,
|
|
||||||
"practice": {
|
"practice": {
|
||||||
"open": "Üben",
|
"open": "Üben",
|
||||||
"title": "Vokabeln üben",
|
"title": "Vokabeln üben",
|
||||||
@@ -810,7 +808,12 @@
|
|||||||
"lessonReviewHintNextDue": "Nächste Fälligkeit: {due}.",
|
"lessonReviewHintNextDue": "Nächste Fälligkeit: {due}.",
|
||||||
"reviewTimeNow": "jetzt",
|
"reviewTimeNow": "jetzt",
|
||||||
"reviewTimeTomorrow": "morgen",
|
"reviewTimeTomorrow": "morgen",
|
||||||
"reviewTimeInDays": "in {count} Tagen"
|
"reviewTimeInDays": "in {count} Tagen",
|
||||||
|
"srsDueStat": "SRS fällig: {count}",
|
||||||
|
"srsEyebrow": "Langzeitgedächtnis",
|
||||||
|
"srsTitle": "{count} Begriffe sind heute fällig",
|
||||||
|
"srsIntro": "Diese Wiederholung kommt aus dem SRS-Plan einzelner Begriffe. Sie hat Vorrang vor neuem Stoff, weil sie kurz vor dem Vergessen stabilisiert.",
|
||||||
|
"srsStart": "Tageswiederholung starten"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -424,8 +424,7 @@
|
|||||||
"subscribe": "Subscribe",
|
"subscribe": "Subscribe",
|
||||||
"subscribeSuccess": "Subscribed. The menu will refresh.",
|
"subscribeSuccess": "Subscribed. The menu will refresh.",
|
||||||
"subscribeError": "Subscribe failed. Invalid code or no access.",
|
"subscribeError": "Subscribe failed. Invalid code or no access.",
|
||||||
"trainerPlaceholder": "Trainer features (words/quizzes) will be the next step."
|
"trainerPlaceholder": "Trainer features (words/quizzes) will be the next step.",
|
||||||
,
|
|
||||||
"chapters": "Chapters",
|
"chapters": "Chapters",
|
||||||
"newChapter": "New chapter",
|
"newChapter": "New chapter",
|
||||||
"createChapter": "Create chapter",
|
"createChapter": "Create chapter",
|
||||||
@@ -437,8 +436,7 @@
|
|||||||
"referenceWord": "Reference",
|
"referenceWord": "Reference",
|
||||||
"add": "Add",
|
"add": "Add",
|
||||||
"addVocabError": "Could not add vocabulary.",
|
"addVocabError": "Could not add vocabulary.",
|
||||||
"noVocabs": "No vocabulary in this chapter yet."
|
"noVocabs": "No vocabulary in this chapter yet.",
|
||||||
,
|
|
||||||
"practice": {
|
"practice": {
|
||||||
"open": "Practice",
|
"open": "Practice",
|
||||||
"title": "Practice vocabulary",
|
"title": "Practice vocabulary",
|
||||||
@@ -810,7 +808,12 @@
|
|||||||
"lessonReviewHintNextDue": "Next due date: {due}.",
|
"lessonReviewHintNextDue": "Next due date: {due}.",
|
||||||
"reviewTimeNow": "now",
|
"reviewTimeNow": "now",
|
||||||
"reviewTimeTomorrow": "tomorrow",
|
"reviewTimeTomorrow": "tomorrow",
|
||||||
"reviewTimeInDays": "in {count} days"
|
"reviewTimeInDays": "in {count} days",
|
||||||
|
"srsDueStat": "SRS due: {count}",
|
||||||
|
"srsEyebrow": "Long-term memory",
|
||||||
|
"srsTitle": "{count} terms are due today",
|
||||||
|
"srsIntro": "This review comes from the SRS schedule of individual terms. It should come before new material because it stabilizes items close to forgetting.",
|
||||||
|
"srsStart": "Start daily review"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -49,11 +49,23 @@
|
|||||||
<p>{{ $t('socialnetwork.vocab.courses.courseFlowIntro') }}</p>
|
<p>{{ $t('socialnetwork.vocab.courses.courseFlowIntro') }}</p>
|
||||||
</div>
|
</div>
|
||||||
<div class="course-flow__stats">
|
<div class="course-flow__stats">
|
||||||
|
<span class="course-flow__stat">{{ $t('socialnetwork.vocab.courses.srsDueStat', { count: srsDueCount }) }}</span>
|
||||||
<span class="course-flow__stat">{{ $t('socialnetwork.vocab.courses.courseFlowReviewStat', { count: dueReviewLessons.length }) }}</span>
|
<span class="course-flow__stat">{{ $t('socialnetwork.vocab.courses.courseFlowReviewStat', { count: dueReviewLessons.length }) }}</span>
|
||||||
<span class="course-flow__stat">{{ $t('socialnetwork.vocab.courses.courseFlowBlockStat', { block: currentBlockNumber || '—' }) }}</span>
|
<span class="course-flow__stat">{{ $t('socialnetwork.vocab.courses.courseFlowBlockStat', { block: currentBlockNumber || '—' }) }}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div v-if="srsDueCount > 0" class="course-srs-plan">
|
||||||
|
<div>
|
||||||
|
<span class="course-srs-plan__eyebrow">{{ $t('socialnetwork.vocab.courses.srsEyebrow') }}</span>
|
||||||
|
<h4>{{ $t('socialnetwork.vocab.courses.srsTitle', { count: srsDueCount }) }}</h4>
|
||||||
|
<p>{{ $t('socialnetwork.vocab.courses.srsIntro') }}</p>
|
||||||
|
</div>
|
||||||
|
<button type="button" class="course-today-plan__action" :disabled="srsLoading" @click="openSrsPractice">
|
||||||
|
{{ srsLoading ? $t('general.loading') : $t('socialnetwork.vocab.courses.srsStart') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div v-if="todayRecommendedSteps.length > 0" class="course-today-plan">
|
<div v-if="todayRecommendedSteps.length > 0" class="course-today-plan">
|
||||||
<h4 class="course-today-plan__title">{{ $t('socialnetwork.vocab.courses.courseTodayPlanTitle') }}</h4>
|
<h4 class="course-today-plan__title">{{ $t('socialnetwork.vocab.courses.courseTodayPlanTitle') }}</h4>
|
||||||
<p class="course-today-plan__intro">
|
<p class="course-today-plan__intro">
|
||||||
@@ -325,6 +337,8 @@ export default {
|
|||||||
course: null,
|
course: null,
|
||||||
progress: [],
|
progress: [],
|
||||||
chapters: [],
|
chapters: [],
|
||||||
|
srsDueItems: [],
|
||||||
|
srsLoading: false,
|
||||||
showAddLessonDialog: false,
|
showAddLessonDialog: false,
|
||||||
assistantSettings: null,
|
assistantSettings: null,
|
||||||
lessonFormTouched: false,
|
lessonFormTouched: false,
|
||||||
@@ -366,6 +380,9 @@ export default {
|
|||||||
currentBlockNumber() {
|
currentBlockNumber() {
|
||||||
return this.currentLesson?.pedagogy?.blockNumber || null;
|
return this.currentLesson?.pedagogy?.blockNumber || null;
|
||||||
},
|
},
|
||||||
|
srsDueCount() {
|
||||||
|
return Array.isArray(this.srsDueItems) ? this.srsDueItems.length : 0;
|
||||||
|
},
|
||||||
dueReviewLessons() {
|
dueReviewLessons() {
|
||||||
return this.sortedLessons
|
return this.sortedLessons
|
||||||
.filter((lesson) => {
|
.filter((lesson) => {
|
||||||
@@ -504,6 +521,7 @@ export default {
|
|||||||
const res = await apiClient.get(`/api/vocab/courses/${this.courseId}`);
|
const res = await apiClient.get(`/api/vocab/courses/${this.courseId}`);
|
||||||
this.course = res.data;
|
this.course = res.data;
|
||||||
await this.loadProgress();
|
await this.loadProgress();
|
||||||
|
await this.loadSrsDueItems();
|
||||||
if (this.course.languageId) {
|
if (this.course.languageId) {
|
||||||
await this.loadChapters();
|
await this.loadChapters();
|
||||||
}
|
}
|
||||||
@@ -522,6 +540,20 @@ export default {
|
|||||||
this.progress = [];
|
this.progress = [];
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
async loadSrsDueItems() {
|
||||||
|
this.srsLoading = true;
|
||||||
|
try {
|
||||||
|
const { data } = await apiClient.get(`/api/vocab/courses/${this.courseId}/srs/due`, {
|
||||||
|
params: { limit: 40 }
|
||||||
|
});
|
||||||
|
this.srsDueItems = Array.isArray(data?.items) ? data.items : [];
|
||||||
|
} catch (e) {
|
||||||
|
console.warn('Konnte SRS-Fälligkeiten nicht laden:', e);
|
||||||
|
this.srsDueItems = [];
|
||||||
|
} finally {
|
||||||
|
this.srsLoading = false;
|
||||||
|
}
|
||||||
|
},
|
||||||
async loadChapters() {
|
async loadChapters() {
|
||||||
try {
|
try {
|
||||||
const res = await apiClient.get(`/api/vocab/languages/${this.course.languageId}/chapters`);
|
const res = await apiClient.get(`/api/vocab/languages/${this.course.languageId}/chapters`);
|
||||||
@@ -814,6 +846,17 @@ export default {
|
|||||||
lessonId: lesson.id
|
lessonId: lesson.id
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
openSrsPractice() {
|
||||||
|
if (!this.srsDueItems.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.$refs.practiceDialog?.open?.({
|
||||||
|
courseId: this.courseId,
|
||||||
|
initialPool: this.srsDueItems,
|
||||||
|
srsMode: true,
|
||||||
|
onClose: () => this.loadSrsDueItems()
|
||||||
|
});
|
||||||
|
},
|
||||||
openLessonReview(lessonId) {
|
openLessonReview(lessonId) {
|
||||||
this.$router.push(`/socialnetwork/vocab/courses/${this.courseId}/lessons/${lessonId}/review`);
|
this.$router.push(`/socialnetwork/vocab/courses/${this.courseId}/lessons/${lessonId}/review`);
|
||||||
},
|
},
|
||||||
@@ -1010,6 +1053,40 @@ export default {
|
|||||||
gap: 14px;
|
gap: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.course-srs-plan {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 16px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
padding: 16px 18px;
|
||||||
|
border-radius: 14px;
|
||||||
|
border: 1px solid rgba(102, 153, 126, 0.38);
|
||||||
|
background: linear-gradient(135deg, rgba(232, 247, 238, 0.95), rgba(255, 251, 240, 0.8));
|
||||||
|
}
|
||||||
|
|
||||||
|
.course-srs-plan__eyebrow {
|
||||||
|
display: inline-flex;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
font-size: 0.72rem;
|
||||||
|
font-weight: 800;
|
||||||
|
letter-spacing: 0.06em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: #4f7b60;
|
||||||
|
}
|
||||||
|
|
||||||
|
.course-srs-plan h4,
|
||||||
|
.course-srs-plan p {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.course-srs-plan p {
|
||||||
|
margin-top: 6px;
|
||||||
|
color: var(--color-text-secondary, #5c534c);
|
||||||
|
font-size: 0.88rem;
|
||||||
|
line-height: 1.45;
|
||||||
|
}
|
||||||
|
|
||||||
.course-flow-card {
|
.course-flow-card {
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
border: 1px solid var(--color-border);
|
border: 1px solid var(--color-border);
|
||||||
|
|||||||
@@ -3602,6 +3602,23 @@ export default {
|
|||||||
normalizeVocab(s) {
|
normalizeVocab(s) {
|
||||||
return this.normalizeComparableText(s);
|
return this.normalizeComparableText(s);
|
||||||
},
|
},
|
||||||
|
reportSrsReviewForCurrentQuestion(isCorrect) {
|
||||||
|
if (!this.currentVocabQuestion?.vocab || !this.courseId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const vocab = this.currentVocabQuestion.vocab;
|
||||||
|
apiClient.post('/api/vocab/srs/review', {
|
||||||
|
courseId: this.courseId,
|
||||||
|
lessonId: vocab.lessonId || this.lessonId,
|
||||||
|
itemKey: vocab.itemKey || null,
|
||||||
|
learning: vocab.learning,
|
||||||
|
reference: vocab.reference,
|
||||||
|
direction: this.vocabTrainerDirection,
|
||||||
|
correct: Boolean(isCorrect)
|
||||||
|
}).catch((error) => {
|
||||||
|
console.warn('[VocabLessonView] SRS review could not be saved:', error);
|
||||||
|
});
|
||||||
|
},
|
||||||
checkVocabAnswer() {
|
checkVocabAnswer() {
|
||||||
if (!this.currentVocabQuestion) return;
|
if (!this.currentVocabQuestion) return;
|
||||||
|
|
||||||
@@ -3638,6 +3655,7 @@ export default {
|
|||||||
stats.wrong++;
|
stats.wrong++;
|
||||||
this.queueFailedVocab(this.currentVocabQuestion.vocab);
|
this.queueFailedVocab(this.currentVocabQuestion.vocab);
|
||||||
}
|
}
|
||||||
|
this.reportSrsReviewForCurrentQuestion(this.vocabTrainerLastCorrect);
|
||||||
|
|
||||||
this.vocabTrainerAnswered = true;
|
this.vocabTrainerAnswered = true;
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user