mirror of
https://github.com/ankitects/anki.git
synced 2025-09-24 08:46:37 -04:00

* Feat/true retention stats * ./ninja fix:minilints * use translatable strings & update style * remove card couts & add more translatable strings * Update statistics.ftl Co-authored-by: user1823 <92206575+user1823@users.noreply.github.com> * add Estimated total knowledge (cards) --------- Co-authored-by: user1823 <92206575+user1823@users.noreply.github.com>
87 lines
4 KiB
TypeScript
87 lines
4 KiB
TypeScript
// Copyright: Ankitects Pty Ltd and contributors
|
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
import type { GraphsResponse } from "@generated/anki/stats_pb";
|
|
import * as tr from "@generated/ftl";
|
|
import { localizedNumber } from "@tslib/i18n";
|
|
import { RevlogRange } from "./graph-helpers";
|
|
|
|
interface TrueRetentionData {
|
|
youngPassed: number;
|
|
youngFailed: number;
|
|
maturePassed: number;
|
|
matureFailed: number;
|
|
}
|
|
|
|
function calculateRetention(passed: number, failed: number): string {
|
|
const total = passed + failed;
|
|
if (total === 0) {
|
|
return "0%";
|
|
}
|
|
return localizedNumber((passed / total) * 100) + "%";
|
|
}
|
|
|
|
function createStatsRow(name: string, data: TrueRetentionData): string {
|
|
const youngRetention = calculateRetention(data.youngPassed, data.youngFailed);
|
|
const matureRetention = calculateRetention(data.maturePassed, data.matureFailed);
|
|
const totalPassed = data.youngPassed + data.maturePassed;
|
|
const totalFailed = data.youngFailed + data.matureFailed;
|
|
const totalRetention = calculateRetention(totalPassed, totalFailed);
|
|
|
|
return `
|
|
<tr>
|
|
<td class="trl">${name}</td>
|
|
<td class="trr">${localizedNumber(data.youngPassed)}</td>
|
|
<td class="trr">${localizedNumber(data.youngFailed)}</td>
|
|
<td class="trr">${youngRetention}</td>
|
|
<td class="trr">${localizedNumber(data.maturePassed)}</td>
|
|
<td class="trr">${localizedNumber(data.matureFailed)}</td>
|
|
<td class="trr">${matureRetention}</td>
|
|
<td class="trr">${localizedNumber(totalPassed)}</td>
|
|
<td class="trr">${localizedNumber(totalFailed)}</td>
|
|
<td class="trr">${totalRetention}</td>
|
|
</tr>`;
|
|
}
|
|
|
|
export function renderTrueRetention(data: GraphsResponse, revlogRange: RevlogRange): string {
|
|
const trueRetention = data.trueRetention!;
|
|
|
|
const tableContent = `
|
|
<style>
|
|
td.trl { border: 1px solid; text-align: left; padding: 5px; }
|
|
td.trr { border: 1px solid; text-align: right; padding: 5px; }
|
|
td.trc { border: 1px solid; text-align: center; padding: 5px; }
|
|
</style>
|
|
<table style="border-collapse: collapse;" cellspacing="0" cellpadding="2">
|
|
<tr>
|
|
<td class="trl" rowspan=3><b>${tr.statisticsTrueRetentionRange()}</b></td>
|
|
<td class="trc" colspan=9><b>${tr.statisticsReviewsTitle()}</b></td>
|
|
</tr>
|
|
<tr>
|
|
<td class="trc" colspan=3><b>${tr.statisticsCountsYoungCards()}</b></td>
|
|
<td class="trc" colspan=3><b>${tr.statisticsCountsMatureCards()}</b></td>
|
|
<td class="trc" colspan=3><b>${tr.statisticsCountsTotalCards()}</b></td>
|
|
</tr>
|
|
<tr>
|
|
<td class="trc">${tr.statisticsTrueRetentionPass()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionFail()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionRetention()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionPass()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionFail()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionRetention()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionPass()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionFail()}</td>
|
|
<td class="trc">${tr.statisticsTrueRetentionRetention()}</td>
|
|
</tr>
|
|
${createStatsRow(tr.statisticsTrueRetentionToday(), trueRetention.today!)}
|
|
${createStatsRow(tr.statisticsTrueRetentionYesterday(), trueRetention.yesterday!)}
|
|
${createStatsRow(tr.statisticsTrueRetentionWeek(), trueRetention.week!)}
|
|
${createStatsRow(tr.statisticsTrueRetentionMonth(), trueRetention.month!)}
|
|
${
|
|
revlogRange === RevlogRange.Year
|
|
? createStatsRow(tr.statisticsTrueRetentionYear(), trueRetention.year!)
|
|
: createStatsRow(tr.statisticsTrueRetentionAllTime(), trueRetention.allTime!)
|
|
}
|
|
</table>`;
|
|
|
|
return tableContent;
|
|
}
|