diff options
| author | nsfisis <nsfisis@gmail.com> | 2026-05-02 11:24:16 +0900 |
|---|---|---|
| committer | nsfisis <nsfisis@gmail.com> | 2026-05-02 11:24:16 +0900 |
| commit | d9b78a9fa440d84c6cd0c1f2a6ebb43df895ccdf (patch) | |
| tree | b7a287338bfbebf9082f800faeab6bd93b195cb3 /src/shared/card-generator.ts | |
| parent | 7ca9941982a7d7a4c126d215770ce71ad2f7f427 (diff) | |
| download | kioku-d9b78a9fa440d84c6cd0c1f2a6ebb43df895ccdf.tar.gz kioku-d9b78a9fa440d84c6cd0c1f2a6ebb43df895ccdf.tar.zst kioku-d9b78a9fa440d84c6cd0c1f2a6ebb43df895ccdf.zip | |
refactor(notes): extract card generation into shared module
Pull pure card-generation logic (template rendering and the initial FSRS
state for new cards) out of the server note repository into
src/shared/card-generator.ts so both server and client can call it. The
server's note.create / createMany now delegate the rendering work to the
shared function and only insert the resulting payload, preserving the
existing behavior. Groundwork for offline-first note CRUD on the client.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Diffstat (limited to 'src/shared/card-generator.ts')
| -rw-r--r-- | src/shared/card-generator.ts | 101 |
1 files changed, 101 insertions, 0 deletions
diff --git a/src/shared/card-generator.ts b/src/shared/card-generator.ts new file mode 100644 index 0000000..f824e3c --- /dev/null +++ b/src/shared/card-generator.ts @@ -0,0 +1,101 @@ +/** + * Shared card generation logic used by both server (note repository) and client + * (offline-first IndexedDB writes). Pure functions: produces card payloads from + * a note type and field values without touching any storage layer. + */ + +const NEW_CARD_STATE = 0; + +export interface NoteTypeForGeneration { + frontTemplate: string; + backTemplate: string; + isReversible: boolean; +} + +export interface FieldTypeForGeneration { + id: string; + name: string; +} + +export interface FieldValueForGeneration { + noteFieldTypeId: string; + value: string; +} + +export interface GeneratedCard { + isReversed: boolean; + front: string; + back: string; + state: number; + due: Date; + stability: number; + difficulty: number; + elapsedDays: number; + scheduledDays: number; + reps: number; + lapses: number; +} + +export function renderCardTemplate( + template: string, + fields: Map<string, string>, +): string { + let result = template; + for (const [name, value] of fields) { + result = result.replace(new RegExp(`\\{\\{${name}\\}\\}`, "g"), value); + } + return result; +} + +export function generateCardsForNote(input: { + noteType: NoteTypeForGeneration; + fieldTypes: FieldTypeForGeneration[]; + fieldValues: FieldValueForGeneration[]; + now?: Date; +}): GeneratedCard[] { + const { noteType, fieldTypes, fieldValues, now = new Date() } = input; + + const fieldMap = new Map<string, string>(); + for (const fv of fieldValues) { + const ft = fieldTypes.find((f) => f.id === fv.noteFieldTypeId); + if (ft) { + fieldMap.set(ft.name, fv.value); + } + } + + const generated: GeneratedCard[] = [ + buildCard(noteType, fieldMap, false, now), + ]; + if (noteType.isReversible) { + generated.push(buildCard(noteType, fieldMap, true, now)); + } + return generated; +} + +function buildCard( + noteType: NoteTypeForGeneration, + fields: Map<string, string>, + isReversed: boolean, + now: Date, +): GeneratedCard { + const frontTemplate = isReversed + ? noteType.backTemplate + : noteType.frontTemplate; + const backTemplate = isReversed + ? noteType.frontTemplate + : noteType.backTemplate; + + return { + isReversed, + front: renderCardTemplate(frontTemplate, fields), + back: renderCardTemplate(backTemplate, fields), + state: NEW_CARD_STATE, + due: new Date(now), + stability: 0, + difficulty: 0, + elapsedDays: 0, + scheduledDays: 0, + reps: 0, + lapses: 0, + }; +} |
