anki/ts/lib/time.ts
Damien Elmes 15dcb09036
Detect incorrect usage of triple slash in TypeScript (#2524)
* Migrate check_copyright to Rust

* Add a new lint to check accidental usages of /// in ts/svelte comments

* Fix a bunch of incorrect jdoc comments

* Move contributor check into minilints

Will allow users to detect the issue locally with './ninja check'
before pushing to CI.

* Make Cargo.toml consistent with other crates
2023-05-26 12:49:44 +10:00

204 lines
5.9 KiB
TypeScript

// Copyright: Ankitects Pty Ltd and contributors
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
import * as tr from "@tslib/ftl";
export const SECOND = 1.0;
export const MINUTE = 60.0 * SECOND;
export const HOUR = 60.0 * MINUTE;
export const DAY = 24.0 * HOUR;
export const MONTH = 30.0 * DAY;
export const YEAR = 12.0 * MONTH;
export enum TimespanUnit {
Seconds,
Minutes,
Hours,
Days,
Months,
Years,
}
export function unitName(unit: TimespanUnit): string {
switch (unit) {
case TimespanUnit.Seconds:
return "seconds";
case TimespanUnit.Minutes:
return "minutes";
case TimespanUnit.Hours:
return "hours";
case TimespanUnit.Days:
return "days";
case TimespanUnit.Months:
return "months";
case TimespanUnit.Years:
return "years";
}
}
export function naturalUnit(secs: number): TimespanUnit {
secs = Math.abs(secs);
if (secs < MINUTE) {
return TimespanUnit.Seconds;
} else if (secs < HOUR) {
return TimespanUnit.Minutes;
} else if (secs < DAY) {
return TimespanUnit.Hours;
} else if (secs < MONTH) {
return TimespanUnit.Days;
} else if (secs < YEAR) {
return TimespanUnit.Months;
} else {
return TimespanUnit.Years;
}
}
/** Number of seconds in a given unit. */
export function unitSeconds(unit: TimespanUnit): number {
switch (unit) {
case TimespanUnit.Seconds:
return SECOND;
case TimespanUnit.Minutes:
return MINUTE;
case TimespanUnit.Hours:
return HOUR;
case TimespanUnit.Days:
return DAY;
case TimespanUnit.Months:
return MONTH;
case TimespanUnit.Years:
return YEAR;
}
}
export function unitAmount(unit: TimespanUnit, secs: number): number {
return secs / unitSeconds(unit);
}
/** Largest unit provided seconds can be divided by without a remainder. */
export function naturalWholeUnit(secs: number): TimespanUnit {
let unit = naturalUnit(secs);
while (unit != TimespanUnit.Seconds) {
const amount = Math.round(unitAmount(unit, secs));
if (Math.abs(secs - amount * unitSeconds(unit)) < Number.EPSILON) {
return unit;
}
unit -= 1;
}
return unit;
}
export function studiedToday(cards: number, secs: number): string {
const unit = naturalUnit(secs);
const amount = unitAmount(unit, secs);
const name = unitName(unit);
let secsPer = 0;
if (cards > 0) {
secsPer = secs / cards;
}
return tr.statisticsStudiedToday({
unit: name,
secsPerCard: secsPer,
cards,
amount,
});
}
function i18nFuncForUnit(
unit: TimespanUnit,
short: boolean,
): (_: { amount: number }) => string {
if (short) {
switch (unit) {
case TimespanUnit.Seconds:
return tr.statisticsElapsedTimeSeconds;
case TimespanUnit.Minutes:
return tr.statisticsElapsedTimeMinutes;
case TimespanUnit.Hours:
return tr.statisticsElapsedTimeHours;
case TimespanUnit.Days:
return tr.statisticsElapsedTimeDays;
case TimespanUnit.Months:
return tr.statisticsElapsedTimeMonths;
case TimespanUnit.Years:
return tr.statisticsElapsedTimeYears;
}
} else {
switch (unit) {
case TimespanUnit.Seconds:
return tr.schedulingTimeSpanSeconds;
case TimespanUnit.Minutes:
return tr.schedulingTimeSpanMinutes;
case TimespanUnit.Hours:
return tr.schedulingTimeSpanHours;
case TimespanUnit.Days:
return tr.schedulingTimeSpanDays;
case TimespanUnit.Months:
return tr.schedulingTimeSpanMonths;
case TimespanUnit.Years:
return tr.schedulingTimeSpanYears;
}
}
}
/** Describe the given seconds using the largest appropriate unit.
If precise is true, show to two decimal places, eg
eg 70 seconds -> "1.17 minutes"
If false, seconds and days are shown without decimals. */
export function timeSpan(seconds: number, short = false): string {
const unit = naturalUnit(seconds);
const amount = unitAmount(unit, seconds);
return i18nFuncForUnit(unit, short)({ amount });
}
export function dayLabel(daysStart: number, daysEnd: number): string {
const larger = Math.max(Math.abs(daysStart), Math.abs(daysEnd));
const smaller = Math.min(Math.abs(daysStart), Math.abs(daysEnd));
if (larger - smaller <= 1) {
// singular
if (daysStart >= 0) {
return tr.statisticsInDaysSingle({ days: daysStart });
} else {
return tr.statisticsDaysAgoSingle({ days: -daysStart });
}
} else {
// range
if (daysStart >= 0) {
return tr.statisticsInDaysRange({
daysStart,
daysEnd: daysEnd - 1,
});
} else {
return tr.statisticsDaysAgoRange({
daysStart: Math.abs(daysEnd - 1),
daysEnd: -daysStart,
});
}
}
}
/** Helper for converting Unix timestamps to date strings. */
export class Timestamp {
private date: Date;
constructor(seconds: number) {
this.date = new Date(seconds * 1000);
}
/** YYYY-MM-DD */
dateString(): string {
const year = this.date.getFullYear();
const month = ("0" + (this.date.getMonth() + 1)).slice(-2);
const date = ("0" + this.date.getDate()).slice(-2);
return `${year}-${month}-${date}`;
}
/** HH:MM */
timeString(): string {
const hours = ("0" + this.date.getHours()).slice(-2);
const minutes = ("0" + this.date.getMinutes()).slice(-2);
return `${hours}:${minutes}`;
}
}