// Copyright: Ankitects Pty Ltd and contributors // License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html /* eslint @typescript-eslint/no-explicit-any: "off", @typescript-eslint/ban-ts-comment: "off" */ import type { Cards, Stats } from "@tslib/proto"; import type { Bin, Selection } from "d3"; import { sum } from "d3"; // amount of data to fetch from backend export enum RevlogRange { Year = 1, All = 2, } export function daysToRevlogRange(days: number): RevlogRange { return days > 365 || days === 0 ? RevlogRange.All : RevlogRange.Year; } // period a graph should cover export enum GraphRange { Month = 0, ThreeMonths = 1, Year = 2, AllTime = 3, } export interface GraphsContext { cards: Cards.Card[]; revlog: Stats.RevlogEntry[]; revlogRange: RevlogRange; nightMode: boolean; } export interface GraphBounds { width: number; height: number; marginLeft: number; marginRight: number; marginTop: number; marginBottom: number; } export function defaultGraphBounds(): GraphBounds { return { width: 600, height: 250, marginLeft: 70, marginRight: 70, marginTop: 20, marginBottom: 25, }; } export function setDataAvailable( svg: Selection, available: boolean, ): void { svg.select(".no-data") .attr("pointer-events", available ? "none" : "all") .transition() .duration(600) .attr("opacity", available ? 0 : 1); } export function millisecondCutoffForRange( range: GraphRange, nextDayAtSecs: number, ): number { let days: number; switch (range) { case GraphRange.Month: days = 31; break; case GraphRange.ThreeMonths: days = 90; break; case GraphRange.Year: days = 365; break; case GraphRange.AllTime: default: return 0; } return (nextDayAtSecs - 86400 * days) * 1000; } export interface TableDatum { label: string; value: string; } export type SearchEventMap = { search: { query: string } }; export type SearchDispatch = >( type: EventKey, detail: SearchEventMap[EventKey], ) => void; /** Convert a protobuf map that protobufjs represents as an object with string keys into a Map with numeric keys. */ export function numericMap(obj: { [k: string]: T }): Map { return new Map(Object.entries(obj).map(([k, v]) => [Number(k), v])); } export function getNumericMapBinValue(d: Bin, number>): number { return sum(d, (d) => d[1]); }