@plandek-utils/ts-parse-dayjs
v6.3.0
Published
utils to parse Dayjs objects in UTC
Downloads
6,194
Readme
@plandek-utils/ts-parse-dayjs
TypeDoc generated docs in here
utils to parse Dayjs objects in UTC
Installation
yarn add @plandek-utils/ts-parse-dayjs
or npm install @plandek-utils/ts-parse-dayjs
.
Usage
Note: it will return Dayjs objects with plugins:
parseDayjs
parses the given input and returns a Dayjs object (in UTC) if the given date is valid, or null
if it results in an invalid date
import { parseDayjs } from "@plandek-utils/ts-parse-dayjs";
parseDayjs(null); // => null
parseDayjs(""); // => null
parseDayjs("waa"); // => null
parseDayjs("2018-01-01"); // => Dayjs object in UTC for 2018-01-01
parseDayjs("2018-01-01T00:00:00.000Z"); // => Dayjs object in UTC for 2018-01-01T00:00:00.000Z
parseDayjs(new Date()); // => Dayjs object in UTC for the given Date
parseDayjs(dayjsObject); // => same Dayjs object
// can be formatted
parseDayjs("2018-01-01T00:00:00.000Z").format("Do MMM YYYY"); // => 1st Jan 2018
Locale
By default the locale used is en-gb
. This package import automatically all the english locales.
You can explicitly ask for a different locale passing a second argument. It can be one of the AvailableLocales or one of the Dayjs locale objects:
note: BREAKING CHANGE from 1.x to 2.x: default locale changed from en
to en-gb
note: BREAKING CHANGE from 2.x to 3.x: The second argument is now an object like { locale: "new-locale" }
instead of the locale directly
import { parseDayjs, AvailableLocales } from "@plandek-utils/ts-parse-dayjs";
parseDayjs("2018-01-01"); // => Dayjs object in UTC for 2018-01-01, locale en-gb (week starts on Monday)
parseDayjs("2018-01-01", { locale: AvailableLocales.EnglishUSA }); // => Dayjs object in UTC for 2018-01-01, locale en (week starts on Sunday)
import esLocale from "dayjs/locale/es";
parseDayjs("2018-01-01", { locale: esLocale }); // => Dayjs object in UTC for 2018-01-01, locale es (spanish)
strict
option
By passing strict: true
in the options, we ensure that we will receive a Dayjs object, or throw an InvalidDateError
. This is the same as parseDayjsOrError
.
import { parseDayjs } from "@plandek-utils/ts-parse-dayjs";
parseDayjs("2018-01-01", { strict: true }); // => Dayjs object in UTC for 2018-01-01, locale en-gb (week starts on Monday)
parseDayjs("waaaa", { strict: true }); // => throws InvalidDateError
dayjsNow
returns a Dayjs (in UTC) of the current time. It can accept a locale as optional argument (see Locale)
import { dayjsNow, AvailableLocales } from "@plandek-utils/ts-parse-dayjs";
dayjsNow(); // => Dayjs object with the current time, in UTC (locale en-gb)
dayjsNow({ locale: AvailableLocales.EnglishUSA }); // => Dayjs object with the current time, in UTC (locale en)
import esLocale from "dayjs/locale/es";
dayjsNow({ locale: esLocale }); // => Dayjs with the current time, in UTC, locale es
dayjsTodayEOD
returns a Dayjs (in UTC) of the current utc date at the end of date. It can accept a locale as optional argument (see Locale)
import {
dayjsNow,
dayjsTodayEOD,
AvailableLocales
} from "@plandek-utils/ts-parse-dayjs";
dayjsTodayEOD(); // => Dayjs object with the date at 23:59:59.999Z, in UTC (locale en-gb)
dayjsTodayEOD({ locale: AvailableLocales.EnglishUSA }); // => same, with locale en
dayjsNow({ time: TimeOverride.EndOfDay }); // => equivalent to dayjsTodayEOD()
import esLocale
from "dayjs/locale/es";
import {
TimeOverride
} from "./time-options";
dayjsTodayEOD({ locale: esLocale }); // => Dayjs with the current time, in UTC, locale es
parseDayjsOrError
same as parseDayjs
but instead of returning null
on an invalid date, it throws an InvalidDateError
. It can accept an optional second argument locale (see Locale)
import { parseDayjsOrError } from "@plandek-utils/ts-parse-dayjs";
parseDayjsOrError("2018-01-01T00:00:00.000Z"); // => Dayjs object in UTC for 2018-01-01T00:00:00.000Z
parseDayjsOrError(null); // => throws InvalidDateError
note: this is the same as passing a strict: true
option to parseDayJs
parseDayjsStartOfDay
and parseDayjsEndOfDay
same as parseDayjs
but if it results in a valid Dayjs object, it will then modify it to be at the beginning or end of the UTC day. It can accept an optional second argument locale (see Locale)
import { parseDayjsOrError } from "@plandek-utils/ts-parse-dayjs";
parseDayjsStartOfDay(null); // => null
parseDayjsEndOfDay(null); // => null
parseDayjsStartOfDay("2018-01-01"); // => Dayjs object in UTC for 2018-01-01T00:00:00.000Z
parseDayjsEndOfDay("2018-01-01"); // => Dayjs object in UTC for 2018-01-01T23:59:59.999Z
parseFromStandardPeriods
given a string like 8d
, 12w
, 9m
or 2y
, it returns an object with the calculated range of 8 days, 12 weeks, 9 months and 2 years ago respectively.
Optionally we can set an origin as second argument, otherwise the current time is used.
note: BREAKING CHANGE from 2.x to 3.x: The second argument is now an object like { origin: originDate }
instead of the origin directly
If the string cannot be parsed correctly, it will return null
.
If the string is parsed correctly, it will return an object with:
to
: the given origin (or current time if missing), at EOD of UTC (e.g. today at 23:59:59.999Z)from
: the calculatedfrom
after subtracting the given amount of the given unit, at the beginning of the period (beginning of the day, of the week, of the month or of the year)
import { parseFromStandardPeriods, parseDayjs } from "@plandek-utils/ts-parse-dayjs";
// if "now" is 2019-10-22T12:34:56.123Z
parseFromStandardPeriods(""); // => null
parseFromStandardPeriods("wat"); // => null
parseFromStandardPeriods("10z"); // => null
parseFromStandardPeriods("10d");
// => { from: Dayjs(2019-10-12T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
parseFromStandardPeriods("2w");
// => { from: Dayjs(2019-10-07T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
parseFromStandardPeriods("2m");
// => { from: Dayjs(2019-08-01T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
parseFromStandardPeriods("2y");
// if the number is 0, it will go to the beginning of the period
parseFromStandardPeriods("0d");
// => { from: Dayjs(2019-10-22T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
parseFromStandardPeriods("0w");
// => { from: Dayjs(2019-10-21T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
parseFromStandardPeriods("0m");
// => { from: Dayjs(2019-10-01T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
parseFromStandardPeriods("0y");
// => { from: Dayjs(2019-01-01T00:00:00.000Z), to: Dayjs(2019-10-22T23:59:59.999Z) }
// with optional origin
const d = parseDayjs("2019-05-10");
parseFromStandardPeriods("3d", { origin: d });
// => { from: Dayjs(2019-05-07T00:00:00.000Z), to: Dayjs(2019-05-10T23:59:59.999Z) }
formatDate()
Used to format a Dayjs object using the AdvancedFormat plugin. The default format is "Do MMM YYYY"
import { formatDate, parseDayjs } from "@plandek-utils/ts-parse-dayjs";
formatDate(null); // => null
formatDate(parseDayjs("2019-01-02")); // => "2nd Jan 2019"
formatDate(parseDayjs("2019-01-02"), "Do ww MMMM YYYY"); // => "2nd 01 January 2019"
formatDateTime()
Same as formatDate()
but with a different default format. The default format is "Do MMM YYYY h:mm A"
import { formatDateTime, parseDayjs } from "@plandek-utils/ts-parse-dayjs";
formatDateTime(null); // => null
formatDateTime(parseDayjs("2019-01-02T13:24:12Z")); // => "2nd Jan 2019 1:24 PM"
formatDateTime(parseDayjs("2019-01-02T13:24:12Z"), "Do ww MMMM YYYY h:mm A"); // => "2nd 01 January 2019 1:24 PM"
formatDate(parseDayjs("2019-01-02T13:24:12Z"), "Do ww MMMM YYYY h:mm A"); // => "2nd 01 January 2019 1:24 PM"
toISOString()
, and types ISODate
, ISODateTime
, ISOTime
Used to get an ISO 8601 string from a Dayjs object.
It is not possible to type more precisely (list every possible values for hours, etc.) as it would result in a warning from TypeScript: "Expression produces a union type that is too complex to represent. ts(2590)"
import type { ISODate, ISODateTime, ISOTime } from "@plandek-utils/ts-parse-dayjs";
import { parseDayjsOrError, toISOString} from "@plandek-utils/ts-parse-dayjs";
const d = parseDayjsOrError("2020-01-01T12:34:56.789Z");
toISOString(d); // => "2020-01-01T12:34:56.789Z"
const x1: ISODate = "2020-01-01" // ok
const x2: ISODate = "2020-02-31" // ok -> does not check for days of each month
const x3: ISODate = "12-01-01" // ok -> only checks that the year is a number
const b1: ISODate = "2020-13-01" // bad -> month is not between 01 and 12
const b2: ISODate = "2020-12-41" // bad -> day is not between 01 and 31
const t1: ISOTime = "12:34:56.789Z" // ok
const dt1: ISODateTime = "2020-01-01T12:34:56.789Z" // ok
minDayjs()
and maxDayjs()
Used to compare an array of Dayjs objects and return the min (earliest) or max (latest).
import { minDayjs, maxDayjs, parseDayjs } from "@plandek-utils/ts-parse-dayjs";
const d1 = parseDayjsOrError("2019-01-01");
const d2 = parseDayjsOrError("2019-02-02");
const d3 = parseDayjsOrError("2019-03-03");
const d4 = parseDayjsOrError("2019-04-04");
minDayjs([]); // => null
minDayjs([d3, d4, d1, d2]); // => d1
maxDayjs([]); // => null
maxDayjs([d3, d4, d1, d2]); // => d4
isStrictDayjsInput()
and isDayjsInput()
return true if the given input is a valid DayjsInput (Date
, Dayjs object, string or number):
isDayjsInput
allows fornull
orundefined
isStrictDayjsInput
does not allow fornull
orundefined
isValidDate()
returns true if the given string, dayjs or Date object represent a valid date, false otherwise. It always returns false if given null
or undefined
import { isValidDate, parseDayjs } from "@plandek-utils/ts-parse-dayjs";
isValidDate(null); // => false
isValidDate(undefined); // => false
isValidDate("2012-12-15"); // => true
isValidDate("2012-12-15T12:31:31Z"); // => true
isValidDate("2012-12-15T12:31:97Z"); // => false
isValidDate("2012-12-32T12:31:00Z"); // => false
isValidDate("2012-12-32"); // => false
isValidDate(new Date("2012-12-15")); // => true
isValidDate(new Date("2012-12-15T12:31:31Z")); // => true
isValidDate(new Date("2012-12-15T12:31:97Z")); // => false
isValidDate(new Date("2012-12-32T12:31:00Z")); // => false
isValidDate(new Date("2012-12-32")); // => false
isValidDate(parseDayjs("2012-12-15")); // => true
isValidDate(parseDayjs("2012-12-15T12:31:31Z")); // => true
isValidDate(parseDayjs("2012-12-15T12:31:97Z")); // => false
isValidDate(parseDayjs("2012-12-32T12:31:00Z")); // => false
isValidDate(parseDayjs(new Date("2012-12-32"))); // => false
durationBetween(a, b)
returns the Duration
type of b.diff(a)
We also have isDuration(x)
function.
import { durationBetween, parseDayjsOrError } from "@plandek-utils/ts-parse-dayjs";
const a = parseDayjsOrError("2012-12-15T12:31:31Z");
const b = parseDayjsOrError("2012-12-15T12:31:41Z");
const d = durationBetween(a, b);
d.asSeconds(); // => 10
isDuration(d); // => true
fromNow(value, withoutSuffix)
and toNow(value, withoutSuffix)
returns a string containing the humanized duration of the value
from the current time.
encapsulates locale configuration of formatting for relativeTime
.
optionally omits "ago" as the string suffix if withoutSuffix
is true. By default is is applied.
import { fromNow } from "@plandek-utils/ts-parse-dayjs";
// if "now" is "2012-12-16T00:00:00Z"
const a = parseDayjsOrError("2012-12-15T00:00:00Z");
const res1 = fromNow(a); // "1 day ago"
const res2 = toNow(a); // "in 1 day"
fromNowStrict(value, withoutSuffix)
and toNowStrict(value, withoutSuffix)
Same as fromNow()
and toNow()
, but it will round down the numbers instead.
import { fromNow, fromNowStrict } from "@plandek-utils/ts-parse-dayjs";
// if "now" is "2012-12-16T23:00:00Z"
const a = parseDayjsOrError("2012-12-15T00:00:00Z");
const res1 = fromNow(a); // "2 days ago" (1d + 23h)
const res2 = fromNowStrict(a); // "1 day ago" (1d + 23h)
const res3 = toNow(a); // "in 2 days" (1d + 23h)
const res4 = toNowStrict(a); // "in 1 day" (1d + 23h)
isTodayOrFuture()
and isTodayOrPast()
import { isTodayOrFuture, isTodayOrPast } from "@plandek-utils/ts-parse-dayjs";
isTodayOrFuture(today); // => true
isTodayOrFuture(aSecondAgo); // => true
isTodayOrFuture(aSecondInFuture); // => true
isTodayOrFuture(aWeekAgo); // => false
isTodayOrFuture(aWeekInFuture); // => true
isTodayOrPast(today); // => true
isTodayOrPast(aSecondAgo); // => true
isTodayOrPast(aSecondInFuture); // => true
isTodayOrPast(aWeekAgo); // => true
isTodayOrPast(aWeekInFuture); // => false
Granularity
we export:
GranularityEnumValues
: "day" | "week" | "month" | "year"GranularityDescriptionEnumValues
"daily" | "weekly" | "monthly" | "yearly"
getGranularityDescription(granularity)
import { getGranularityDescription } from "@plandek-utils/ts-parse-dayjs";
// null -> daily
getGranularityDescription(null) // => "daily"
// day -> daily
getGranularityDescription("day") // => "daily"
// week -> weekly
getGranularityDescription("week") // => "weekly"
// month -> monthly
getGranularityDescription("month") // => "monthly"
// year -> yearly
getGranularityDescription("year") // => "yearly"
getGranularityOptionsFromRange(range)
// empty array with missing or invalid dates
getGranularityOptionsFromRange({}) // => []
getGranularityOptionsFromRange({ from: "2022-01-01" }) // => []
getGranularityOptionsFromRange({ to: "2022-01-01" }) // => []
getGranularityOptionsFromRange({ from: "nope", to: "2022-10-10" }) // => []
// 2 days, starts Tue -> day
getGranularityOptionsFromRange({ from: "2022-07-12", to: "2022-07-13" }) // => ["day"]
// 10 days, starts Tue -> day
getGranularityOptionsFromRange({ from: "2022-07-12", to: "2022-07-21" }) // => ["day"]
// 20 days, starts Tue -> day
getGranularityOptionsFromRange({ from: "2022-07-12", to: "2022-07-31" }) // => ["day"]
// 21 days, starts Tue -> day
getGranularityOptionsFromRange({ from: "2022-07-12", to: "2022-08-01" }) // => ["day"]
// 41 days, starts Tue -> day
getGranularityOptionsFromRange({ from: "2022-07-12", to: "2022-08-21" }) // => ["day"]
// 2 days, starts Mon -> day
getGranularityOptionsFromRange({ from: "2022-07-11", to: "2022-07-13" }) // => ["day"]
// 10 days, starts Mon -> day, week
getGranularityOptionsFromRange({ from: "2022-07-11", to: "2022-07-20" }) // => ["day", "week"]
// 20 days, starts Mon -> day, week
getGranularityOptionsFromRange({ from: "2022-07-11", to: "2022-07-30" }) // => ["day", "week"]
// 21 days, starts Mon -> day, week
getGranularityOptionsFromRange({ from: "2022-07-11", to: "2022-07-31" }) // => ["day", "week"]
// 41 days, starts Mon -> day, week
getGranularityOptionsFromRange({ from: "2022-07-11", to: "2022-07-31" }) // => ["day", "week"]
// 2 days, starts Mon, Nov 1st -> day
getGranularityOptionsFromRange({ from: "2021-11-01", to: "2021-11-02" }) // => ["day"]
// 10 days, starts Mon, Nov 1st -> day, week
getGranularityOptionsFromRange({ from: "2021-11-01", to: "2021-11-10" }) // => ["day", "week"]
// 20 days, starts Mon, Nov 1st -> day, week
getGranularityOptionsFromRange({ from: "2021-11-01", to: "2021-11-20" }) // => ["day", "week"]
// 21 days, starts Mon, Nov 1st -> day, week
getGranularityOptionsFromRange({ from: "2021-11-01", to: "2021-11-21" }) // => ["day", "week"]
// 41 days, starts Mon, Nov 1st -> day, week, month
getGranularityOptionsFromRange({ from: "2021-11-01", to: "2021-12-11" }) // => ["day", "week", "month"]
// 500 days, starts Mon, Nov 1st -> day, week, month
getGranularityOptionsFromRange({ from: "2021-11-01", to: "2023-03-16" }) // => ["day", "week", "month"]
// 40 days, starts Friday, Jan 1st -> day, month
getGranularityOptionsFromRange({ from: "2021-01-01", to: "2021-02-09" }) // => ["day", "month"]
// 500 days, starts Friday, Jan 1st -> day, month, year
getGranularityOptionsFromRange({ from: "2021-01-01", to: "2022-05-16" }) // => ["day", "month", "year"]
// 40 days, starts Mon, Jan 1st -> day, week, month
getGranularityOptionsFromRange({ from: "2018-01-01", to: "2018-02-09" }) // => ["day", "week", "month"]
// 500 days, starts Mon, Jan 1st -> day, week, month, year
getGranularityOptionsFromRange({ from: "2018-01-01", to: "2019-05-16" }) // => ["day", "week", "month", "year"]
calculateDateRangeDescription(opts)
calculateDateRangeDescription({ from: dayjsFrom, to: dayjsTo }) // => "5th Aug 2017 - 15th Aug 2019"
calculateDateRangeDescription({ from: today, to: today, granularity: "day" }) // => "Today"
calculateDateRangeDescription({ from: today, to: today, granularity: "week" }) // => "This week"
calculateDateRangeDescription({ from: today, to: today, granularity: "month" }) // => "This month"
calculateDateRangeDescription({ from: today, to: today, granularity: "year" }) // => "This year"
calculateDateRangeDescription({ from: thisMonday, to: dayjsTo, granularity: "week" }) // => "This week"
Print utils
We have a few print utils to help you print the date with a given static prefix. By default we export printSince and printStarted, but you can make your own function
import { printSince, printStarted, makePrintWithPrefix } from "@plandek-utils/ts-parse-dayjs";
printSince("2020-01-01") // => "Since 1st Jan 2020"
princeStarted("2020-01-01") // => "Started 1st Jan 2020"
const fn = makePrintWithPrefix("Whatever ");
fn("2020-01-01") // => "Whatever 1st Jan 2020"
You can also pass the format to use as a second argument.
Print range
// if dates are different - returns a from date and to date formatted as per the provided format separated by '-'
printRange({ from: dayjsFrom, to: dayjsTo }) // => "5th Aug 2017 - 15th Aug 2019"
// it("if dates are the same - returns a formatted from date with no separator
printRange({ from: dayjsFrom, to: dayjsFrom }) // => "5th Aug 2017"
Development, Commits, versioning and publishing
See The Typescript-Starter docs.
Commits and CHANGELOG
For commits, you should use commitizen
yarn global add commitizen
#commit your changes:
git cz
As typescript-starter docs state:
This project is tooled for conventional changelog to make managing releases easier. See the standard-version documentation for more information on the workflow, or CHANGELOG.md
for an example.
# bump package.json version, update CHANGELOG.md, git tag the release
yarn run version
You may find a tool like wip
helpful for managing work in progress before you're ready to create a meaningful commit.
Creating the first version
Once you are ready to create the first version, run the following (note that reset
is destructive and will remove all files not in the git repo from the directory).
# Reset the repo to the latest commit and build everything
yarn run reset && yarn run test && yarn run doc:html
# Then version it with standard-version options. e.g.:
# don't bump package.json version
yarn run version -- --first-release
# Other popular options include:
# PGP sign it:
# $ yarn run version -- --sign
# alpha release:
# $ yarn run version -- --prerelease alpha
And after that, remember to publish the docs.
And finally push the new tags to github and publish the package to npm.
# Push to git
git push --follow-tags origin master
# Publish to NPM (allowing public access, required if the package name is namespaced like `@somewhere/some-lib`)
yarn publish --access public
Publish the Docs
yarn run doc:html && yarn run doc:publish
This will generate the docs and publish them in github pages.
Generate a version
There is a single yarn command for preparing a new release. See One-step publish preparation script in TypeScript-Starter
# Prepare a standard release
yarn prepare-release
# Push to git
git push --follow-tags origin master
# Publish to NPM (allowing public access, required if the package name is namespaced like `@somewhere/some-lib`)
yarn publish --access public