anki/ts/graphs/reviews.ts

437 lines
14 KiB
TypeScript
Raw Normal View History

2020-06-27 04:35:13 +02:00
// Copyright: Ankitects Pty Ltd and contributors
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
/* eslint
@typescript-eslint/no-non-null-assertion: "off",
@typescript-eslint/no-explicit-any: "off",
*/
import pb from "lib/backend_proto";
import { timeSpan, dayLabel } from "lib/time";
2020-06-27 04:35:13 +02:00
import {
interpolateGreens,
interpolateReds,
interpolateOranges,
2021-01-19 12:22:13 +01:00
interpolatePurples,
select,
pointer,
scaleLinear,
scaleSequential,
axisBottom,
axisLeft,
axisRight,
area,
curveBasis,
min,
histogram,
sum,
max,
cumsum,
} from "d3";
import type { Bin } from "d3";
import type { TableDatum } from "./graph-helpers";
import { GraphBounds, setDataAvailable, GraphRange } from "./graph-helpers";
import { showTooltip, hideTooltip } from "./tooltip";
import * as tr from "lib/i18n";
2020-06-27 04:35:13 +02:00
interface Reviews {
learn: number;
relearn: number;
2021-01-19 12:22:13 +01:00
young: number;
mature: number;
2020-06-27 04:35:13 +02:00
early: number;
}
export interface GraphData {
// indexed by day, where day is relative to today
reviewCount: Map<number, Reviews>;
reviewTime: Map<number, Reviews>;
}
const ReviewKind = pb.BackendProto.RevlogEntry.ReviewKind;
type BinType = Bin<Map<number, Reviews[]>, number>;
export function gatherData(data: pb.BackendProto.GraphsResponse): GraphData {
2020-06-27 04:35:13 +02:00
const reviewCount = new Map<number, Reviews>();
const reviewTime = new Map<number, Reviews>();
const empty = { mature: 0, young: 0, learn: 0, relearn: 0, early: 0 };
for (const review of data.revlog as pb.BackendProto.RevlogEntry[]) {
if (review.reviewKind == ReviewKind.MANUAL) {
// don't count days with only manual scheduling
continue;
}
2020-06-27 04:35:13 +02:00
const day = Math.ceil(
((review.id as number) / 1000 - data.nextDayAtSecs) / 86400
);
const countEntry =
reviewCount.get(day) ?? reviewCount.set(day, { ...empty }).get(day)!;
const timeEntry =
reviewTime.get(day) ?? reviewTime.set(day, { ...empty }).get(day)!;
switch (review.reviewKind) {
2021-01-19 12:22:13 +01:00
case ReviewKind.LEARNING:
countEntry.learn += 1;
timeEntry.learn += review.takenMillis;
break;
case ReviewKind.RELEARNING:
countEntry.relearn += 1;
timeEntry.relearn += review.takenMillis;
break;
2020-06-27 04:35:13 +02:00
case ReviewKind.REVIEW:
if (review.lastInterval < 21) {
2020-06-27 04:35:13 +02:00
countEntry.young += 1;
timeEntry.young += review.takenMillis;
} else {
countEntry.mature += 1;
timeEntry.mature += review.takenMillis;
}
break;
case ReviewKind.EARLY_REVIEW:
countEntry.early += 1;
timeEntry.early += review.takenMillis;
break;
}
}
return { reviewCount, reviewTime };
}
function totalsForBin(bin: BinType): number[] {
const total = [0, 0, 0, 0, 0];
for (const entry of bin) {
2021-01-19 12:22:13 +01:00
total[0] += entry[1].learn;
total[1] += entry[1].relearn;
total[2] += entry[1].young;
total[3] += entry[1].mature;
2020-06-27 04:35:13 +02:00
total[4] += entry[1].early;
}
return total;
}
/// eg idx=0 is mature count, idx=1 is mature+young count, etc
function cumulativeBinValue(bin: BinType, idx: number): number {
return sum(totalsForBin(bin).slice(0, idx + 1));
}
export function renderReviews(
svgElem: SVGElement,
bounds: GraphBounds,
sourceData: GraphData,
range: GraphRange,
showTime: boolean
): TableDatum[] {
2020-07-06 06:01:49 +02:00
const svg = select(svgElem);
const trans = svg.transition().duration(600) as any;
2020-06-28 11:34:19 +02:00
const xMax = 1;
2020-06-27 04:35:13 +02:00
let xMin = 0;
// cap max to selected range
switch (range) {
case GraphRange.Month:
xMin = -30;
2020-06-27 04:35:13 +02:00
break;
case GraphRange.ThreeMonths:
xMin = -89;
2020-06-27 04:35:13 +02:00
break;
case GraphRange.Year:
xMin = -364;
2020-06-27 04:35:13 +02:00
break;
case GraphRange.AllTime:
2020-06-27 04:35:13 +02:00
xMin = min(sourceData.reviewCount.keys())!;
break;
}
const desiredBars = Math.min(70, Math.abs(xMin!));
2020-07-06 11:00:14 +02:00
const x = scaleLinear().domain([xMin!, xMax]).nice(desiredBars);
x.domain([x.domain()[0], xMax]);
2020-06-27 04:35:13 +02:00
const sourceMap = showTime ? sourceData.reviewTime : sourceData.reviewCount;
const bins = histogram()
.value((m) => {
return m[0];
})
.domain(x.domain() as any)
.thresholds(x.ticks(desiredBars))(sourceMap.entries() as any);
2020-07-06 06:01:49 +02:00
// empty graph?
2020-08-04 06:28:46 +02:00
const totalDays = sum(bins, (bin) => bin.length);
if (!totalDays) {
2020-07-06 06:01:49 +02:00
setDataAvailable(svg, false);
2020-08-04 06:28:46 +02:00
return [];
2020-07-06 06:01:49 +02:00
} else {
setDataAvailable(svg, true);
}
2020-06-27 04:35:13 +02:00
x.range([bounds.marginLeft, bounds.width - bounds.marginRight]);
2021-04-02 05:25:38 +02:00
svg.select<SVGGElement>(".x-ticks")
.call((selection) =>
selection.transition(trans).call(axisBottom(x).ticks(7).tickSizeOuter(0))
)
.attr("direction", "ltr");
2020-06-27 04:35:13 +02:00
// y scale
2020-08-05 07:01:51 +02:00
const yTickFormat = (n: number): string => {
if (showTime) {
return timeSpan(n / 1000, true);
2020-08-05 07:01:51 +02:00
} else {
if (Math.round(n) != n) {
return "";
} else {
return n.toLocaleString();
2020-08-05 07:01:51 +02:00
}
}
};
2020-06-27 04:35:13 +02:00
const yMax = max(bins, (b: Bin<any, any>) => cumulativeBinValue(b, 4))!;
const y = scaleLinear()
.range([bounds.height - bounds.marginBottom, bounds.marginTop])
2020-08-05 07:01:51 +02:00
.domain([0, yMax])
.nice();
2021-04-02 05:25:38 +02:00
svg.select<SVGGElement>(".y-ticks")
.call((selection) =>
selection.transition(trans).call(
axisLeft(y)
.ticks(bounds.height / 50)
.tickSizeOuter(0)
.tickFormat(yTickFormat as any)
)
)
2021-04-02 05:25:38 +02:00
.attr("direction", "ltr");
2020-06-27 04:35:13 +02:00
// x bars
2021-01-30 02:08:01 +01:00
function barWidth(d: Bin<number, number>): number {
const width = Math.max(0, x(d.x1!) - x(d.x0!) - 1);
return width ?? 0;
2020-06-27 04:35:13 +02:00
}
2020-08-05 08:06:34 +02:00
const cappedRange = scaleLinear().range([0.3, 0.5]);
2020-06-27 04:35:13 +02:00
const shiftedRange = scaleLinear().range([0.4, 0.7]);
const darkerGreens = scaleSequential((n) =>
interpolateGreens(shiftedRange(n)!)
2020-06-27 04:35:13 +02:00
).domain(x.domain() as any);
const lighterGreens = scaleSequential((n) =>
interpolateGreens(cappedRange(n)!)
2020-06-27 04:35:13 +02:00
).domain(x.domain() as any);
const reds = scaleSequential((n) => interpolateReds(cappedRange(n)!)).domain(
2020-06-27 04:35:13 +02:00
x.domain() as any
);
const oranges = scaleSequential((n) => interpolateOranges(cappedRange(n)!)).domain(
2020-06-27 04:35:13 +02:00
x.domain() as any
);
2021-01-19 12:22:13 +01:00
const purples = scaleSequential((n) => interpolatePurples(cappedRange(n)!)).domain(
x.domain() as any
);
2020-06-27 04:35:13 +02:00
2020-06-28 11:34:19 +02:00
function valueLabel(n: number): string {
if (showTime) {
return timeSpan(n / 1000);
2020-06-28 11:34:19 +02:00
} else {
return tr.statisticsReviews({ reviews: n });
2020-06-28 11:34:19 +02:00
}
}
2020-06-27 07:35:34 +02:00
function tooltipText(d: BinType, cumulative: number): string {
const day = dayLabel(d.x0!, d.x1!);
2020-06-27 07:35:34 +02:00
const totals = totalsForBin(d);
const dayTotal = valueLabel(sum(totals));
2020-07-16 05:25:13 +02:00
let buf = `<table><tr><td>${day}</td><td align=right>${dayTotal}</td></tr>`;
2020-06-27 07:35:34 +02:00
const lines = [
[oranges(1), tr.statisticsCountsLearningCards(), valueLabel(totals[0])],
[reds(1), tr.statisticsCountsRelearningCards(), valueLabel(totals[1])],
[lighterGreens(1), tr.statisticsCountsYoungCards(), valueLabel(totals[2])],
[darkerGreens(1), tr.statisticsCountsMatureCards(), valueLabel(totals[3])],
[purples(1), tr.statisticsCountsEarlyCards(), valueLabel(totals[4])],
["transparent", tr.statisticsRunningTotal(), valueLabel(cumulative)],
2020-06-27 07:35:34 +02:00
];
2020-07-16 05:25:13 +02:00
for (const [colour, label, detail] of lines) {
buf += `<tr>
2020-08-10 07:02:46 +02:00
<td><span style="color: ${colour};"></span> ${label}</td>
2020-07-16 05:25:13 +02:00
<td align=right>${detail}</td>
</tr>`;
2020-06-27 07:35:34 +02:00
}
return buf;
}
2020-06-27 04:35:13 +02:00
const updateBar = (sel: any, idx: number): any => {
return sel
.attr("width", barWidth)
.transition(trans)
.attr("x", (d: any) => x(d.x0))
.attr("y", (d: any) => y(cumulativeBinValue(d, idx))!)
.attr("height", (d: any) => y(0)! - y(cumulativeBinValue(d, idx))!)
2020-06-27 04:35:13 +02:00
.attr("fill", (d: any) => {
switch (idx) {
case 0:
2021-01-19 12:22:13 +01:00
return oranges(d.x0);
2020-06-27 04:35:13 +02:00
case 1:
2021-01-19 12:22:13 +01:00
return reds(d.x0);
2020-06-27 04:35:13 +02:00
case 2:
2021-01-19 12:22:13 +01:00
return lighterGreens(d.x0);
2020-06-27 04:35:13 +02:00
case 3:
2021-01-19 12:22:13 +01:00
return darkerGreens(d.x0);
2020-06-27 04:35:13 +02:00
case 4:
2021-01-19 12:22:13 +01:00
return purples(d.x0);
2020-06-27 04:35:13 +02:00
}
});
};
for (const barNum of [0, 1, 2, 3, 4]) {
svg.select(`g.bars${barNum}`)
.selectAll("rect")
.data(bins)
.join(
(enter) =>
enter
.append("rect")
.attr("rx", 1)
.attr("x", (d: any) => x(d.x0)!)
.attr("y", y(0)!)
2020-06-27 04:35:13 +02:00
.attr("height", 0)
.call((d) => updateBar(d, barNum)),
(update) => update.call((d) => updateBar(d, barNum)),
(remove) =>
remove.call((remove) =>
remove.transition(trans).attr("height", 0).attr("y", y(0)!)
2020-06-27 04:35:13 +02:00
)
);
}
// cumulative area
const areaCounts = bins.map((d: any) => cumulativeBinValue(d, 4));
areaCounts.unshift(0);
const areaData = cumsum(areaCounts);
2020-08-04 06:28:46 +02:00
const yCumMax = areaData.slice(-1)[0];
2020-08-05 07:01:51 +02:00
const yAreaScale = y.copy().domain([0, yCumMax]).nice();
2020-06-27 04:35:13 +02:00
2020-08-04 06:28:46 +02:00
if (yCumMax) {
2021-04-02 05:25:38 +02:00
svg.select<SVGGElement>(".y2-ticks")
.call((selection) =>
selection.transition(trans).call(
axisRight(yAreaScale)
.ticks(bounds.height / 50)
.tickFormat(yTickFormat as any)
.tickSizeOuter(0)
)
)
2021-04-02 05:25:38 +02:00
.attr("direction", "ltr");
2020-08-05 07:01:51 +02:00
svg.select("path.cumulative-overlay")
2020-06-27 04:35:13 +02:00
.datum(areaData as any)
.attr(
"d",
area()
.curve(curveBasis)
2021-01-30 02:08:01 +01:00
.x((_d: [number, number], idx: number) => {
2020-06-27 04:35:13 +02:00
if (idx === 0) {
return x(bins[0].x0!)!;
2020-06-27 04:35:13 +02:00
} else {
return x(bins[idx - 1].x1!)!;
2020-06-27 04:35:13 +02:00
}
})
.y0(bounds.height - bounds.marginBottom)
.y1((d: any) => yAreaScale(d)!) as any
2020-06-27 04:35:13 +02:00
);
}
const hoverData: [Bin<number, number>, number][] = bins.map(
(bin: Bin<number, number>, index: number) => [bin, areaData[index + 1]]
);
2021-01-30 02:08:01 +01:00
// hover/tooltip
svg.select("g.hover-columns")
2020-06-27 04:35:13 +02:00
.selectAll("rect")
2021-01-30 02:08:01 +01:00
.data(hoverData)
2020-06-27 04:35:13 +02:00
.join("rect")
2021-01-30 02:08:01 +01:00
.attr("x", ([bin]) => x(bin.x0!))
.attr("y", () => y(yMax))
.attr("width", ([bin]) => barWidth(bin))
.attr("height", () => y(0) - y(yMax))
.on("mousemove", (event: MouseEvent, [bin, area]): void => {
2021-01-30 02:35:33 +01:00
const [x, y] = pointer(event, document.body);
2021-01-30 02:08:01 +01:00
showTooltip(tooltipText(bin as any, area), x, y);
2020-06-27 04:35:13 +02:00
})
.on("mouseout", hideTooltip);
2020-08-04 06:28:46 +02:00
const periodDays = -xMin + 1;
2020-08-04 06:28:46 +02:00
const studiedDays = sum(bins, (bin) => bin.length);
const total = yCumMax;
const periodAvg = total / periodDays;
const studiedAvg = total / studiedDays;
let totalString: string,
averageForDaysStudied: string,
averageForPeriod: string,
averageAnswerTime: string,
averageAnswerTimeLabel: string;
if (showTime) {
totalString = timeSpan(total / 1000, false);
averageForDaysStudied = tr.statisticsMinutesPerDay({
2020-08-04 06:28:46 +02:00
count: Math.round(studiedAvg / 1000 / 60),
});
averageForPeriod = tr.statisticsMinutesPerDay({
2020-08-04 06:28:46 +02:00
count: Math.round(periodAvg / 1000 / 60),
});
averageAnswerTimeLabel = tr.statisticsAverageAnswerTimeLabel();
2020-08-04 06:28:46 +02:00
// need to get total review count to calculate average time
const countBins = histogram()
.value((m) => {
return m[0];
})
.domain(x.domain() as any)(sourceData.reviewCount.entries() as any);
const totalReviews = sum(countBins, (bin) => cumulativeBinValue(bin as any, 4));
const totalSecs = total / 1000;
const avgSecs = totalSecs / totalReviews;
const cardsPerMin = (totalReviews * 60) / totalSecs;
averageAnswerTime = tr.statisticsAverageAnswerTime({
averageSeconds: avgSecs,
cardsPerMinute: cardsPerMin,
2020-08-04 06:28:46 +02:00
});
} else {
totalString = tr.statisticsReviews({ reviews: total });
averageForDaysStudied = tr.statisticsReviewsPerDay({
2020-08-04 06:28:46 +02:00
count: Math.round(studiedAvg),
});
averageForPeriod = tr.statisticsReviewsPerDay({
2020-08-04 06:28:46 +02:00
count: Math.round(periodAvg),
});
averageAnswerTime = averageAnswerTimeLabel = "";
}
const tableData: TableDatum[] = [
{
label: tr.statisticsDaysStudied(),
value: tr.statisticsAmountOfTotalWithPercentage({
2020-08-04 06:28:46 +02:00
amount: studiedDays,
total: periodDays,
percent: Math.round((studiedDays / periodDays) * 100),
}),
},
2020-08-04 06:28:46 +02:00
{ label: tr.statisticsTotal(), value: totalString },
2020-08-04 06:28:46 +02:00
{
label: tr.statisticsAverageForDaysStudied(),
value: averageForDaysStudied,
},
2020-08-04 06:28:46 +02:00
{
label: tr.statisticsAverageOverPeriod(),
value: averageForPeriod,
},
2020-08-04 06:28:46 +02:00
];
if (averageAnswerTime) {
tableData.push({ label: averageAnswerTimeLabel, value: averageAnswerTime });
2020-08-04 06:28:46 +02:00
}
return tableData;
2020-06-27 04:35:13 +02:00
}