mirror of
https://github.com/RGBCube/serenity
synced 2025-07-26 12:17:35 +00:00
LibJS: Implement Temporal.TimeZone.prototype.getOffsetNanosecondsFor()
This commit is contained in:
parent
d428787e18
commit
96e63415b6
6 changed files with 61 additions and 0 deletions
|
@ -182,6 +182,7 @@ namespace JS {
|
||||||
P(getMilliseconds) \
|
P(getMilliseconds) \
|
||||||
P(getMinutes) \
|
P(getMinutes) \
|
||||||
P(getMonth) \
|
P(getMonth) \
|
||||||
|
P(getOffsetNanosecondsFor) \
|
||||||
P(getOwnPropertyDescriptor) \
|
P(getOwnPropertyDescriptor) \
|
||||||
P(getOwnPropertyDescriptors) \
|
P(getOwnPropertyDescriptors) \
|
||||||
P(getOwnPropertyNames) \
|
P(getOwnPropertyNames) \
|
||||||
|
|
|
@ -92,6 +92,15 @@ TimeZone* create_temporal_time_zone(GlobalObject& global_object, String const& i
|
||||||
return object;
|
return object;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 11.6.5 GetIANATimeZoneOffsetNanoseconds ( epochNanoseconds, timeZoneIdentifier ), https://tc39.es/proposal-temporal/#sec-temporal-getianatimezoneoffsetnanoseconds
|
||||||
|
i64 get_iana_time_zone_offset_nanoseconds([[maybe_unused]] BigInt const& epoch_nanoseconds, [[maybe_unused]] String const& time_zone_identifier)
|
||||||
|
{
|
||||||
|
// The abstract operation GetIANATimeZoneOffsetNanoseconds is an implementation-defined algorithm that returns an integer representing the offset of the IANA time zone identified by timeZoneIdentifier from UTC, at the instant corresponding to epochNanoseconds.
|
||||||
|
// Given the same values of epochNanoseconds and timeZoneIdentifier, the result must be the same for the lifetime of the surrounding agent.
|
||||||
|
// TODO: Implement this
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
// https://tc39.es/proposal-temporal/#prod-TimeZoneNumericUTCOffset
|
// https://tc39.es/proposal-temporal/#prod-TimeZoneNumericUTCOffset
|
||||||
static bool parse_time_zone_numeric_utc_offset_syntax(String const& offset_string, StringView& sign, StringView& hours, Optional<StringView>& minutes, Optional<StringView>& seconds, Optional<StringView>& fraction)
|
static bool parse_time_zone_numeric_utc_offset_syntax(String const& offset_string, StringView& sign, StringView& hours, Optional<StringView>& minutes, Optional<StringView>& seconds, Optional<StringView>& fraction)
|
||||||
{
|
{
|
||||||
|
|
|
@ -39,6 +39,7 @@ bool is_valid_time_zone_name(String const& time_zone);
|
||||||
String canonicalize_time_zone_name(String const& time_zone);
|
String canonicalize_time_zone_name(String const& time_zone);
|
||||||
String default_time_zone();
|
String default_time_zone();
|
||||||
TimeZone* create_temporal_time_zone(GlobalObject&, String const& identifier, FunctionObject* new_target = nullptr);
|
TimeZone* create_temporal_time_zone(GlobalObject&, String const& identifier, FunctionObject* new_target = nullptr);
|
||||||
|
i64 get_iana_time_zone_offset_nanoseconds(BigInt const& epoch_nanoseconds, String const& time_zone_identifier);
|
||||||
double parse_time_zone_offset_string(GlobalObject&, String const&);
|
double parse_time_zone_offset_string(GlobalObject&, String const&);
|
||||||
String format_time_zone_offset_string(double offset_nanoseconds);
|
String format_time_zone_offset_string(double offset_nanoseconds);
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <LibJS/Runtime/GlobalObject.h>
|
#include <LibJS/Runtime/GlobalObject.h>
|
||||||
|
#include <LibJS/Runtime/Temporal/Instant.h>
|
||||||
#include <LibJS/Runtime/Temporal/TimeZone.h>
|
#include <LibJS/Runtime/Temporal/TimeZone.h>
|
||||||
#include <LibJS/Runtime/Temporal/TimeZonePrototype.h>
|
#include <LibJS/Runtime/Temporal/TimeZonePrototype.h>
|
||||||
|
|
||||||
|
@ -24,6 +25,7 @@ void TimeZonePrototype::initialize(GlobalObject& global_object)
|
||||||
|
|
||||||
u8 attr = Attribute::Writable | Attribute::Configurable;
|
u8 attr = Attribute::Writable | Attribute::Configurable;
|
||||||
define_native_accessor(vm.names.id, id_getter, {}, Attribute::Configurable);
|
define_native_accessor(vm.names.id, id_getter, {}, Attribute::Configurable);
|
||||||
|
define_native_function(vm.names.getOffsetNanosecondsFor, get_offset_nanoseconds_for, 1, attr);
|
||||||
define_native_function(vm.names.toString, to_string, 0, attr);
|
define_native_function(vm.names.toString, to_string, 0, attr);
|
||||||
define_native_function(vm.names.toJSON, to_json, 0, attr);
|
define_native_function(vm.names.toJSON, to_json, 0, attr);
|
||||||
|
|
||||||
|
@ -54,6 +56,28 @@ JS_DEFINE_NATIVE_FUNCTION(TimeZonePrototype::id_getter)
|
||||||
return js_string(vm, time_zone.to_string(global_object));
|
return js_string(vm, time_zone.to_string(global_object));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 11.4.4 Temporal.TimeZone.prototype.getOffsetNanosecondsFor ( instant ), https://tc39.es/proposal-temporal/#sec-temporal.timezone.prototype.getoffsetnanosecondsfor
|
||||||
|
JS_DEFINE_NATIVE_FUNCTION(TimeZonePrototype::get_offset_nanoseconds_for)
|
||||||
|
{
|
||||||
|
// 1. Let timeZone be the this value.
|
||||||
|
// 2. Perform ? RequireInternalSlot(timeZone, [[InitializedTemporalTimeZone]]).
|
||||||
|
auto* time_zone = typed_this(global_object);
|
||||||
|
if (vm.exception())
|
||||||
|
return {};
|
||||||
|
|
||||||
|
// 3. Set instant to ? ToTemporalInstant(instant).
|
||||||
|
auto* instant = to_temporal_instant(global_object, vm.argument(0));
|
||||||
|
if (vm.exception())
|
||||||
|
return {};
|
||||||
|
|
||||||
|
// 4. If timeZone.[[OffsetNanoseconds]] is not undefined, return timeZone.[[OffsetNanoseconds]].
|
||||||
|
if (time_zone->offset_nanoseconds().has_value())
|
||||||
|
return Value(*time_zone->offset_nanoseconds());
|
||||||
|
|
||||||
|
// 5. Return ! GetIANATimeZoneOffsetNanoseconds(instant.[[Nanoseconds]], timeZone.[[Identifier]]).
|
||||||
|
return Value((double)get_iana_time_zone_offset_nanoseconds(instant->nanoseconds(), time_zone->identifier()));
|
||||||
|
}
|
||||||
|
|
||||||
// 11.4.11 Temporal.TimeZone.prototype.toString ( ), https://tc39.es/proposal-temporal/#sec-temporal.timezone.prototype.tostring
|
// 11.4.11 Temporal.TimeZone.prototype.toString ( ), https://tc39.es/proposal-temporal/#sec-temporal.timezone.prototype.tostring
|
||||||
JS_DEFINE_NATIVE_FUNCTION(TimeZonePrototype::to_string)
|
JS_DEFINE_NATIVE_FUNCTION(TimeZonePrototype::to_string)
|
||||||
{
|
{
|
||||||
|
|
|
@ -20,6 +20,7 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
JS_DECLARE_NATIVE_FUNCTION(id_getter);
|
JS_DECLARE_NATIVE_FUNCTION(id_getter);
|
||||||
|
JS_DECLARE_NATIVE_FUNCTION(get_offset_nanoseconds_for);
|
||||||
JS_DECLARE_NATIVE_FUNCTION(to_string);
|
JS_DECLARE_NATIVE_FUNCTION(to_string);
|
||||||
JS_DECLARE_NATIVE_FUNCTION(to_json);
|
JS_DECLARE_NATIVE_FUNCTION(to_json);
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
describe("correct behavior", () => {
|
||||||
|
test("length is 1", () => {
|
||||||
|
expect(Temporal.TimeZone.prototype.getOffsetNanosecondsFor).toHaveLength(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("basic functionality", () => {
|
||||||
|
const timeZone = new Temporal.TimeZone("UTC");
|
||||||
|
const instant = new Temporal.Instant(0n);
|
||||||
|
expect(timeZone.getOffsetNanosecondsFor(instant)).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("custom offset", () => {
|
||||||
|
const timeZone = new Temporal.TimeZone("+01:30");
|
||||||
|
const instant = new Temporal.Instant(0n);
|
||||||
|
expect(timeZone.getOffsetNanosecondsFor(instant)).toBe(5400000000000);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("errors", () => {
|
||||||
|
test("this value must be a Temporal.TimeZone object", () => {
|
||||||
|
expect(() => {
|
||||||
|
Temporal.TimeZone.prototype.getOffsetNanosecondsFor.call("foo");
|
||||||
|
}).toThrowWithMessage(TypeError, "Not a Temporal.TimeZone");
|
||||||
|
});
|
||||||
|
});
|
Loading…
Add table
Add a link
Reference in a new issue