2021-07-11 21:04:11 +03:00
|
|
|
|
/*
|
|
|
|
|
|
* Copyright (c) 2021, Idan Horowitz <idan.horowitz@serenityos.org>
|
2023-01-26 12:07:38 +00:00
|
|
|
|
* Copyright (c) 2021-2023, Linus Groh <linusg@serenityos.org>
|
2024-11-16 13:14:14 -05:00
|
|
|
|
* Copyright (c) 2024, Tim Flynn <trflynn89@ladybird.org>
|
2021-07-11 21:04:11 +03:00
|
|
|
|
*
|
|
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
2021-08-31 00:15:36 +01:00
|
|
|
|
#include <AK/Variant.h>
|
2024-11-18 10:36:03 -05:00
|
|
|
|
#include <LibGC/Ptr.h>
|
2021-07-18 21:44:05 +01:00
|
|
|
|
#include <LibJS/Forward.h>
|
2021-09-15 23:03:38 +01:00
|
|
|
|
#include <LibJS/Runtime/Completion.h>
|
2024-11-16 12:51:53 -05:00
|
|
|
|
#include <LibJS/Runtime/VM.h>
|
2023-10-06 17:54:21 +02:00
|
|
|
|
#include <LibJS/Runtime/ValueInlines.h>
|
2024-11-18 11:58:51 -05:00
|
|
|
|
#include <math.h>
|
2021-07-11 21:04:11 +03:00
|
|
|
|
|
|
|
|
|
|
namespace JS::Temporal {
|
|
|
|
|
|
|
2024-11-18 12:48:10 -05:00
|
|
|
|
enum class ArithmeticOperation {
|
|
|
|
|
|
Add,
|
|
|
|
|
|
Subtract,
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2024-11-18 11:58:51 -05:00
|
|
|
|
// https://tc39.es/proposal-temporal/#sec-temporal-units
|
|
|
|
|
|
enum class Unit {
|
|
|
|
|
|
Year,
|
|
|
|
|
|
Month,
|
|
|
|
|
|
Week,
|
|
|
|
|
|
Day,
|
|
|
|
|
|
Hour,
|
|
|
|
|
|
Minute,
|
|
|
|
|
|
Second,
|
|
|
|
|
|
Millisecond,
|
|
|
|
|
|
Microsecond,
|
|
|
|
|
|
Nanosecond,
|
|
|
|
|
|
};
|
|
|
|
|
|
StringView temporal_unit_to_string(Unit);
|
|
|
|
|
|
|
|
|
|
|
|
// https://tc39.es/proposal-temporal/#sec-temporal-units
|
|
|
|
|
|
enum class UnitCategory {
|
|
|
|
|
|
Date,
|
|
|
|
|
|
Time,
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// https://tc39.es/proposal-temporal/#sec-temporal-units
|
|
|
|
|
|
enum class UnitGroup {
|
|
|
|
|
|
Date,
|
|
|
|
|
|
Time,
|
|
|
|
|
|
DateTime,
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct Unset { };
|
|
|
|
|
|
using RoundingIncrement = Variant<Unset, u64>;
|
|
|
|
|
|
|
|
|
|
|
|
struct RelativeTo {
|
|
|
|
|
|
// FIXME: Make these objects represent their actual types when we re-implement them.
|
|
|
|
|
|
GC::Ptr<JS::Object> plain_relative_to; // [[PlainRelativeTo]]
|
|
|
|
|
|
GC::Ptr<JS::Object> zoned_relative_to; // [[ZonedRelativeTo]]
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
ThrowCompletionOr<RelativeTo> get_temporal_relative_to_option(VM&, Object const& options);
|
2024-11-18 12:48:10 -05:00
|
|
|
|
Unit larger_of_two_temporal_units(Unit, Unit);
|
2024-11-18 11:58:51 -05:00
|
|
|
|
bool is_calendar_unit(Unit);
|
|
|
|
|
|
UnitCategory temporal_unit_category(Unit);
|
|
|
|
|
|
ThrowCompletionOr<GC::Ref<Duration>> parse_temporal_duration_string(VM&, StringView iso_string);
|
|
|
|
|
|
|
|
|
|
|
|
// 13.38 ToIntegerWithTruncation ( argument ), https://tc39.es/proposal-temporal/#sec-tointegerwithtruncation
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
|
|
ThrowCompletionOr<double> to_integer_with_truncation(VM& vm, Value argument, ErrorType error_type, Args&&... args)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 1. Let number be ? ToNumber(argument).
|
|
|
|
|
|
auto number = TRY(argument.to_number(vm));
|
|
|
|
|
|
|
|
|
|
|
|
// 2. If number is NaN, +∞𝔽 or -∞𝔽, throw a RangeError exception.
|
|
|
|
|
|
if (number.is_nan() || number.is_infinity())
|
|
|
|
|
|
return vm.throw_completion<RangeError>(error_type, forward<Args>(args)...);
|
|
|
|
|
|
|
|
|
|
|
|
// 3. Return truncate(ℝ(number)).
|
|
|
|
|
|
return trunc(number.as_double());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 13.38 ToIntegerWithTruncation ( argument ), https://tc39.es/proposal-temporal/#sec-tointegerwithtruncation
|
|
|
|
|
|
// AD-HOC: We often need to use this AO when we have a parsed StringView. This overload allows callers to avoid creating
|
|
|
|
|
|
// a PrimitiveString for the primary definition.
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
|
|
ThrowCompletionOr<double> to_integer_with_truncation(VM& vm, StringView argument, ErrorType error_type, Args&&... args)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 1. Let number be ? ToNumber(argument).
|
|
|
|
|
|
auto number = string_to_number(argument);
|
|
|
|
|
|
|
|
|
|
|
|
// 2. If number is NaN, +∞𝔽 or -∞𝔽, throw a RangeError exception.
|
|
|
|
|
|
if (isnan(number) || isinf(number))
|
|
|
|
|
|
return vm.throw_completion<RangeError>(error_type, forward<Args>(args)...);
|
|
|
|
|
|
|
|
|
|
|
|
// 3. Return truncate(ℝ(number)).
|
|
|
|
|
|
return trunc(number);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-16 13:14:14 -05:00
|
|
|
|
// 13.39 ToIntegerIfIntegral ( argument ), https://tc39.es/proposal-temporal/#sec-tointegerifintegral
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
|
|
ThrowCompletionOr<double> to_integer_if_integral(VM& vm, Value argument, ErrorType error_type, Args&&... args)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 1. Let number be ? ToNumber(argument).
|
|
|
|
|
|
auto number = TRY(argument.to_number(vm));
|
|
|
|
|
|
|
|
|
|
|
|
// 2. If number is not an integral Number, throw a RangeError exception.
|
|
|
|
|
|
if (!number.is_integral_number())
|
|
|
|
|
|
return vm.throw_completion<RangeError>(error_type, forward<Args>(args)...);
|
|
|
|
|
|
|
|
|
|
|
|
// 3. Return ℝ(number).
|
|
|
|
|
|
return number.as_double();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2021-07-11 22:28:49 +03:00
|
|
|
|
enum class OptionType {
|
|
|
|
|
|
Boolean,
|
|
|
|
|
|
String,
|
|
|
|
|
|
};
|
2021-07-18 21:41:29 +01:00
|
|
|
|
|
2024-11-16 13:14:14 -05:00
|
|
|
|
struct DefaultRequired { };
|
|
|
|
|
|
using OptionDefault = Variant<DefaultRequired, Empty, bool, StringView, double>;
|
2022-06-09 22:42:42 +01:00
|
|
|
|
|
2024-11-18 10:36:03 -05:00
|
|
|
|
ThrowCompletionOr<GC::Ref<Object>> get_options_object(VM&, Value options);
|
2023-02-05 19:02:54 +00:00
|
|
|
|
ThrowCompletionOr<Value> get_option(VM&, Object const& options, PropertyKey const& property, OptionType type, ReadonlySpan<StringView> values, OptionDefault const&);
|
2021-07-11 21:04:11 +03:00
|
|
|
|
|
2022-06-14 23:03:25 +01:00
|
|
|
|
template<size_t Size>
|
2022-08-20 08:52:42 +01:00
|
|
|
|
ThrowCompletionOr<Value> get_option(VM& vm, Object const& options, PropertyKey const& property, OptionType type, StringView const (&values)[Size], OptionDefault const& default_)
|
2022-06-14 23:03:25 +01:00
|
|
|
|
{
|
2023-02-05 19:02:54 +00:00
|
|
|
|
return get_option(vm, options, property, type, ReadonlySpan<StringView> { values }, default_);
|
2022-06-14 23:03:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
2021-07-11 21:04:11 +03:00
|
|
|
|
}
|