UNPKG

@taiga-ui/cdk

Version:

Base library for creating Angular components and applications using Taiga UI principles regarding of actual visual appearance

161 lines • 26.1 kB
/// <reference types="@taiga-ui/tsconfig/ng-dev-mode" /> import { CHAR_NO_BREAK_SPACE } from '@taiga-ui/cdk/constants'; import { tuiInRange } from '@taiga-ui/cdk/utils/math'; import { HOURS_IN_DAY, MILLISECONDS_IN_DAY, MILLISECONDS_IN_HOUR, MILLISECONDS_IN_MINUTE, MILLISECONDS_IN_SECOND, MINUTES_IN_HOUR, SECONDS_IN_MINUTE, } from './date-time'; /** * Immutable time object with hours, minutes, seconds and ms */ export class TuiTime { constructor(hours, minutes, seconds = 0, ms = 0) { this.hours = hours; this.minutes = minutes; this.seconds = seconds; this.ms = ms; ngDevMode && console.assert( // Currently `TuiTime` could have hours more than 23 // in order to not break current behaviour of `isValidTime` the logic is duplicated Number.isInteger(hours) && tuiInRange(hours, 0, Infinity) && Number.isInteger(minutes) && tuiInRange(minutes, 0, MINUTES_IN_HOUR) && Number.isInteger(seconds) && tuiInRange(seconds, 0, SECONDS_IN_MINUTE) && Number.isInteger(ms) && tuiInRange(ms, 0, 1000), 'Time must be real, but got:', hours, minutes, seconds, ms); } /** * Checks if time is valid */ static isValidTime(hours, minutes, seconds = 0, ms = 0) { return (Number.isInteger(hours) && tuiInRange(hours, 0, HOURS_IN_DAY) && Number.isInteger(minutes) && tuiInRange(minutes, 0, MINUTES_IN_HOUR) && Number.isInteger(seconds) && tuiInRange(seconds, 0, SECONDS_IN_MINUTE) && Number.isInteger(ms) && tuiInRange(ms, 0, 1000)); } /** * Current UTC time. */ static current() { return TuiTime.fromAbsoluteMilliseconds(Date.now() % MILLISECONDS_IN_DAY); } /** * Current time in local timezone */ static currentLocal() { const date = new Date(); return TuiTime.fromAbsoluteMilliseconds((Date.now() - date.getTimezoneOffset() * MILLISECONDS_IN_MINUTE) % MILLISECONDS_IN_DAY); } /** * Calculates TuiTime from milliseconds */ static fromAbsoluteMilliseconds(milliseconds) { ngDevMode && console.assert(Number.isInteger(milliseconds)); ngDevMode && console.assert(tuiInRange(milliseconds, 0, MILLISECONDS_IN_DAY), `Milliseconds must be below ${MILLISECONDS_IN_DAY} (milliseconds in a day).`); const hours = Math.floor(milliseconds / MILLISECONDS_IN_HOUR); const minutes = Math.floor((milliseconds % MILLISECONDS_IN_HOUR) / MILLISECONDS_IN_MINUTE); const seconds = Math.floor(((milliseconds % MILLISECONDS_IN_HOUR) % MILLISECONDS_IN_MINUTE) / 1000) || 0; const ms = Math.floor(((milliseconds % MILLISECONDS_IN_HOUR) % MILLISECONDS_IN_MINUTE) % 1000) || 0; return new TuiTime(hours, minutes, seconds, ms); } /** * Parses string into TuiTime object */ static fromString(time) { const hours = this.parseHours(time); const minutes = Number(time.slice(3, 5)) || 0; const seconds = Number(time.slice(6, 8)) || 0; const ms = Number(time.slice(9, 12)) || 0; return new TuiTime(hours, minutes, seconds, ms); } /** * Converts Date object into TuiTime * @param date */ static fromLocalNativeDate(date) { return new TuiTime(date.getHours(), date.getMinutes(), date.getSeconds(), date.getMilliseconds()); } static parseMeridiemPeriod(time) { return (/[AP]M/.exec(time.toUpperCase().replaceAll(/\W/g, ''))?.[0] || null); } static parseHours(time) { const hours = Number(time.slice(0, 2)); const meridiem = this.parseMeridiemPeriod(time); if (!meridiem) { return hours; } if (hours === 12) { return meridiem === 'AM' ? 0 : 12; } return meridiem === 'PM' ? hours + 12 : hours; } /** * Shifts time by hours and minutes */ shift({ hours = 0, minutes = 0, seconds = 0, ms = 0 }) { const totalMs = this.toAbsoluteMilliseconds() + hours * MILLISECONDS_IN_HOUR + minutes * MILLISECONDS_IN_MINUTE + seconds * MILLISECONDS_IN_SECOND + ms; const totalSeconds = Math.floor(totalMs / MILLISECONDS_IN_SECOND); const totalMinutes = Math.floor(totalSeconds / SECONDS_IN_MINUTE); const totalHours = Math.floor(totalMinutes / MINUTES_IN_HOUR); return new TuiTime(this.normalizeToRange(totalHours, HOURS_IN_DAY), this.normalizeToRange(totalMinutes, MINUTES_IN_HOUR), this.normalizeToRange(totalSeconds, SECONDS_IN_MINUTE), this.normalizeToRange(totalMs, MILLISECONDS_IN_SECOND)); } /** * Converts TuiTime to string */ toString(mode) { const needAddMs = mode?.startsWith('HH:MM:SS.MSS') || (!mode && this.ms > 0); const needAddSeconds = needAddMs || mode?.startsWith('HH:MM:SS') || (!mode && this.seconds > 0); const { hours = this.hours, meridiem = '' } = mode?.includes('AA') ? this.toTwelveHour(this.hours) : {}; const hhMm = `${this.formatTime(hours)}:${this.formatTime(this.minutes)}`; const ss = needAddSeconds ? `:${this.formatTime(this.seconds)}` : ''; const mss = needAddMs ? `.${this.formatTime(this.ms, 3)}` : ''; const aa = meridiem && `${CHAR_NO_BREAK_SPACE}${meridiem}`; return `${hhMm}${ss}${mss}${aa}`; } valueOf() { return this.toAbsoluteMilliseconds(); } /** * Returns the primitive value of the given Date object. * Depending on the argument, the method can return either a string or a number. * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/@@toPrimitive */ [Symbol.toPrimitive](hint) { return Date.prototype[Symbol.toPrimitive].call(this, hint); } /** * Converts TuiTime to milliseconds */ toAbsoluteMilliseconds() { return (this.hours * MILLISECONDS_IN_HOUR + this.minutes * MILLISECONDS_IN_MINUTE + this.seconds * 1000 + this.ms); } formatTime(time, digits = 2) { return String(time).padStart(digits, '0'); } toTwelveHour(hours) { const meridiem = hours >= 12 ? 'PM' : 'AM'; if (hours === 0 || hours === 12) { return { meridiem, hours: 12 }; } return { meridiem, hours: hours % 12 }; } normalizeToRange(value, modulus) { return ((value % modulus) + modulus) % modulus; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"time.js","sourceRoot":"","sources":["../../../../projects/cdk/date-time/time.ts"],"names":[],"mappings":"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","sourcesContent":["/// <reference types=\"@taiga-ui/tsconfig/ng-dev-mode\" />\n\nimport {CHAR_NO_BREAK_SPACE} from '@taiga-ui/cdk/constants';\nimport {tuiInRange} from '@taiga-ui/cdk/utils/math';\n\nimport {\n    HOURS_IN_DAY,\n    MILLISECONDS_IN_DAY,\n    MILLISECONDS_IN_HOUR,\n    MILLISECONDS_IN_MINUTE,\n    MILLISECONDS_IN_SECOND,\n    MINUTES_IN_HOUR,\n    SECONDS_IN_MINUTE,\n} from './date-time';\nimport type {TuiTimeLike, TuiTimeMode} from './types';\n\n/**\n * Immutable time object with hours, minutes, seconds and ms\n */\nexport class TuiTime implements TuiTimeLike {\n    constructor(\n        public readonly hours: number,\n        public readonly minutes: number,\n        public readonly seconds = 0,\n        public readonly ms = 0,\n    ) {\n        ngDevMode &&\n            console.assert(\n                // Currently `TuiTime` could have hours more than 23\n                // in order to not break current behaviour of `isValidTime` the logic is duplicated\n                Number.isInteger(hours) &&\n                    tuiInRange(hours, 0, Infinity) &&\n                    Number.isInteger(minutes) &&\n                    tuiInRange(minutes, 0, MINUTES_IN_HOUR) &&\n                    Number.isInteger(seconds) &&\n                    tuiInRange(seconds, 0, SECONDS_IN_MINUTE) &&\n                    Number.isInteger(ms) &&\n                    tuiInRange(ms, 0, 1000),\n                'Time must be real, but got:',\n                hours,\n                minutes,\n                seconds,\n                ms,\n            );\n    }\n\n    /**\n     * Checks if time is valid\n     */\n    public static isValidTime(\n        hours: number,\n        minutes: number,\n        seconds = 0,\n        ms = 0,\n    ): boolean {\n        return (\n            Number.isInteger(hours) &&\n            tuiInRange(hours, 0, HOURS_IN_DAY) &&\n            Number.isInteger(minutes) &&\n            tuiInRange(minutes, 0, MINUTES_IN_HOUR) &&\n            Number.isInteger(seconds) &&\n            tuiInRange(seconds, 0, SECONDS_IN_MINUTE) &&\n            Number.isInteger(ms) &&\n            tuiInRange(ms, 0, 1000)\n        );\n    }\n\n    /**\n     * Current UTC time.\n     */\n    public static current(): TuiTime {\n        return TuiTime.fromAbsoluteMilliseconds(Date.now() % MILLISECONDS_IN_DAY);\n    }\n\n    /**\n     * Current time in local timezone\n     */\n    public static currentLocal(): TuiTime {\n        const date = new Date();\n\n        return TuiTime.fromAbsoluteMilliseconds(\n            (Date.now() - date.getTimezoneOffset() * MILLISECONDS_IN_MINUTE) %\n                MILLISECONDS_IN_DAY,\n        );\n    }\n\n    /**\n     * Calculates TuiTime from milliseconds\n     */\n    public static fromAbsoluteMilliseconds(milliseconds: number): TuiTime {\n        ngDevMode && console.assert(Number.isInteger(milliseconds));\n        ngDevMode &&\n            console.assert(\n                tuiInRange(milliseconds, 0, MILLISECONDS_IN_DAY),\n                `Milliseconds must be below ${MILLISECONDS_IN_DAY} (milliseconds in a day).`,\n            );\n\n        const hours = Math.floor(milliseconds / MILLISECONDS_IN_HOUR);\n        const minutes = Math.floor(\n            (milliseconds % MILLISECONDS_IN_HOUR) / MILLISECONDS_IN_MINUTE,\n        );\n        const seconds =\n            Math.floor(\n                ((milliseconds % MILLISECONDS_IN_HOUR) % MILLISECONDS_IN_MINUTE) / 1000,\n            ) || 0;\n        const ms =\n            Math.floor(\n                ((milliseconds % MILLISECONDS_IN_HOUR) % MILLISECONDS_IN_MINUTE) % 1000,\n            ) || 0;\n\n        return new TuiTime(hours, minutes, seconds, ms);\n    }\n\n    /**\n     * Parses string into TuiTime object\n     */\n    public static fromString(time: string): TuiTime {\n        const hours = this.parseHours(time);\n        const minutes = Number(time.slice(3, 5)) || 0;\n        const seconds = Number(time.slice(6, 8)) || 0;\n        const ms = Number(time.slice(9, 12)) || 0;\n\n        return new TuiTime(hours, minutes, seconds, ms);\n    }\n\n    /**\n     * Converts Date object into TuiTime\n     * @param date\n     */\n    public static fromLocalNativeDate(date: Date): TuiTime {\n        return new TuiTime(\n            date.getHours(),\n            date.getMinutes(),\n            date.getSeconds(),\n            date.getMilliseconds(),\n        );\n    }\n\n    private static parseMeridiemPeriod(time: string): 'AM' | 'PM' | null {\n        return (\n            (/[AP]M/.exec(time.toUpperCase().replaceAll(/\\W/g, ''))?.[0] as\n                | 'AM'\n                | 'PM') || null\n        );\n    }\n\n    private static parseHours(time: string): number {\n        const hours = Number(time.slice(0, 2));\n        const meridiem = this.parseMeridiemPeriod(time);\n\n        if (!meridiem) {\n            return hours;\n        }\n\n        if (hours === 12) {\n            return meridiem === 'AM' ? 0 : 12;\n        }\n\n        return meridiem === 'PM' ? hours + 12 : hours;\n    }\n\n    /**\n     * Shifts time by hours and minutes\n     */\n    public shift({hours = 0, minutes = 0, seconds = 0, ms = 0}: TuiTimeLike): TuiTime {\n        const totalMs =\n            this.toAbsoluteMilliseconds() +\n            hours * MILLISECONDS_IN_HOUR +\n            minutes * MILLISECONDS_IN_MINUTE +\n            seconds * MILLISECONDS_IN_SECOND +\n            ms;\n        const totalSeconds = Math.floor(totalMs / MILLISECONDS_IN_SECOND);\n        const totalMinutes = Math.floor(totalSeconds / SECONDS_IN_MINUTE);\n        const totalHours = Math.floor(totalMinutes / MINUTES_IN_HOUR);\n\n        return new TuiTime(\n            this.normalizeToRange(totalHours, HOURS_IN_DAY),\n            this.normalizeToRange(totalMinutes, MINUTES_IN_HOUR),\n            this.normalizeToRange(totalSeconds, SECONDS_IN_MINUTE),\n            this.normalizeToRange(totalMs, MILLISECONDS_IN_SECOND),\n        );\n    }\n\n    /**\n     * Converts TuiTime to string\n     */\n    public toString(\n        mode?: TuiTimeMode, // TODO(v5): remove usage of `TuiTimeMode` and inline all modes as huge union type\n    ): string {\n        const needAddMs = mode?.startsWith('HH:MM:SS.MSS') || (!mode && this.ms > 0);\n        const needAddSeconds =\n            needAddMs || mode?.startsWith('HH:MM:SS') || (!mode && this.seconds > 0);\n        const {hours = this.hours, meridiem = ''} = mode?.includes('AA')\n            ? this.toTwelveHour(this.hours)\n            : {};\n        const hhMm = `${this.formatTime(hours)}:${this.formatTime(this.minutes)}`;\n        const ss = needAddSeconds ? `:${this.formatTime(this.seconds)}` : '';\n        const mss = needAddMs ? `.${this.formatTime(this.ms, 3)}` : '';\n        const aa = meridiem && `${CHAR_NO_BREAK_SPACE}${meridiem}`;\n\n        return `${hhMm}${ss}${mss}${aa}`;\n    }\n\n    public valueOf(): number {\n        return this.toAbsoluteMilliseconds();\n    }\n\n    /**\n     * Returns the primitive value of the given Date object.\n     * Depending on the argument, the method can return either a string or a number.\n     * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/@@toPrimitive\n     */\n    public [Symbol.toPrimitive](hint: string): number | string {\n        return Date.prototype[Symbol.toPrimitive].call(this, hint);\n    }\n\n    /**\n     * Converts TuiTime to milliseconds\n     */\n    public toAbsoluteMilliseconds(): number {\n        return (\n            this.hours * MILLISECONDS_IN_HOUR +\n            this.minutes * MILLISECONDS_IN_MINUTE +\n            this.seconds * 1000 +\n            this.ms\n        );\n    }\n\n    private formatTime(time: number, digits = 2): string {\n        return String(time).padStart(digits, '0');\n    }\n\n    private toTwelveHour(hours: number): {hours: number; meridiem: string} {\n        const meridiem = hours >= 12 ? 'PM' : 'AM';\n\n        if (hours === 0 || hours === 12) {\n            return {meridiem, hours: 12};\n        }\n\n        return {meridiem, hours: hours % 12};\n    }\n\n    private normalizeToRange(value: number, modulus: number): number {\n        return ((value % modulus) + modulus) % modulus;\n    }\n}\n"]}