
Instead we just use a specific constructor. With this set of constructors using curly braces for constructing is highly recommended. As then it will not do too many implicit conversions which could lead to unexpected loss of data or calling the much slower double constructor. Also to ensure we don't feed (Un)SignedBigInteger infinities we throw RangeError earlier for Durations.
237 lines
9.8 KiB
C++
237 lines
9.8 KiB
C++
/*
|
||
* Copyright (c) 2021-2022, Linus Groh <linusg@serenityos.org>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include <AK/Time.h>
|
||
#include <LibCrypto/BigInt/SignedBigInteger.h>
|
||
#include <LibJS/Runtime/Completion.h>
|
||
#include <LibJS/Runtime/GlobalObject.h>
|
||
#include <LibJS/Runtime/Temporal/Calendar.h>
|
||
#include <LibJS/Runtime/Temporal/Instant.h>
|
||
#include <LibJS/Runtime/Temporal/Now.h>
|
||
#include <LibJS/Runtime/Temporal/PlainDate.h>
|
||
#include <LibJS/Runtime/Temporal/PlainDateTime.h>
|
||
#include <LibJS/Runtime/Temporal/PlainTime.h>
|
||
#include <LibJS/Runtime/Temporal/TimeZone.h>
|
||
#include <LibJS/Runtime/Temporal/ZonedDateTime.h>
|
||
|
||
namespace JS::Temporal {
|
||
|
||
// 2 The Temporal.Now Object, https://tc39.es/proposal-temporal/#sec-temporal-now-object
|
||
Now::Now(Realm& realm)
|
||
: Object(*realm.global_object().object_prototype())
|
||
{
|
||
}
|
||
|
||
void Now::initialize(Realm& realm)
|
||
{
|
||
Object::initialize(realm);
|
||
|
||
auto& vm = this->vm();
|
||
|
||
// 2.1.1 Temporal.Now [ @@toStringTag ], https://tc39.es/proposal-temporal/#sec-temporal-now-@@tostringtag
|
||
define_direct_property(*vm.well_known_symbol_to_string_tag(), js_string(vm, "Temporal.Now"), Attribute::Configurable);
|
||
|
||
u8 attr = Attribute::Writable | Attribute::Configurable;
|
||
define_native_function(realm, vm.names.timeZone, time_zone, 0, attr);
|
||
define_native_function(realm, vm.names.instant, instant, 0, attr);
|
||
define_native_function(realm, vm.names.plainDateTime, plain_date_time, 1, attr);
|
||
define_native_function(realm, vm.names.plainDateTimeISO, plain_date_time_iso, 0, attr);
|
||
define_native_function(realm, vm.names.zonedDateTime, zoned_date_time, 1, attr);
|
||
define_native_function(realm, vm.names.zonedDateTimeISO, zoned_date_time_iso, 0, attr);
|
||
define_native_function(realm, vm.names.plainDate, plain_date, 1, attr);
|
||
define_native_function(realm, vm.names.plainDateISO, plain_date_iso, 0, attr);
|
||
define_native_function(realm, vm.names.plainTimeISO, plain_time_iso, 0, attr);
|
||
}
|
||
|
||
// 2.2.1 Temporal.Now.timeZone ( ), https://tc39.es/proposal-temporal/#sec-temporal.now.timezone
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::time_zone)
|
||
{
|
||
// 1. Return ! SystemTimeZone().
|
||
return system_time_zone(vm);
|
||
}
|
||
|
||
// 2.2.2 Temporal.Now.instant ( ), https://tc39.es/proposal-temporal/#sec-temporal.now.instant
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::instant)
|
||
{
|
||
// 1. Return ! SystemInstant().
|
||
return system_instant(vm);
|
||
}
|
||
|
||
// 2.2.3 Temporal.Now.plainDateTime ( calendarLike [ , temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindatetime
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date_time)
|
||
{
|
||
auto calendar_like = vm.argument(0);
|
||
auto temporal_time_zone_like = vm.argument(1);
|
||
|
||
// 1. Return ? SystemDateTime(temporalTimeZoneLike, calendarLike).
|
||
return TRY(system_date_time(vm, temporal_time_zone_like, calendar_like));
|
||
}
|
||
|
||
// 2.2.4 Temporal.Now.plainDateTimeISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindatetimeiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date_time_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1, Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Return ? SystemDateTime(temporalTimeZoneLike, calendar).
|
||
return TRY(system_date_time(vm, temporal_time_zone_like, calendar));
|
||
}
|
||
|
||
// 2.2.5 Temporal.Now.zonedDateTime ( calendarLike [ , temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.zoneddatetime
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::zoned_date_time)
|
||
{
|
||
auto calendar_like = vm.argument(0);
|
||
auto temporal_time_zone_like = vm.argument(1);
|
||
|
||
// 1. Return ? SystemZonedDateTime(temporalTimeZoneLike, calendarLike).
|
||
return TRY(system_zoned_date_time(vm, temporal_time_zone_like, calendar_like));
|
||
}
|
||
|
||
// 2.2.6 Temporal.Now.zonedDateTimeISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.zoneddatetimeiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::zoned_date_time_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1, Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Return ? SystemZonedDateTime(temporalTimeZoneLike, calendar).
|
||
return TRY(system_zoned_date_time(vm, temporal_time_zone_like, calendar));
|
||
}
|
||
|
||
// 2.2.7 Temporal.Now.plainDate ( calendarLike [ , temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindate
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date)
|
||
{
|
||
auto calendar_like = vm.argument(0);
|
||
auto temporal_time_zone_like = vm.argument(1);
|
||
|
||
// 1. Let dateTime be ? SystemDateTime(temporalTimeZoneLike, calendarLike).
|
||
auto* date_time = TRY(system_date_time(vm, temporal_time_zone_like, calendar_like));
|
||
|
||
// 2. Return ! CreateTemporalDate(dateTime.[[ISOYear]], dateTime.[[ISOMonth]], dateTime.[[ISODay]], dateTime.[[Calendar]]).
|
||
return MUST(create_temporal_date(vm, date_time->iso_year(), date_time->iso_month(), date_time->iso_day(), date_time->calendar()));
|
||
}
|
||
|
||
// 2.2.8 Temporal.Now.plainDateISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindateiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1. Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Let dateTime be ? SystemDateTime(temporalTimeZoneLike, calendar).
|
||
auto* date_time = TRY(system_date_time(vm, temporal_time_zone_like, calendar));
|
||
|
||
// 3. Return ! CreateTemporalDate(dateTime.[[ISOYear]], dateTime.[[ISOMonth]], dateTime.[[ISODay]], dateTime.[[Calendar]]).
|
||
return MUST(create_temporal_date(vm, date_time->iso_year(), date_time->iso_month(), date_time->iso_day(), date_time->calendar()));
|
||
}
|
||
|
||
// 2.2.9 Temporal.Now.plainTimeISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaintimeiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_time_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1. Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Let dateTime be ? SystemDateTime(temporalTimeZoneLike, calendar).
|
||
auto* date_time = TRY(system_date_time(vm, temporal_time_zone_like, calendar));
|
||
|
||
// 3. Return ! CreateTemporalTime(dateTime.[[ISOHour]], dateTime.[[ISOMinute]], dateTime.[[ISOSecond]], dateTime.[[ISOMillisecond]], dateTime.[[ISOMicrosecond]], dateTime.[[ISONanosecond]]).
|
||
return MUST(create_temporal_time(vm, date_time->iso_hour(), date_time->iso_minute(), date_time->iso_second(), date_time->iso_millisecond(), date_time->iso_microsecond(), date_time->iso_nanosecond()));
|
||
}
|
||
|
||
// 2.3.1 SystemTimeZone ( ), https://tc39.es/proposal-temporal/#sec-temporal-systemtimezone
|
||
TimeZone* system_time_zone(VM& vm)
|
||
{
|
||
// 1. Let identifier be ! DefaultTimeZone().
|
||
auto identifier = default_time_zone();
|
||
|
||
// 2. Return ! CreateTemporalTimeZone(identifier).
|
||
return MUST(create_temporal_time_zone(vm, identifier));
|
||
}
|
||
|
||
// 2.3.2 SystemUTCEpochNanoseconds ( ), https://tc39.es/proposal-temporal/#sec-temporal-systemutcepochnanoseconds
|
||
BigInt* system_utc_epoch_nanoseconds(VM& vm)
|
||
{
|
||
// 1. Let ns be the approximate current UTC date and time, in nanoseconds since the epoch.
|
||
auto now = Time::now_realtime().to_nanoseconds();
|
||
auto ns = Crypto::SignedBigInteger { now };
|
||
|
||
// 2. Set ns to the result of clamping ns between nsMinInstant and nsMaxInstant.
|
||
// NOTE: Time::to_nanoseconds() already clamps between -(2^63) and 2^63 - 1, the range of an i64,
|
||
// if an overflow occurs during seconds -> nanoseconds conversion.
|
||
|
||
// 3. Return ℤ(ns).
|
||
return js_bigint(vm, move(ns));
|
||
}
|
||
|
||
// 2.3.3 SystemInstant ( ), https://tc39.es/proposal-temporal/#sec-temporal-systeminstant
|
||
Instant* system_instant(VM& vm)
|
||
{
|
||
// 1. Let ns be ! SystemUTCEpochNanoseconds().
|
||
auto* ns = system_utc_epoch_nanoseconds(vm);
|
||
|
||
// 2. Return ! CreateTemporalInstant(ns).
|
||
return MUST(create_temporal_instant(vm, *ns));
|
||
}
|
||
|
||
// 2.3.4 SystemDateTime ( temporalTimeZoneLike, calendarLike ), https://tc39.es/proposal-temporal/#sec-temporal-systemdatetime
|
||
ThrowCompletionOr<PlainDateTime*> system_date_time(VM& vm, Value temporal_time_zone_like, Value calendar_like)
|
||
{
|
||
Object* time_zone;
|
||
|
||
// 1. If temporalTimeZoneLike is undefined, then
|
||
if (temporal_time_zone_like.is_undefined()) {
|
||
// a. Let timeZone be ! SystemTimeZone().
|
||
time_zone = system_time_zone(vm);
|
||
}
|
||
// 2. Else,
|
||
else {
|
||
// a. Let timeZone be ? ToTemporalTimeZone(temporalTimeZoneLike).
|
||
time_zone = TRY(to_temporal_time_zone(vm, temporal_time_zone_like));
|
||
}
|
||
|
||
// 3. Let calendar be ? ToTemporalCalendar(calendarLike).
|
||
auto* calendar = TRY(to_temporal_calendar(vm, calendar_like));
|
||
|
||
// 4. Let instant be ! SystemInstant().
|
||
auto* instant = system_instant(vm);
|
||
|
||
// 5. Return ? BuiltinTimeZoneGetPlainDateTimeFor(timeZone, instant, calendar).
|
||
return builtin_time_zone_get_plain_date_time_for(vm, time_zone, *instant, *calendar);
|
||
}
|
||
|
||
// 2.3.5 SystemZonedDateTime ( temporalTimeZoneLike, calendarLike ), https://tc39.es/proposal-temporal/#sec-temporal-systemzoneddatetime
|
||
ThrowCompletionOr<ZonedDateTime*> system_zoned_date_time(VM& vm, Value temporal_time_zone_like, Value calendar_like)
|
||
{
|
||
Object* time_zone;
|
||
|
||
// 1. If temporalTimeZoneLike is undefined, then
|
||
if (temporal_time_zone_like.is_undefined()) {
|
||
// a. Let timeZone be ! SystemTimeZone().
|
||
time_zone = system_time_zone(vm);
|
||
}
|
||
// 2. Else,
|
||
else {
|
||
// a. Let timeZone be ? ToTemporalTimeZone(temporalTimeZoneLike).
|
||
time_zone = TRY(to_temporal_time_zone(vm, temporal_time_zone_like));
|
||
}
|
||
|
||
// 3. Let calendar be ? ToTemporalCalendar(calendarLike).
|
||
auto* calendar = TRY(to_temporal_calendar(vm, calendar_like));
|
||
|
||
// 4. Let ns be ! SystemUTCEpochNanoseconds().
|
||
auto* ns = system_utc_epoch_nanoseconds(vm);
|
||
|
||
// 5. Return ? CreateTemporalZonedDateTime(ns, timeZone, calendar).
|
||
return create_temporal_zoned_date_time(vm, *ns, *time_zone, *calendar);
|
||
}
|
||
|
||
}
|