2020-01-02 15:13:47 -05:00
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
2019-04-28 01:07:11 +02:00
2020-08-15 16:37:17 +02:00
import { DateTimeFormatter } from "./formatter.ts" ;
2019-01-11 14:19:42 +09:00
2020-06-27 15:40:45 +04:30
export const SECOND = 1 e3 ;
export const MINUTE = SECOND * 60 ;
export const HOUR = MINUTE * 60 ;
export const DAY = HOUR * 24 ;
export const WEEK = DAY * 7 ;
2020-07-14 13:23:54 -05:00
const DAYS_PER_WEEK = 7 ;
enum Day {
Sun ,
Mon ,
Tue ,
Wed ,
Thu ,
Fri ,
Sat ,
}
2020-06-27 15:40:45 +04:30
2019-01-03 18:19:20 +03:00
/ * *
* Parse date from string using format string
2020-08-15 16:37:17 +02:00
* @param dateString Date string
2019-03-14 15:24:54 +01:00
* @param format Format string
* @return Parsed date
2019-01-03 18:19:20 +03:00
* /
2020-08-15 16:37:17 +02:00
export function parse ( dateString : string , formatString : string ) : Date {
const formatter = new DateTimeFormatter ( formatString ) ;
const parts = formatter . parseToParts ( dateString ) ;
2020-12-12 21:21:48 +08:00
const sortParts = formatter . sortDateTimeFormatPart ( parts ) ;
return formatter . partsToDate ( sortParts ) ;
2019-01-03 18:19:20 +03:00
}
/ * *
2020-08-15 16:37:17 +02:00
* Format date using format string
* @param date Date
2019-03-14 15:24:54 +01:00
* @param format Format string
2020-08-15 16:37:17 +02:00
* @return formatted date string
2019-01-03 18:19:20 +03:00
* /
2020-08-15 16:37:17 +02:00
export function format ( date : Date , formatString : string ) : string {
const formatter = new DateTimeFormatter ( formatString ) ;
return formatter . format ( date ) ;
2019-01-03 18:19:20 +03:00
}
2019-03-11 16:00:30 +01:00
/ * *
* Get number of the day in the year
2019-03-14 15:24:54 +01:00
* @return Number of the day in year
2019-03-11 16:00:30 +01:00
* /
2019-03-14 15:24:54 +01:00
export function dayOfYear ( date : Date ) : number {
2020-08-27 11:12:49 +02:00
// Values from 0 to 99 map to the years 1900 to 1999. All other values are the actual year. (https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/Date)
// Using setFullYear as a workaround
2020-09-15 16:09:40 +02:00
2020-08-27 11:12:49 +02:00
const yearStart = new Date ( date ) ;
2020-09-15 16:09:40 +02:00
yearStart . setUTCFullYear ( date . getUTCFullYear ( ) , 0 , 0 ) ;
2020-07-14 15:24:17 -04:00
const diff = date . getTime ( ) -
2019-03-11 16:00:30 +01:00
yearStart . getTime ( ) +
( yearStart . getTimezoneOffset ( ) - date . getTimezoneOffset ( ) ) * 60 * 1000 ;
2020-09-15 16:09:40 +02:00
2020-06-27 15:40:45 +04:30
return Math . floor ( diff / DAY ) ;
2019-03-11 16:00:30 +01:00
}
2020-07-09 20:50:42 +01:00
/ * *
* Get number of the week in the year ( ISO - 8601 )
* @return Number of the week in year
* /
export function weekOfYear ( date : Date ) : number {
const workingDate = new Date (
2020-07-14 15:24:17 -04:00
Date . UTC ( date . getFullYear ( ) , date . getMonth ( ) , date . getDate ( ) ) ,
2020-07-09 20:50:42 +01:00
) ;
2020-07-14 13:23:54 -05:00
const day = workingDate . getUTCDay ( ) ;
2020-07-14 15:24:17 -04:00
const nearestThursday = workingDate . getUTCDate ( ) +
2020-07-14 13:23:54 -05:00
Day . Thu -
( day === Day . Sun ? DAYS_PER_WEEK : day ) ;
workingDate . setUTCDate ( nearestThursday ) ;
2020-07-09 20:50:42 +01:00
// Get first day of year
const yearStart = new Date ( Date . UTC ( workingDate . getUTCFullYear ( ) , 0 , 1 ) ) ;
// return the calculated full weeks to nearest Thursday
2020-07-14 13:23:54 -05:00
return Math . ceil ( ( workingDate . getTime ( ) - yearStart . getTime ( ) + DAY ) / WEEK ) ;
2019-01-03 18:19:20 +03:00
}
2019-04-28 01:07:11 +02:00
/ * *
2020-12-11 02:45:45 +07:00
* Parse a date to return a IMF formatted string date
2019-04-28 01:07:11 +02:00
* RFC : https : //tools.ietf.org/html/rfc7231#section-7.1.1.1
* IMF is the time format to use when generating times in HTTP
* headers . The time being formatted must be in UTC for Format to
* generate the correct format .
* @param date Date to parse
2020-12-11 02:45:45 +07:00
* @return IMF date formatted string
2019-04-28 01:07:11 +02:00
* /
export function toIMF ( date : Date ) : string {
2019-10-06 01:02:34 +09:00
function dtPad ( v : string , lPad = 2 ) : string {
2020-03-31 12:34:13 +02:00
return v . padStart ( lPad , "0" ) ;
2019-04-28 01:07:11 +02:00
}
const d = dtPad ( date . getUTCDate ( ) . toString ( ) ) ;
const h = dtPad ( date . getUTCHours ( ) . toString ( ) ) ;
const min = dtPad ( date . getUTCMinutes ( ) . toString ( ) ) ;
const s = dtPad ( date . getUTCSeconds ( ) . toString ( ) ) ;
const y = date . getUTCFullYear ( ) ;
2019-11-16 20:24:07 +07:00
const days = [ "Sun" , "Mon" , "Tue" , "Wed" , "Thu" , "Fri" , "Sat" ] ;
2019-04-28 01:07:11 +02:00
const months = [
"Jan" ,
"Feb" ,
"Mar" ,
2019-11-16 20:24:07 +07:00
"Apr" ,
2019-04-28 01:07:11 +02:00
"May" ,
"Jun" ,
"Jul" ,
"Aug" ,
"Sep" ,
"Oct" ,
"Nov" ,
2020-03-29 04:03:49 +11:00
"Dec" ,
2019-04-28 01:07:11 +02:00
] ;
2019-05-13 20:03:24 +02:00
return ` ${ days [ date . getUTCDay ( ) ] } , ${ d } ${
2019-04-28 01:07:11 +02:00
months [ date . getUTCMonth ( ) ]
} $ { y } $ { h } : $ { min } : $ { s } GMT ` ;
}
2020-06-27 15:40:45 +04:30
/ * *
* Check given year is a leap year or not .
* based on : https : //docs.microsoft.com/en-us/office/troubleshoot/excel/determine-a-leap-year
* @param year year in number or Date format
* /
export function isLeap ( year : Date | number ) : boolean {
const yearNumber = year instanceof Date ? year . getFullYear ( ) : year ;
return (
( yearNumber % 4 === 0 && yearNumber % 100 !== 0 ) || yearNumber % 400 === 0
) ;
}
export type Unit =
2020-09-15 11:48:49 +09:00
| "milliseconds"
2020-06-27 15:40:45 +04:30
| "seconds"
| "minutes"
| "hours"
| "days"
| "weeks"
| "months"
| "quarters"
| "years" ;
export type DifferenceFormat = Partial < Record < Unit , number > > ;
export type DifferenceOptions = {
units? : Unit [ ] ;
} ;
/ * *
* Calculate difference between two dates .
* @param from Year to calculate difference
* @param to Year to calculate difference with
* @param options Options for determining how to respond
*
* example :
*
* ` ` ` typescript
* datetime . difference ( new Date ( "2020/1/1" ) , new Date ( "2020/2/2" ) , { units : [ "days" , "months" ] } )
* ` ` `
* /
export function difference (
from : Date ,
to : Date ,
2020-07-14 15:24:17 -04:00
options? : DifferenceOptions ,
2020-06-27 15:40:45 +04:30
) : DifferenceFormat {
2020-07-14 15:24:17 -04:00
const uniqueUnits = options ? . units ? [ . . . new Set ( options ? . units ) ] : [
2020-09-15 11:48:49 +09:00
"milliseconds" ,
2020-07-14 15:24:17 -04:00
"seconds" ,
"minutes" ,
"hours" ,
"days" ,
"weeks" ,
"months" ,
"quarters" ,
"years" ,
] ;
2020-06-27 15:40:45 +04:30
const bigger = Math . max ( from . getTime ( ) , to . getTime ( ) ) ;
const smaller = Math . min ( from . getTime ( ) , to . getTime ( ) ) ;
const differenceInMs = bigger - smaller ;
const differences : DifferenceFormat = { } ;
for ( const uniqueUnit of uniqueUnits ) {
switch ( uniqueUnit ) {
2020-09-15 11:48:49 +09:00
case "milliseconds" :
differences . milliseconds = differenceInMs ;
2020-06-27 15:40:45 +04:30
break ;
case "seconds" :
differences . seconds = Math . floor ( differenceInMs / SECOND ) ;
break ;
case "minutes" :
differences . minutes = Math . floor ( differenceInMs / MINUTE ) ;
break ;
case "hours" :
differences . hours = Math . floor ( differenceInMs / HOUR ) ;
break ;
case "days" :
differences . days = Math . floor ( differenceInMs / DAY ) ;
break ;
case "weeks" :
differences . weeks = Math . floor ( differenceInMs / WEEK ) ;
break ;
case "months" :
differences . months = calculateMonthsDifference ( bigger , smaller ) ;
break ;
case "quarters" :
differences . quarters = Math . floor (
( typeof differences . months !== "undefined" &&
differences . months / 4 ) ||
2020-07-14 15:24:17 -04:00
calculateMonthsDifference ( bigger , smaller ) / 4 ,
2020-06-27 15:40:45 +04:30
) ;
break ;
case "years" :
differences . years = Math . floor (
( typeof differences . months !== "undefined" &&
differences . months / 12 ) ||
2020-07-14 15:24:17 -04:00
calculateMonthsDifference ( bigger , smaller ) / 12 ,
2020-06-27 15:40:45 +04:30
) ;
break ;
}
}
return differences ;
}
function calculateMonthsDifference ( bigger : number , smaller : number ) : number {
const biggerDate = new Date ( bigger ) ;
const smallerDate = new Date ( smaller ) ;
const yearsDiff = biggerDate . getFullYear ( ) - smallerDate . getFullYear ( ) ;
const monthsDiff = biggerDate . getMonth ( ) - smallerDate . getMonth ( ) ;
2020-12-11 02:45:45 +07:00
const calendarDifferences = Math . abs ( yearsDiff * 12 + monthsDiff ) ;
2020-06-27 15:40:45 +04:30
const compareResult = biggerDate > smallerDate ? 1 : - 1 ;
biggerDate . setMonth (
2020-12-11 02:45:45 +07:00
biggerDate . getMonth ( ) - compareResult * calendarDifferences ,
2020-06-27 15:40:45 +04:30
) ;
2020-07-14 15:24:17 -04:00
const isLastMonthNotFull = biggerDate > smallerDate
? 1
: - 1 === - compareResult
? 1
: 0 ;
2020-12-11 02:45:45 +07:00
const months = compareResult * ( calendarDifferences - isLastMonthNotFull ) ;
2020-06-27 15:40:45 +04:30
return months === 0 ? 0 : months ;
}