mirror of
https://github.com/therootcompany/tz.js.git
synced 2024-11-16 17:29:01 +00:00
Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| f03e75b898 | |||
| fd0f3a265b | |||
| 6417eca7f6 | |||
| ffba59336f | |||
| b5649558b7 | |||
| ba1c97cda2 | |||
| 3f5d230261 | |||
| a195b6c5d5 | |||
| bf0130780d | |||
| ff996abbc0 | |||
| 6a095b0966 | |||
| 0501ab1899 | |||
| c3f690ae6d | |||
| 7acf670aa6 | |||
| d314200e03 |
121
README.md
121
README.md
@ -8,52 +8,37 @@ in ~100 LoC. For Node.js & Browsers.
|
||||
XTZ is a poor man's `Temporal` polyfill, but just for time zones. \
|
||||
Demo: <https://therootcompany.github.io/tz.js/>
|
||||
|
||||
> What UTC time will it be when it's 3:15am in New York?
|
||||
```js
|
||||
// What's the current time, in ISO+Offset format?
|
||||
|
||||
TZ.toLocalISOString(new Date()); // "2021-11-07T03:15:59.000-0500"
|
||||
TZ.timeZone(); // "America/New_York"
|
||||
```
|
||||
|
||||
```js
|
||||
// Relative New York time to Absolute UTC Time
|
||||
TZ.toUTCISOString("2021-11-07 03:15:59.000", "America/New_York");
|
||||
// What will the ISO+Offset datetime string be
|
||||
// when it's 3:15am in New York?
|
||||
//
|
||||
// (Relative New York time to Absolute ISO+Offset Time)
|
||||
|
||||
TZ.toOffsetISOString("2021-11-07 03:15:59.000", "America/New_York");
|
||||
// "2021-11-07T03:15:59.000-0500"
|
||||
```
|
||||
|
||||
```js
|
||||
var tzDate = TZ.toUTC("2021-11-07 03:15:59.000", "America/New_York");
|
||||
// {
|
||||
// year: 2021, month: 10, day: 7,
|
||||
// hour: 3, minute: 15, second: 59, millisecond: 0,
|
||||
// offset: -300, timeZoneName: "Eastern Standard Time"
|
||||
// }
|
||||
// What time will it be in New York
|
||||
// when it's 7:15am UTC?
|
||||
//
|
||||
// (Absolute UTC Zulu time to Relative New York time)
|
||||
|
||||
tzDate.toISOString();
|
||||
// "2021-11-07T03:15:59.000-0500"
|
||||
// (same as "2021-11-07T08:15:59.000Z")
|
||||
```
|
||||
|
||||
> What time will it be in New York when it's 7:15am UTC?
|
||||
|
||||
```js
|
||||
// Absolute UTC time to Relative New York time
|
||||
TZ.toTimeZoneISOString("2021-03-14T07:15:59.000Z", "America/New_York");
|
||||
// "2021-03-14T03:15:59.000-0400"
|
||||
```
|
||||
|
||||
```js
|
||||
var utcDate = TZ.toTimeZone("2021-03-14T07:15:59.000Z", "America/New_York");
|
||||
// {
|
||||
// year: 2021, month: 2, day: 14,
|
||||
// hour: 3, minute: 15, second: 59, millisecond: 0,
|
||||
// offset: -240, timeZoneName: "Eastern Daylight Time"
|
||||
// }
|
||||
|
||||
utcDate.toISOString();
|
||||
// "2021-03-14T03:15:59.000-0400"
|
||||
// (same as "2021-11-07T07:15:59.000Z")
|
||||
```
|
||||
|
||||
# Features
|
||||
|
||||
- [x] Translate a UTC time to a Time Zone
|
||||
- [x] Translate a Zoned time to UTC
|
||||
- [x] Translate a UTC Zulu time to a Time Zone
|
||||
- [x] Translate a Zoned time to ISO+Offset
|
||||
- [x] Handles **Daylight Savings**, Weird Time Zones, etc...
|
||||
- [x] Well-tested `npm run test`
|
||||
- [x] Lightweight (No deps)
|
||||
@ -87,37 +72,60 @@ var TZ = require("xtz");
|
||||
|
||||
See <https://therootcompany.github.io/tz.js/>.
|
||||
|
||||
## How was this built?
|
||||
|
||||
I live-streamed the creation of this entire project.
|
||||
|
||||
If you'd like to learn how I did it and what challenges I encountered, you can watch here:
|
||||
https://www.youtube.com/playlist?list=PLxki0D-ilnqa6horOJ2G18WMZlJeQFlAt
|
||||
|
||||
(though there have been a few minor updates and bug fixes off-camera)
|
||||
|
||||
# API
|
||||
|
||||
- `toLocalISOString(dateOrNull)`
|
||||
- `toTimeZone(utcDate, timeZone)`
|
||||
- `toTimeZoneISOString(isoString, timeZone)`
|
||||
- `toUTC(dtString, timeZone)`
|
||||
- `toUTCISOString(dtString, timeZone)`
|
||||
- `fromTimeZone(dtString, timeZone)`
|
||||
- `toOffsetISOString(dtString, timeZone)`
|
||||
|
||||
## `toTimeZone(utcDate, timeZone)`
|
||||
|
||||
> Convert UTC into a Target Time Zone
|
||||
|
||||
Use ISO timestamps representing the absolute UTC time (with or without offset):
|
||||
Use ISO timestamps representing the absolute UTC time (ISO with or without offset):
|
||||
|
||||
```txt
|
||||
"2021-11-07T08:15:59.000Z"
|
||||
```
|
||||
|
||||
Convert directly to an ISO String:
|
||||
```js
|
||||
var utcDate = TZ.toTimeZone("2021-03-14T07:15:59.000Z", "America/New_York");
|
||||
// {
|
||||
// year: 2021, month: 2, day: 14,
|
||||
// hour: 3, minute: 15, second: 59, millisecond: 0,
|
||||
// offset: -240, timeZoneName: "Eastern Daylight Time"
|
||||
// }
|
||||
|
||||
utcDate.toISOString();
|
||||
// "2021-03-14T03:15:59.000-0400"
|
||||
// (same as "2021-11-07T07:15:59.000Z")
|
||||
```
|
||||
|
||||
### Convert directly to an ISO String:
|
||||
|
||||
```js
|
||||
TZ.toTimeZoneISOString("2021-11-07T08:15:59.000Z", "America/New_York");
|
||||
// "2021-11-07T03:15:59.000-0500"
|
||||
```
|
||||
|
||||
Or use our bespoke (custom) date object:
|
||||
### Or use our bespoke (custom) date object:
|
||||
|
||||
```js
|
||||
var tzDate = TZ.toTimeZone("2021-11-07T08:15:59.000Z", "America/New_York");
|
||||
```
|
||||
|
||||
You can also use a date object with an absolute UTC time:
|
||||
### You can also use a date object with an absolute ISO time:
|
||||
|
||||
```js
|
||||
var tzDate = TZ.toTimeZone(
|
||||
@ -131,16 +139,16 @@ console.log(tzDate.toISOString());
|
||||
// "2021-11-07T03:15:59.000-0500"
|
||||
```
|
||||
|
||||
Our ISO Strings + Offsets work with JavaScript's native Date object!!
|
||||
### Our ISO Strings + Offsets work with JavaScript's native Date object!!
|
||||
|
||||
```js
|
||||
new Date("2021-11-07T03:15:59.000-0500").toISOString());
|
||||
// "2021-11-07T08:15:59.000Z"
|
||||
```
|
||||
|
||||
## `toUTC(dtString, timeZone)`
|
||||
## `fromTimeZone(dtString, timeZone)`
|
||||
|
||||
> Convert a Target Time Zone into UTC
|
||||
> Convert a Target Time Zone into ISO
|
||||
|
||||
Use ISO-like timestamps representing the _local_ time in the target time zone:
|
||||
|
||||
@ -148,24 +156,39 @@ Use ISO-like timestamps representing the _local_ time in the target time zone:
|
||||
"2021-11-0 03:15:59.000"
|
||||
```
|
||||
|
||||
Convert directly to an offset ISO String:
|
||||
```js
|
||||
var tzDate = TZ.fromTimeZone("2021-11-07 03:15:59.000", "America/New_York");
|
||||
// {
|
||||
// year: 2021, month: 10, day: 7,
|
||||
// hour: 3, minute: 15, second: 59, millisecond: 0,
|
||||
// offset: -300, timeZoneName: "Eastern Standard Time"
|
||||
// }
|
||||
|
||||
tzDate.toISOString();
|
||||
// "2021-11-07T03:15:59.000-0500"
|
||||
// (same as "2021-11-07T08:15:59.000Z")
|
||||
```
|
||||
|
||||
### Convert directly to an offset ISO String:
|
||||
|
||||
```js
|
||||
TZ.toUTCISOString("2021-11-07 03:15:59.000", "America/New_York");
|
||||
TZ.toOffsetISOString("2021-11-07 03:15:59.000", "America/New_York");
|
||||
// "2021-11-07T03:15:59.000-0500"
|
||||
```
|
||||
|
||||
Or our bespoke date object:
|
||||
### Or our bespoke date object:
|
||||
|
||||
```js
|
||||
var utcDate = TZ.toUTC("2021-11-07 03:15:59.000", "America/New_York");
|
||||
var utcDate = TZ.fromTimeZone("2021-11-07 03:15:59.000", "America/New_York");
|
||||
```
|
||||
|
||||
### Use a Date as a source time
|
||||
|
||||
You can also use a date object as the source time, but the date's UTC time will be treated as **_relative to time
|
||||
zone_** rather than absolute (this is a workaround for JavaScript's lack of bi-directional timezone support).
|
||||
|
||||
```js
|
||||
var utcDate = TZ.toUTC(
|
||||
var utcDate = TZ.fromTimeZone(
|
||||
new Date("2021-11-07T03:15:59.000Z"),
|
||||
"America/New_York"
|
||||
);
|
||||
@ -190,7 +213,7 @@ Q: What happens in March when 2am is skipped?
|
||||
- A: Although 2am is not a valid time, rather than throwing an error this library will resolve to 1am instead, which
|
||||
is an hour early in real ("tick-tock" or "monotonic") time.
|
||||
```js
|
||||
var utcDate = TZ.toUTC("2021-03-14 02:15:59.000", "America/New_York");
|
||||
var utcDate = TZ.fromTimeZone("2021-03-14 02:15:59.000", "America/New_York");
|
||||
utcDate.toISOString();
|
||||
// "2021-03-14T02:15:59.000-0400"
|
||||
// (same as "2021-03-14T01:15:59.000-0500")
|
||||
@ -201,7 +224,7 @@ Q: What happens in November when 1am happens twice?
|
||||
- A: Although both 1ams are distinguishable with ISO offset times, only the first can be resolved from a local time
|
||||
with this library.
|
||||
```js
|
||||
var utcDate = TZ.toUTC("2021-11-07 01:15:59.000", "America/New_York");
|
||||
var utcDate = TZ.fromTimeZone("2021-11-07 01:15:59.000", "America/New_York");
|
||||
utcDate.toISOString();
|
||||
// "2021-11-07T01:15:59.000-0400", same as "2021-11-07T05:15:59.000Z"
|
||||
// (an hour before the 2nd 1am at "2021-11-07T01:15:59.000-0500")
|
||||
|
||||
12
examples.js
12
examples.js
@ -22,9 +22,9 @@ var XTZ;
|
||||
console.info();
|
||||
|
||||
console.info("\t// during daylight savings");
|
||||
console.info(`\tXTZ.toUTC("2021-03-14 08:15:59.000", "America/New_York")`);
|
||||
console.info(`\tXTZ.fromTimeZone("2021-03-14 08:15:59.000", "America/New_York")`);
|
||||
console.info(`\ttzDate.toISOString()`);
|
||||
tzDate = XTZ.toUTC("2021-03-14 08:15:59.000", "America/New_York");
|
||||
tzDate = XTZ.fromTimeZone("2021-03-14 08:15:59.000", "America/New_York");
|
||||
console.info(
|
||||
"\t" + tzDate.toISOString(),
|
||||
"// same as",
|
||||
@ -33,9 +33,9 @@ var XTZ;
|
||||
console.info();
|
||||
|
||||
console.info("\t// during standard time");
|
||||
console.info(`\tXTZ.toUTC("2021-11-07 08:15:59.000", "America/New_York")`);
|
||||
console.info(`\tXTZ.fromTimeZone("2021-11-07 08:15:59.000", "America/New_York")`);
|
||||
console.info(`\ttzDate.toISOString()`);
|
||||
tzDate = XTZ.toUTC("2021-11-07 08:15:59.000", "America/New_York");
|
||||
tzDate = XTZ.fromTimeZone("2021-11-07 08:15:59.000", "America/New_York");
|
||||
console.info(
|
||||
"\t" + tzDate.toISOString(),
|
||||
"// same as",
|
||||
@ -65,9 +65,9 @@ var XTZ;
|
||||
console.info();
|
||||
|
||||
console.info("\t// during standard time");
|
||||
console.info(`\tXTZ.toUTC("2021-11-07T08:15:59.000Z", "America/New_York")`);
|
||||
console.info(`\tXTZ.fromTimeZone("2021-11-07T08:15:59.000Z", "America/New_York")`);
|
||||
console.info(`\ttzDate.toISOString()`);
|
||||
tzDate = XTZ.toUTC("2021-11-07T08:15:59.000Z", "America/New_York");
|
||||
tzDate = XTZ.fromTimeZone("2021-11-07T08:15:59.000Z", "America/New_York");
|
||||
console.info(
|
||||
"\t" + tzDate.toISOString(),
|
||||
"// same as",
|
||||
|
||||
36
index.html
36
index.html
@ -31,8 +31,11 @@
|
||||
<br />
|
||||
<span class="js-my-dt">YYYY-MM-DDThh:mm:ss.mmm+0000</span>
|
||||
</h3>
|
||||
<pre><code>new Intl.DateTimeFormat().resolvedOptions().timeZone;</code></pre>
|
||||
Current Time Zone: <span class="js-my-tz">America/XXXX</span>
|
||||
<br />
|
||||
<pre><code>String(((new Date()).getTimezoneOffset() / 60)).padStart(2, '0') +
|
||||
String(((new Date()).getTimezoneOffset() % 60)).padStart(2, '0')</code></pre>
|
||||
Current Time Offset: <span class="js-my-offset">-0000</span>
|
||||
<hr />
|
||||
|
||||
@ -83,9 +86,21 @@
|
||||
|
||||
<hr />
|
||||
|
||||
<form class="js-local">
|
||||
<h3>Local ISO+Offset String:</h3>
|
||||
<pre><code>XTZ.toLocalISOString(new Date())</code></pre>
|
||||
<label>
|
||||
ISO:
|
||||
<input class="js-my-dt-local" type="text" disabled />
|
||||
</label>
|
||||
<br />
|
||||
</form>
|
||||
|
||||
<hr />
|
||||
|
||||
<form class="js-tz2utc">
|
||||
<h3>Relative TimeZone to Absolute UTC:</h3>
|
||||
<pre><code>XTZ.toUTC("<span class="js-dtx-tz">YYYY-03-14 03:15:69.000</span>", "<span class="js-tzx-tz">UTC</span>")
|
||||
<h3>TimeZone-Relative (Local) to Absolute (ISO+Offset) String:</h3>
|
||||
<pre><code>XTZ.fromTimeZone("<span class="js-dtx-tz">YYYY-03-14 03:15:69.000</span>", "<span class="js-tzx-tz">UTC</span>")
|
||||
.toISOString()
|
||||
// <span class="js-myx-dt-tz"></span></code></pre>
|
||||
<label>
|
||||
@ -115,14 +130,14 @@
|
||||
</label>
|
||||
<br />
|
||||
<br />
|
||||
<button type="submit">Convert to UTC!</button>
|
||||
<button type="submit">Convert to ISO+Offset!</button>
|
||||
<br />
|
||||
</form>
|
||||
|
||||
<hr />
|
||||
|
||||
<form class="js-utc2tz">
|
||||
<h3>Absolute UTC to Relative TimeZone</h3>
|
||||
<h3>Absolute (UTC) to TimeZone-Relative (ISO+Offset) String:</h3>
|
||||
<pre><code>XTZ.toTimeZone("<span class="js-dtx-utc">YYYY-03-14 03:15:69.000</span>", "<span class="js-tzx-utc">UTC</span>")
|
||||
.toISOString()
|
||||
// <span class="js-myx-dt-utc"></span></code></pre>
|
||||
@ -179,23 +194,24 @@ Asia/Kolkata +0530 (No DST) (30-min)</pre
|
||||
>github.com/therootcompany/tz.js</a
|
||||
>
|
||||
<script src="./xtz.js"></script>
|
||||
<script src="./tolocalisostring.js"></script>
|
||||
<script>
|
||||
function $(sel, el) {
|
||||
return (el || document.body).querySelector(sel);
|
||||
}
|
||||
|
||||
function translate(dt, tz, tz2) {
|
||||
var utc = new Date(XTZ.toUTC(dt, tz).toISOString());
|
||||
var utc = new Date(XTZ.fromTimeZone(dt, tz).toISOString());
|
||||
if (!tz2) {
|
||||
return utc;
|
||||
}
|
||||
return XTZ.toTimeZone(utc, tz2);
|
||||
}
|
||||
|
||||
function toUTC() {
|
||||
function fromTimeZone() {
|
||||
var curDt = $(".js-dt-tz").value;
|
||||
var curTz = $(".js-tz-tz").value;
|
||||
$(".js-my-dt-tz").value = XTZ.toUTC(curDt, curTz).toISOString();
|
||||
$(".js-my-dt-tz").value = XTZ.fromTimeZone(curDt, curTz).toISOString();
|
||||
|
||||
$(".js-dtx-tz").innerText = curDt;
|
||||
$(".js-tzx-tz").innerText = curTz;
|
||||
@ -219,6 +235,8 @@ Asia/Kolkata +0530 (No DST) (30-min)</pre
|
||||
$(".js-my-dt-xtz").value = translate(curDt, inTz, outTz).toISOString();
|
||||
}
|
||||
|
||||
$(".js-my-dt-local").value = XTZ.toLocalISOString();
|
||||
|
||||
var myTz = new Intl.DateTimeFormat("default", {}).resolvedOptions()
|
||||
.timeZone;
|
||||
$(".js-my-tz").innerText = myTz;
|
||||
@ -247,10 +265,10 @@ Asia/Kolkata +0530 (No DST) (30-min)</pre
|
||||
$("form.js-tz2utc").addEventListener("submit", function (ev) {
|
||||
ev.preventDefault();
|
||||
ev.stopPropagation();
|
||||
toUTC();
|
||||
fromTimeZone();
|
||||
});
|
||||
|
||||
toUTC();
|
||||
fromTimeZone();
|
||||
toTZ();
|
||||
xTZ();
|
||||
</script>
|
||||
|
||||
@ -1,8 +1,12 @@
|
||||
{
|
||||
"name": "xtz",
|
||||
"version": "1.0.2",
|
||||
"version": "1.3.2",
|
||||
"description": "A fast, lightweight, zero-dependency library to translate between Time Zones and UTC with native Intl.DateTimeFormat in ~100 LoC. For Node.js & Browsers.",
|
||||
"main": "xtz.js",
|
||||
"files": [
|
||||
"xtz.js",
|
||||
"tolocalisostring.js"
|
||||
],
|
||||
"scripts": {
|
||||
"test": "node ./test.js",
|
||||
"build": "uglifyjs ./xtz.js > ./xtz.min.js ; gzip -k -f ./xtz.min.js"
|
||||
|
||||
74
test.js
74
test.js
@ -4,9 +4,10 @@ var TZ = require("./");
|
||||
|
||||
function testUtcToTz(t) {
|
||||
var result = TZ.toTimeZone.apply(TZ, t.inputs).toISOString();
|
||||
if (t.result !== result) {
|
||||
var result2 = TZ.toTimeZoneISOString.apply(TZ, t.inputs);
|
||||
if (result !== result2 || t.result !== result) {
|
||||
throw new Error(
|
||||
`Invalid UTC to TZ conversion for ${t.desc}:\n` +
|
||||
`Invalid UTC/ISO+Offset to TZ conversion for ${t.desc}:\n` +
|
||||
`\tExpected: ${t.result}\n` +
|
||||
`\tActual: ${result}\n`
|
||||
);
|
||||
@ -14,9 +15,11 @@ function testUtcToTz(t) {
|
||||
}
|
||||
|
||||
function testTzToUtc(t) {
|
||||
var result = TZ.toUTC.apply(TZ, t.inputs);
|
||||
if (t.result !== result.toISOString()) {
|
||||
console.log(result);
|
||||
var result = TZ.fromTimeZone.apply(TZ, t.inputs).toISOString();
|
||||
var result2 = TZ.toOffsetISOString.apply(TZ, t.inputs);
|
||||
var result3 = TZ.toUTC.apply(TZ, t.inputs).toISOString();
|
||||
if (t.result !== result || t.result !== result2 || t.result !== result3) {
|
||||
console.error(result);
|
||||
throw new Error(
|
||||
`Invalid TZ to UTC conversion for ${t.desc}:\n` +
|
||||
`\tExpected: ${t.result}\n` +
|
||||
@ -44,23 +47,23 @@ function testTzToUtc(t) {
|
||||
|
||||
// 12:15am NY -0500 => -0400
|
||||
{
|
||||
desc: "UTC to 12:15am NY EST",
|
||||
desc: "UTC Zulu to 12:15am NY EST",
|
||||
inputs: ["2021-03-14T05:15:59.000Z", "America/New_York"],
|
||||
result: "2021-03-14T00:15:59.000-0500",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 12:15am NY EST (2)",
|
||||
desc: "ISO+Offset to 12:15am NY EST (2)",
|
||||
inputs: ["2021-03-14T00:15:59.000-0500", "America/New_York"],
|
||||
result: "2021-03-14T00:15:59.000-0500",
|
||||
},
|
||||
// 1:15am NY (non-DST)
|
||||
{
|
||||
desc: "UTC to 1:15am NY EST",
|
||||
desc: "UTC Zulu to 1:15am NY EST",
|
||||
inputs: ["2021-03-14T06:15:59.000Z", "America/New_York"],
|
||||
result: "2021-03-14T01:15:59.000-0500",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 1:15am NY EST (2)",
|
||||
desc: "ISO+Offset to 1:15am NY EST (2)",
|
||||
inputs: ["2021-03-14T01:15:59.000-0500", "America/New_York"],
|
||||
result: "2021-03-14T01:15:59.000-0500",
|
||||
},
|
||||
@ -69,23 +72,23 @@ function testTzToUtc(t) {
|
||||
|
||||
// 3:15am NY (DST)
|
||||
{
|
||||
desc: "UTC to 3:15am NY EDT",
|
||||
desc: "UTC Zulu to 3:15am NY EDT",
|
||||
inputs: ["2021-03-14T07:15:59.000Z", "America/New_York"],
|
||||
result: "2021-03-14T03:15:59.000-0400",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 3:15am NY EDT (2)",
|
||||
desc: "ISO+Offset to 3:15am NY EDT (2)",
|
||||
inputs: ["2021-03-14T03:15:59.000-0400", "America/New_York"],
|
||||
result: "2021-03-14T03:15:59.000-0400",
|
||||
},
|
||||
// 4:15am NY
|
||||
{
|
||||
desc: "UTC to 4:15am NY EDT",
|
||||
desc: "UTC Zulu to 4:15am NY EDT",
|
||||
inputs: ["2021-03-14T08:15:59.000Z", "America/New_York"],
|
||||
result: "2021-03-14T04:15:59.000-0400",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 4:15am NY EDT (2)",
|
||||
desc: "ISO+Offset to 4:15am NY EDT (2)",
|
||||
inputs: ["2021-03-14T04:15:59.000-0400", "America/New_York"],
|
||||
result: "2021-03-14T04:15:59.000-0400",
|
||||
},
|
||||
@ -106,57 +109,57 @@ function testTzToUtc(t) {
|
||||
|
||||
// 12:15am NY -0400 => -0500
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 12:15am NY EDT",
|
||||
desc: "UTC Zulu to 2021 Nov 7, 12:15am NY EDT",
|
||||
inputs: ["2021-11-07T04:15:59.000Z", "America/New_York"],
|
||||
result: "2021-11-07T00:15:59.000-0400",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 12:15am NY EDT (2)",
|
||||
desc: "ISO+Offset to 2021 Nov 7, 12:15am NY EDT (2)",
|
||||
inputs: ["2021-11-07T00:15:59.000-0400", "America/New_York"],
|
||||
result: "2021-11-07T00:15:59.000-0400",
|
||||
},
|
||||
// 1:15am NY (DST) -0400
|
||||
// NOTE: 1:15am happens TWICE (with different offsets)
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 1:15am NY EDT",
|
||||
desc: "UTC Zulu to 2021 Nov 7, 1:15am NY EDT",
|
||||
inputs: ["2021-11-07T05:15:59.000Z", "America/New_York"],
|
||||
result: "2021-11-07T01:15:59.000-0400",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 1:15am NY EDT (2)",
|
||||
desc: "ISO+Offset to 2021 Nov 7, 1:15am NY EDT (2)",
|
||||
inputs: ["2021-11-07T01:15:59.000-0400", "America/New_York"],
|
||||
result: "2021-11-07T01:15:59.000-0400",
|
||||
},
|
||||
// 1:15am NY (non-DST) -0500
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 1:15am NY EST",
|
||||
desc: "UTC Zulu to 2021 Nov 7, 1:15am NY EST",
|
||||
inputs: ["2021-11-07T06:15:59.000Z", "America/New_York"],
|
||||
result: "2021-11-07T01:15:59.000-0500",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 1:15am NY EST (2)",
|
||||
desc: "ISO+Offset to 2021 Nov 7, 1:15am NY EST (2)",
|
||||
inputs: ["2021-11-07T01:15:59.000-0500", "America/New_York"],
|
||||
result: "2021-11-07T01:15:59.000-0500",
|
||||
},
|
||||
// 2:15am NY -0500
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 2:15am NY EST",
|
||||
desc: "UTC Zulu to 2021 Nov 7, 2:15am NY EST",
|
||||
inputs: ["2021-11-07T07:15:59.000Z", "America/New_York"],
|
||||
result: "2021-11-07T02:15:59.000-0500",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 2:15am NY EST (2)",
|
||||
desc: "ISO+Offset to 2021 Nov 7, 2:15am NY EST (2)",
|
||||
inputs: ["2021-11-07T02:15:59.000-0500", "America/New_York"],
|
||||
result: "2021-11-07T02:15:59.000-0500",
|
||||
},
|
||||
// 3:15am NY
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 3:15am NY EST",
|
||||
desc: "UTC Zulu to 2021 Nov 7, 3:15am NY EST",
|
||||
inputs: ["2021-11-07T08:15:59.000Z", "America/New_York"],
|
||||
result: "2021-11-07T03:15:59.000-0500",
|
||||
},
|
||||
{
|
||||
desc: "UTC to 2021 Nov 7, 3:15am NY EST (2)",
|
||||
desc: "ISO+Offset to 2021 Nov 7, 3:15am NY EST (2)",
|
||||
inputs: ["2021-11-07T03:15:59.000-0500", "America/New_York"],
|
||||
result: "2021-11-07T03:15:59.000-0500",
|
||||
},
|
||||
@ -167,27 +170,29 @@ function testTzToUtc(t) {
|
||||
|
||||
// Colombo +0530 (not DST)
|
||||
{
|
||||
desc: "UTC to Asia/Colombo (1)",
|
||||
desc: "UTC Zulu to Asia/Colombo (1)",
|
||||
inputs: ["2021-03-14T08:15:59.000Z", "Asia/Colombo"],
|
||||
result: "2021-03-14T13:45:59.000+0530",
|
||||
},
|
||||
{
|
||||
desc: "UTC to Asia/Colombo (2)",
|
||||
desc: "ISO+Offset to Asia/Colombo (2)",
|
||||
inputs: ["2021-03-14T13:45:59.000+0530", "Asia/Colombo"],
|
||||
result: "2021-03-14T13:45:59.000+0530",
|
||||
},
|
||||
{
|
||||
desc: "UTC to Asia/Colombo (3)",
|
||||
desc: "UTC Zulu to Asia/Colombo (3)",
|
||||
inputs: ["2021-11-07T08:15:59.000Z", "Asia/Colombo"],
|
||||
result: "2021-11-07T13:45:59.000+0530",
|
||||
},
|
||||
{
|
||||
desc: "UTC to Asia/Colombo (4)",
|
||||
desc: "ISO+Offset to Asia/Colombo (4)",
|
||||
inputs: ["2021-11-07T13:45:59.000+0530", "Asia/Colombo"],
|
||||
result: "2021-11-07T13:45:59.000+0530",
|
||||
},
|
||||
].forEach(testUtcToTz);
|
||||
console.info("Pass: UTC to TZ for America/New_York and Asia/Colombo");
|
||||
console.info(
|
||||
"Pass: UTC/ISO+Offset to TZ for America/New_York and Asia/Colombo"
|
||||
);
|
||||
|
||||
[
|
||||
//
|
||||
@ -314,3 +319,16 @@ console.info("Pass: UTC to TZ for America/New_York and Asia/Colombo");
|
||||
},
|
||||
].forEach(testTzToUtc);
|
||||
console.info("Pass: TZ to UTC for America/New_York and Asia/Colombo");
|
||||
|
||||
var localISOString = TZ.toLocalISOString();
|
||||
var reISOString = /^\d\d\d\d-\d\d-\d\dT\d\d:\d\d:\d\d\.\d\d\d[+-]\d\d\d\d$/;
|
||||
if (!reISOString.test(localISOString)) {
|
||||
throw new Error("Couldn't get local time as iso+offset");
|
||||
}
|
||||
console.info("Pass: can get local time as ISO+Offset");
|
||||
|
||||
var tzName = TZ.timeZone();
|
||||
if (!/^[A-Z]\w+\/[A-Z]\w+$/.test(tzName)) {
|
||||
throw new Error("Couldn't get local Time Zone");
|
||||
}
|
||||
console.info("Pass: can get local timezone");
|
||||
|
||||
50
tolocalisostring.js
Normal file
50
tolocalisostring.js
Normal file
@ -0,0 +1,50 @@
|
||||
(function (exports) {
|
||||
"use strict";
|
||||
|
||||
// See <https://github.com/coolaj86/AJScript/issues/27>
|
||||
|
||||
function toLocalISOString(dateOrStr) {
|
||||
var d;
|
||||
if (dateOrStr) {
|
||||
d = new Date(dateOrStr);
|
||||
} else {
|
||||
d = new Date();
|
||||
}
|
||||
|
||||
var YYYY = d.getFullYear();
|
||||
var MM = p2(d.getMonth() + 1);
|
||||
var DD = p2(d.getDate());
|
||||
var hh = p2(d.getHours());
|
||||
var mm = p2(d.getMinutes());
|
||||
var ss = p2(d.getSeconds());
|
||||
var sss = d.getMilliseconds().toString().padStart(3, "0");
|
||||
|
||||
var offset = formatOffset(-d.getTimezoneOffset());
|
||||
|
||||
return `${YYYY}-${MM}-${DD}T${hh}:${mm}:${ss}.${sss}${offset}`;
|
||||
}
|
||||
|
||||
function formatOffset(minutes) {
|
||||
if (!minutes) {
|
||||
return "Z";
|
||||
}
|
||||
|
||||
var h = Math.floor(Math.abs(minutes) / 60);
|
||||
var m = Math.abs(minutes) % 60;
|
||||
var offset = "";
|
||||
if (minutes > 0) {
|
||||
offset = "+";
|
||||
} else if (minutes < 0) {
|
||||
offset = "-";
|
||||
}
|
||||
|
||||
// +0500, -0730
|
||||
return offset + p2(h) + p2(m);
|
||||
}
|
||||
|
||||
function p2(x) {
|
||||
return String(x).padStart(2, "0");
|
||||
}
|
||||
|
||||
exports.toLocalISOString = toLocalISOString;
|
||||
})(("undefined" === typeof module && window) || exports);
|
||||
70
xtz.js
70
xtz.js
@ -1,6 +1,4 @@
|
||||
var XTZ;
|
||||
|
||||
(function () {
|
||||
(function (exports) {
|
||||
"use strict";
|
||||
|
||||
function toTimeZone(date, timeZone) {
|
||||
@ -62,15 +60,16 @@ var XTZ;
|
||||
|
||||
function toTimeZoneISOString(date, timeZone) {
|
||||
var whole = toTimeZone(date, timeZone);
|
||||
return toOffsetISOString(whole);
|
||||
return formatAsOffsetISOString(whole);
|
||||
}
|
||||
|
||||
function _toOffsetISOString() {
|
||||
return toOffsetISOString(this);
|
||||
/* jshint validthis: true */
|
||||
return formatAsOffsetISOString(this);
|
||||
}
|
||||
|
||||
function getOffset(utcDate, tzD2) {
|
||||
var tzDate = new Date(toOffsetISOString(tzD2));
|
||||
var tzDate = new Date(formatAsOffsetISOString(tzD2));
|
||||
var diff = Math.round((tzDate.valueOf() - utcDate.valueOf()) / (60 * 1000));
|
||||
return diff;
|
||||
}
|
||||
@ -103,7 +102,16 @@ var XTZ;
|
||||
);
|
||||
}
|
||||
|
||||
function toOffsetISOString(d) {
|
||||
function toOffsetISOString(date, timeZone) {
|
||||
if ("object" === typeof date && "offset" in date && "year" in date) {
|
||||
return formatAsOffsetISOString(date);
|
||||
}
|
||||
|
||||
var whole = fromTimeZone(date, timeZone);
|
||||
return formatAsOffsetISOString(whole);
|
||||
}
|
||||
|
||||
function formatAsOffsetISOString(d) {
|
||||
var offset = formatOffset(d.offset);
|
||||
return (
|
||||
`${d.year}-${p2(d.month + 1)}-${p2(d.day)}` +
|
||||
@ -113,7 +121,7 @@ var XTZ;
|
||||
);
|
||||
}
|
||||
|
||||
function toUTC(dt, tz) {
|
||||
function fromTimeZone(dt, tz) {
|
||||
if ("string" === typeof dt) {
|
||||
// Either of these formats should work:
|
||||
// 2021-03-14 01:15:59
|
||||
@ -150,19 +158,47 @@ var XTZ;
|
||||
return tzD3;
|
||||
}
|
||||
|
||||
function toUTCISOString(date, timeZone) {
|
||||
var whole = toUTC(date, timeZone);
|
||||
return toOffsetISOString(whole);
|
||||
function toLocalISOString(dateOrStr) {
|
||||
var d;
|
||||
if (dateOrStr) {
|
||||
d = new Date(dateOrStr);
|
||||
} else {
|
||||
d = new Date();
|
||||
}
|
||||
|
||||
var YYYY = d.getFullYear();
|
||||
var MM = p2(d.getMonth() + 1);
|
||||
var DD = p2(d.getDate());
|
||||
var hh = p2(d.getHours());
|
||||
var mm = p2(d.getMinutes());
|
||||
var ss = p2(d.getSeconds());
|
||||
var sss = d.getMilliseconds().toString().padStart(3, "0");
|
||||
|
||||
var offset = formatOffset(-d.getTimezoneOffset());
|
||||
|
||||
return `${YYYY}-${MM}-${DD}T${hh}:${mm}:${ss}.${sss}${offset}`;
|
||||
}
|
||||
|
||||
XTZ = {
|
||||
function getTimeZone() {
|
||||
return new Intl.DateTimeFormat().resolvedOptions().timeZone;
|
||||
}
|
||||
|
||||
exports.XTZ = {
|
||||
// bespoke date =>
|
||||
// 2021-11-07T3:15:59-0500
|
||||
// (todo?)
|
||||
// xtzToISOString: formatAsOffsetISOString,
|
||||
// (deprecated)
|
||||
toOffsetISOString: toOffsetISOString,
|
||||
|
||||
// -240 => -0400
|
||||
formatOffset: formatOffset,
|
||||
|
||||
// "America/New_York"
|
||||
timeZone: getTimeZone,
|
||||
// "2021-11-07T03:15:59-0500"
|
||||
toLocalISOString: toLocalISOString,
|
||||
|
||||
// [ "2021-11-07T08:15:59Z", "America/New_York" ]
|
||||
// => "2021-11-07T03:15:59-0500" // 2021-11-07 03:15:59
|
||||
toTimeZone: toTimeZone,
|
||||
@ -170,11 +206,13 @@ var XTZ;
|
||||
|
||||
// [ "2021-11-07 03:15:59", "America/New_York" ]
|
||||
// => "2021-11-07T03:15:59-0500" // 2021-11-07T08:15:59Z
|
||||
toUTC: toUTC,
|
||||
toUTCISOString: toUTCISOString,
|
||||
toUTC: fromTimeZone,
|
||||
fromTimeZone: fromTimeZone,
|
||||
// deprecated
|
||||
toUTCISOString: toOffsetISOString,
|
||||
};
|
||||
|
||||
if ("undefined" != typeof module && module.exports) {
|
||||
module.exports = XTZ;
|
||||
module.exports = exports.XTZ;
|
||||
}
|
||||
})();
|
||||
})(("undefined" === typeof module && window) || exports);
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user