2020-06-26 02:42:10 +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",
|
|
|
|
*/
|
|
|
|
|
2020-11-05 02:01:30 +01:00
|
|
|
import pb from "anki/backend_proto";
|
2021-03-26 11:23:43 +01:00
|
|
|
|
2021-01-30 01:13:47 +01:00
|
|
|
import {
|
|
|
|
interpolateRdYlGn,
|
|
|
|
select,
|
|
|
|
pointer,
|
|
|
|
scaleLinear,
|
|
|
|
scaleBand,
|
|
|
|
scaleSequential,
|
|
|
|
axisBottom,
|
|
|
|
axisLeft,
|
|
|
|
sum,
|
|
|
|
} from "d3";
|
2020-06-26 02:42:10 +02:00
|
|
|
import { showTooltip, hideTooltip } from "./tooltip";
|
2020-07-17 05:46:06 +02:00
|
|
|
import {
|
|
|
|
GraphBounds,
|
|
|
|
setDataAvailable,
|
|
|
|
GraphRange,
|
|
|
|
millisecondCutoffForRange,
|
2020-11-01 05:26:58 +01:00
|
|
|
} from "./graph-helpers";
|
2021-03-26 11:23:43 +01:00
|
|
|
import * as tr from "anki/i18n";
|
2020-06-26 02:42:10 +02:00
|
|
|
|
|
|
|
type ButtonCounts = [number, number, number, number];
|
|
|
|
|
|
|
|
export interface GraphData {
|
|
|
|
learning: ButtonCounts;
|
|
|
|
young: ButtonCounts;
|
|
|
|
mature: ButtonCounts;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ReviewKind = pb.BackendProto.RevlogEntry.ReviewKind;
|
|
|
|
|
2020-07-17 05:46:06 +02:00
|
|
|
export function gatherData(
|
|
|
|
data: pb.BackendProto.GraphsOut,
|
|
|
|
range: GraphRange
|
|
|
|
): GraphData {
|
|
|
|
const cutoff = millisecondCutoffForRange(range, data.nextDayAtSecs);
|
2020-06-26 02:42:10 +02:00
|
|
|
const learning: ButtonCounts = [0, 0, 0, 0];
|
|
|
|
const young: ButtonCounts = [0, 0, 0, 0];
|
|
|
|
const mature: ButtonCounts = [0, 0, 0, 0];
|
|
|
|
|
|
|
|
for (const review of data.revlog as pb.BackendProto.RevlogEntry[]) {
|
2020-07-17 05:46:06 +02:00
|
|
|
if (cutoff && (review.id as number) < cutoff) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-06-26 02:42:10 +02:00
|
|
|
let buttonNum = review.buttonChosen;
|
|
|
|
if (buttonNum <= 0 || buttonNum > 4) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
let buttons = learning;
|
|
|
|
switch (review.reviewKind) {
|
|
|
|
case ReviewKind.LEARNING:
|
|
|
|
case ReviewKind.RELEARNING:
|
|
|
|
// V1 scheduler only had 3 buttons in learning
|
|
|
|
if (buttonNum === 4 && data.schedulerVersion === 1) {
|
|
|
|
buttonNum = 3;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ReviewKind.REVIEW:
|
|
|
|
case ReviewKind.EARLY_REVIEW:
|
|
|
|
if (review.lastInterval < 21) {
|
|
|
|
buttons = young;
|
|
|
|
} else {
|
|
|
|
buttons = mature;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
buttons[buttonNum - 1] += 1;
|
|
|
|
}
|
|
|
|
return { learning, young, mature };
|
|
|
|
}
|
|
|
|
|
2020-07-06 08:21:51 +02:00
|
|
|
type GroupKind = "learning" | "young" | "mature";
|
|
|
|
|
2020-06-26 02:42:10 +02:00
|
|
|
interface Datum {
|
2020-07-06 08:21:51 +02:00
|
|
|
buttonNum: number;
|
|
|
|
group: GroupKind;
|
2020-06-26 02:42:10 +02:00
|
|
|
count: number;
|
|
|
|
}
|
|
|
|
|
2020-07-06 08:21:51 +02:00
|
|
|
interface TotalCorrect {
|
|
|
|
total: number;
|
|
|
|
correct: number;
|
|
|
|
percent: string;
|
|
|
|
}
|
|
|
|
|
2020-06-26 02:42:10 +02:00
|
|
|
export function renderButtons(
|
|
|
|
svgElem: SVGElement,
|
|
|
|
bounds: GraphBounds,
|
2020-07-17 05:46:06 +02:00
|
|
|
origData: pb.BackendProto.GraphsOut,
|
|
|
|
range: GraphRange
|
2020-06-26 02:42:10 +02:00
|
|
|
): void {
|
2020-07-17 05:46:06 +02:00
|
|
|
const sourceData = gatherData(origData, range);
|
2020-06-26 02:42:10 +02:00
|
|
|
const data = [
|
|
|
|
...sourceData.learning.map((count: number, idx: number) => {
|
|
|
|
return {
|
2020-07-06 08:21:51 +02:00
|
|
|
buttonNum: idx + 1,
|
2020-06-26 02:42:10 +02:00
|
|
|
group: "learning",
|
|
|
|
count,
|
|
|
|
} as Datum;
|
|
|
|
}),
|
|
|
|
...sourceData.young.map((count: number, idx: number) => {
|
|
|
|
return {
|
2020-07-06 08:21:51 +02:00
|
|
|
buttonNum: idx + 1,
|
2020-06-26 02:42:10 +02:00
|
|
|
group: "young",
|
|
|
|
count,
|
|
|
|
} as Datum;
|
|
|
|
}),
|
|
|
|
...sourceData.mature.map((count: number, idx: number) => {
|
|
|
|
return {
|
2020-07-06 08:21:51 +02:00
|
|
|
buttonNum: idx + 1,
|
2020-06-26 02:42:10 +02:00
|
|
|
group: "mature",
|
|
|
|
count,
|
|
|
|
} as Datum;
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
2020-07-06 08:21:51 +02:00
|
|
|
const totalCorrect = (kind: GroupKind): TotalCorrect => {
|
|
|
|
const groupData = data.filter((d) => d.group == kind);
|
|
|
|
const total = sum(groupData, (d) => d.count);
|
|
|
|
const correct = sum(
|
|
|
|
groupData.filter((d) => d.buttonNum > 1),
|
|
|
|
(d) => d.count
|
|
|
|
);
|
|
|
|
const percent = total ? ((correct / total) * 100).toFixed(2) : "0";
|
|
|
|
return { total, correct, percent };
|
|
|
|
};
|
|
|
|
|
2020-06-26 02:42:10 +02:00
|
|
|
const yMax = Math.max(...data.map((d) => d.count));
|
|
|
|
|
|
|
|
const svg = select(svgElem);
|
|
|
|
const trans = svg.transition().duration(600) as any;
|
|
|
|
|
2020-07-06 06:01:49 +02:00
|
|
|
if (!yMax) {
|
|
|
|
setDataAvailable(svg, false);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
setDataAvailable(svg, true);
|
|
|
|
}
|
|
|
|
|
2020-06-26 02:42:10 +02:00
|
|
|
const xGroup = scaleBand()
|
|
|
|
.domain(["learning", "young", "mature"])
|
|
|
|
.range([bounds.marginLeft, bounds.width - bounds.marginRight]);
|
2021-03-21 13:47:52 +01:00
|
|
|
svg.select<SVGGElement>(".x-ticks").call((selection) =>
|
|
|
|
selection.transition(trans).call(
|
2020-06-28 12:52:38 +02:00
|
|
|
axisBottom(xGroup)
|
2020-07-06 08:21:51 +02:00
|
|
|
.tickFormat(((d: GroupKind) => {
|
|
|
|
let kind: string;
|
2020-06-28 12:52:38 +02:00
|
|
|
switch (d) {
|
|
|
|
case "learning":
|
2021-03-26 11:23:43 +01:00
|
|
|
kind = tr.statisticsCountsLearningCards();
|
2020-07-06 08:21:51 +02:00
|
|
|
break;
|
2020-06-28 12:52:38 +02:00
|
|
|
case "young":
|
2021-03-26 11:23:43 +01:00
|
|
|
kind = tr.statisticsCountsYoungCards();
|
2020-07-06 08:21:51 +02:00
|
|
|
break;
|
2020-06-28 12:52:38 +02:00
|
|
|
case "mature":
|
|
|
|
default:
|
2021-03-26 11:23:43 +01:00
|
|
|
kind = tr.statisticsCountsMatureCards();
|
2020-07-06 08:21:51 +02:00
|
|
|
break;
|
2020-06-28 12:52:38 +02:00
|
|
|
}
|
2020-07-22 07:15:52 +02:00
|
|
|
return `${kind} \u200e(${totalCorrect(d).percent}%)`;
|
2020-06-28 12:52:38 +02:00
|
|
|
}) as any)
|
|
|
|
.tickSizeOuter(0)
|
2021-03-21 13:47:52 +01:00
|
|
|
)
|
|
|
|
);
|
2020-06-26 02:42:10 +02:00
|
|
|
|
|
|
|
const xButton = scaleBand()
|
|
|
|
.domain(["1", "2", "3", "4"])
|
|
|
|
.range([0, xGroup.bandwidth()])
|
|
|
|
.paddingOuter(1)
|
|
|
|
.paddingInner(0.1);
|
|
|
|
|
|
|
|
const colour = scaleSequential(interpolateRdYlGn).domain([1, 4]);
|
|
|
|
|
|
|
|
// y scale
|
|
|
|
|
|
|
|
const y = scaleLinear()
|
|
|
|
.range([bounds.height - bounds.marginBottom, bounds.marginTop])
|
|
|
|
.domain([0, yMax]);
|
2021-03-21 13:47:52 +01:00
|
|
|
svg.select<SVGGElement>(".y-ticks").call((selection) =>
|
|
|
|
selection.transition(trans).call(
|
2020-06-26 02:42:10 +02:00
|
|
|
axisLeft(y)
|
2020-07-16 03:50:04 +02:00
|
|
|
.ticks(bounds.height / 50)
|
2020-06-26 02:42:10 +02:00
|
|
|
.tickSizeOuter(0)
|
2021-03-21 13:47:52 +01:00
|
|
|
)
|
|
|
|
);
|
2020-06-26 02:42:10 +02:00
|
|
|
|
|
|
|
// x bars
|
|
|
|
|
|
|
|
const updateBar = (sel: any): any => {
|
|
|
|
return sel
|
|
|
|
.attr("width", xButton.bandwidth())
|
2020-08-05 07:56:21 +02:00
|
|
|
.attr("opacity", 0.8)
|
2020-06-26 02:42:10 +02:00
|
|
|
.transition(trans)
|
2020-07-06 08:21:51 +02:00
|
|
|
.attr(
|
|
|
|
"x",
|
|
|
|
(d: Datum) => xGroup(d.group)! + xButton(d.buttonNum.toString())!
|
|
|
|
)
|
2020-06-26 02:42:10 +02:00
|
|
|
.attr("y", (d: Datum) => y(d.count)!)
|
2020-09-29 14:13:25 +02:00
|
|
|
.attr("height", (d: Datum) => y(0)! - y(d.count)!)
|
2020-07-06 08:21:51 +02:00
|
|
|
.attr("fill", (d: Datum) => colour(d.buttonNum));
|
2020-06-26 02:42:10 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
svg.select("g.bars")
|
|
|
|
.selectAll("rect")
|
|
|
|
.data(data)
|
|
|
|
.join(
|
|
|
|
(enter) =>
|
|
|
|
enter
|
|
|
|
.append("rect")
|
|
|
|
.attr("rx", 1)
|
2020-07-06 08:21:51 +02:00
|
|
|
.attr(
|
|
|
|
"x",
|
|
|
|
(d: Datum) =>
|
|
|
|
xGroup(d.group)! + xButton(d.buttonNum.toString())!
|
|
|
|
)
|
2020-09-29 14:13:25 +02:00
|
|
|
.attr("y", y(0)!)
|
2020-06-26 02:42:10 +02:00
|
|
|
.attr("height", 0)
|
|
|
|
.call(updateBar),
|
|
|
|
(update) => update.call(updateBar),
|
|
|
|
(remove) =>
|
|
|
|
remove.call((remove) =>
|
2020-09-29 14:13:25 +02:00
|
|
|
remove.transition(trans).attr("height", 0).attr("y", y(0)!)
|
2020-06-26 02:42:10 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
// hover/tooltip
|
2020-06-28 12:52:38 +02:00
|
|
|
|
|
|
|
function tooltipText(d: Datum): string {
|
2021-03-26 11:23:43 +01:00
|
|
|
const button = tr.statisticsAnswerButtonsButtonNumber();
|
|
|
|
const timesPressed = tr.statisticsAnswerButtonsButtonPressed();
|
|
|
|
const correctStr = tr.statisticsHoursCorrect(totalCorrect(d.group));
|
2020-07-06 08:21:51 +02:00
|
|
|
return `${button}: ${d.buttonNum}<br>${timesPressed}: ${d.count}<br>${correctStr}`;
|
2020-06-28 12:52:38 +02:00
|
|
|
}
|
|
|
|
|
2021-03-21 10:58:39 +01:00
|
|
|
svg.select("g.hover-columns")
|
2020-06-26 02:42:10 +02:00
|
|
|
.selectAll("rect")
|
|
|
|
.data(data)
|
|
|
|
.join("rect")
|
2020-07-06 08:21:51 +02:00
|
|
|
.attr("x", (d: Datum) => xGroup(d.group)! + xButton(d.buttonNum.toString())!)
|
2020-09-29 14:13:25 +02:00
|
|
|
.attr("y", () => y(yMax!)!)
|
2020-06-26 02:42:10 +02:00
|
|
|
.attr("width", xButton.bandwidth())
|
2020-09-29 14:13:25 +02:00
|
|
|
.attr("height", () => y(0)! - y(yMax!)!)
|
2021-01-30 01:13:47 +01:00
|
|
|
.on("mousemove", (event: MouseEvent, d: Datum) => {
|
2021-01-30 02:35:33 +01:00
|
|
|
const [x, y] = pointer(event, document.body);
|
2020-06-26 02:42:10 +02:00
|
|
|
showTooltip(tooltipText(d), x, y);
|
|
|
|
})
|
|
|
|
.on("mouseout", hideTooltip);
|
|
|
|
}
|