DurationFormat.h 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. /*
  2. * Copyright (c) 2022, Idan Horowitz <idan.horowitz@serenityos.org>
  3. * Copyright (c) 2022-2024, Tim Flynn <trflynn89@ladybird.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/Array.h>
  9. #include <AK/String.h>
  10. #include <LibCrypto/BigFraction/BigFraction.h>
  11. #include <LibJS/Runtime/Intl/AbstractOperations.h>
  12. #include <LibJS/Runtime/Object.h>
  13. #include <LibUnicode/Locale.h>
  14. namespace JS::Intl {
  15. class DurationFormat final : public Object {
  16. JS_OBJECT(DurationFormat, Object);
  17. GC_DECLARE_ALLOCATOR(DurationFormat);
  18. public:
  19. enum class Style {
  20. Long,
  21. Short,
  22. Narrow,
  23. Digital,
  24. };
  25. static Style style_from_string(StringView style);
  26. static StringView style_to_string(Style);
  27. enum class ValueStyle {
  28. Long,
  29. Short,
  30. Narrow,
  31. Numeric,
  32. TwoDigit,
  33. Fractional,
  34. };
  35. static ValueStyle value_style_from_string(StringView);
  36. static StringView value_style_to_string(ValueStyle);
  37. static_assert(to_underlying(ValueStyle::Long) == to_underlying(Unicode::Style::Long));
  38. static_assert(to_underlying(ValueStyle::Short) == to_underlying(Unicode::Style::Short));
  39. static_assert(to_underlying(ValueStyle::Narrow) == to_underlying(Unicode::Style::Narrow));
  40. enum class Display {
  41. Auto,
  42. Always,
  43. };
  44. static Display display_from_string(StringView display);
  45. static StringView display_to_string(Display);
  46. enum class Unit {
  47. Years,
  48. Months,
  49. Weeks,
  50. Days,
  51. Hours,
  52. Minutes,
  53. Seconds,
  54. Milliseconds,
  55. Microseconds,
  56. Nanoseconds,
  57. };
  58. static constexpr auto relevant_extension_keys()
  59. {
  60. // 1.3.3 Internal slots, https://tc39.es/proposal-intl-duration-format/#sec-Intl.DurationFormat-internal-slots
  61. // The value of the [[RelevantExtensionKeys]] internal slot is « "nu" ».
  62. return AK::Array { "nu"sv };
  63. }
  64. virtual ~DurationFormat() override = default;
  65. void set_locale(String locale) { m_locale = move(locale); }
  66. String const& locale() const { return m_locale; }
  67. void set_numbering_system(String numbering_system) { m_numbering_system = move(numbering_system); }
  68. String const& numbering_system() const { return m_numbering_system; }
  69. void set_hour_minute_separator(String hour_minute_separator) { m_hour_minute_separator = move(hour_minute_separator); }
  70. String const& hour_minute_separator() const { return m_hour_minute_separator; }
  71. void set_minute_second_separator(String minute_second_separator) { m_minute_second_separator = move(minute_second_separator); }
  72. String const& minute_second_separator() const { return m_minute_second_separator; }
  73. void set_style(StringView style) { m_style = style_from_string(style); }
  74. Style style() const { return m_style; }
  75. StringView style_string() const { return style_to_string(m_style); }
  76. void set_years_style(ValueStyle years_style) { m_years_style = years_style; }
  77. ValueStyle years_style() const { return m_years_style; }
  78. StringView years_style_string() const { return value_style_to_string(m_years_style); }
  79. void set_years_display(Display years_display) { m_years_display = years_display; }
  80. Display years_display() const { return m_years_display; }
  81. StringView years_display_string() const { return display_to_string(m_years_display); }
  82. void set_months_style(ValueStyle months_style) { m_months_style = months_style; }
  83. ValueStyle months_style() const { return m_months_style; }
  84. StringView months_style_string() const { return value_style_to_string(m_months_style); }
  85. void set_months_display(Display months_display) { m_months_display = months_display; }
  86. Display months_display() const { return m_months_display; }
  87. StringView months_display_string() const { return display_to_string(m_months_display); }
  88. void set_weeks_style(ValueStyle weeks_style) { m_weeks_style = weeks_style; }
  89. ValueStyle weeks_style() const { return m_weeks_style; }
  90. StringView weeks_style_string() const { return value_style_to_string(m_weeks_style); }
  91. void set_weeks_display(Display weeks_display) { m_weeks_display = weeks_display; }
  92. Display weeks_display() const { return m_weeks_display; }
  93. StringView weeks_display_string() const { return display_to_string(m_weeks_display); }
  94. void set_days_style(ValueStyle days_style) { m_days_style = days_style; }
  95. ValueStyle days_style() const { return m_days_style; }
  96. StringView days_style_string() const { return value_style_to_string(m_days_style); }
  97. void set_days_display(Display days_display) { m_days_display = days_display; }
  98. Display days_display() const { return m_days_display; }
  99. StringView days_display_string() const { return display_to_string(m_days_display); }
  100. void set_hours_style(ValueStyle hours_style) { m_hours_style = hours_style; }
  101. ValueStyle hours_style() const { return m_hours_style; }
  102. StringView hours_style_string() const { return value_style_to_string(m_hours_style); }
  103. void set_hours_display(Display hours_display) { m_hours_display = hours_display; }
  104. Display hours_display() const { return m_hours_display; }
  105. StringView hours_display_string() const { return display_to_string(m_hours_display); }
  106. void set_minutes_style(ValueStyle minutes_style) { m_minutes_style = minutes_style; }
  107. ValueStyle minutes_style() const { return m_minutes_style; }
  108. StringView minutes_style_string() const { return value_style_to_string(m_minutes_style); }
  109. void set_minutes_display(Display minutes_display) { m_minutes_display = minutes_display; }
  110. Display minutes_display() const { return m_minutes_display; }
  111. StringView minutes_display_string() const { return display_to_string(m_minutes_display); }
  112. void set_seconds_style(ValueStyle seconds_style) { m_seconds_style = seconds_style; }
  113. ValueStyle seconds_style() const { return m_seconds_style; }
  114. StringView seconds_style_string() const { return value_style_to_string(m_seconds_style); }
  115. void set_seconds_display(Display seconds_display) { m_seconds_display = seconds_display; }
  116. Display seconds_display() const { return m_seconds_display; }
  117. StringView seconds_display_string() const { return display_to_string(m_seconds_display); }
  118. void set_milliseconds_style(ValueStyle milliseconds_style) { m_milliseconds_style = milliseconds_style; }
  119. ValueStyle milliseconds_style() const { return m_milliseconds_style; }
  120. StringView milliseconds_style_string() const { return value_style_to_string(m_milliseconds_style); }
  121. void set_milliseconds_display(Display milliseconds_display) { m_milliseconds_display = milliseconds_display; }
  122. Display milliseconds_display() const { return m_milliseconds_display; }
  123. StringView milliseconds_display_string() const { return display_to_string(m_milliseconds_display); }
  124. void set_microseconds_style(ValueStyle microseconds_style) { m_microseconds_style = microseconds_style; }
  125. ValueStyle microseconds_style() const { return m_microseconds_style; }
  126. StringView microseconds_style_string() const { return value_style_to_string(m_microseconds_style); }
  127. void set_microseconds_display(Display microseconds_display) { m_microseconds_display = microseconds_display; }
  128. Display microseconds_display() const { return m_microseconds_display; }
  129. StringView microseconds_display_string() const { return display_to_string(m_microseconds_display); }
  130. void set_nanoseconds_style(ValueStyle nanoseconds_style) { m_nanoseconds_style = nanoseconds_style; }
  131. ValueStyle nanoseconds_style() const { return m_nanoseconds_style; }
  132. StringView nanoseconds_style_string() const { return value_style_to_string(m_nanoseconds_style); }
  133. void set_nanoseconds_display(Display nanoseconds_display) { m_nanoseconds_display = nanoseconds_display; }
  134. Display nanoseconds_display() const { return m_nanoseconds_display; }
  135. StringView nanoseconds_display_string() const { return display_to_string(m_nanoseconds_display); }
  136. void set_fractional_digits(Optional<u8> fractional_digits) { m_fractional_digits = move(fractional_digits); }
  137. bool has_fractional_digits() const { return m_fractional_digits.has_value(); }
  138. u8 fractional_digits() const { return m_fractional_digits.value(); }
  139. private:
  140. explicit DurationFormat(Object& prototype);
  141. String m_locale; // [[Locale]]
  142. String m_numbering_system; // [[NumberingSystem]]
  143. String m_hour_minute_separator; // [[HourMinutesSeparator]]
  144. String m_minute_second_separator; // [[MinutesSecondsSeparator]]
  145. Style m_style { Style::Long }; // [[Style]]
  146. ValueStyle m_years_style { ValueStyle::Long }; // [[YearsStyle]]
  147. Display m_years_display { Display::Auto }; // [[YearsDisplay]]
  148. ValueStyle m_months_style { ValueStyle::Long }; // [[MonthsStyle]]
  149. Display m_months_display { Display::Auto }; // [[MonthsDisplay]]
  150. ValueStyle m_weeks_style { ValueStyle::Long }; // [[WeeksStyle]]
  151. Display m_weeks_display { Display::Auto }; // [[WeeksDisplay]]
  152. ValueStyle m_days_style { ValueStyle::Long }; // [[DaysStyle]]
  153. Display m_days_display { Display::Auto }; // [[DaysDisplay]]
  154. ValueStyle m_hours_style { ValueStyle::Long }; // [[HoursStyle]]
  155. Display m_hours_display { Display::Auto }; // [[HoursDisplay]]
  156. ValueStyle m_minutes_style { ValueStyle::Long }; // [[MinutesStyle]]
  157. Display m_minutes_display { Display::Auto }; // [[MinutesDisplay]]
  158. ValueStyle m_seconds_style { ValueStyle::Long }; // [[SecondsStyle]]
  159. Display m_seconds_display { Display::Auto }; // [[SecondsDisplay]]
  160. ValueStyle m_milliseconds_style { ValueStyle::Long }; // [[MillisecondsStyle]]
  161. Display m_milliseconds_display { Display::Auto }; // [[MillisecondsDisplay]]
  162. ValueStyle m_microseconds_style { ValueStyle::Long }; // [[MicrosecondsStyle]]
  163. Display m_microseconds_display { Display::Auto }; // [[MicrosecondsDisplay]]
  164. ValueStyle m_nanoseconds_style { ValueStyle::Long }; // [[NanosecondsStyle]]
  165. Display m_nanoseconds_display { Display::Auto }; // [[NanosecondsDisplay]]
  166. Optional<u8> m_fractional_digits; // [[FractionalDigits]]
  167. };
  168. // 1.1.1 Duration Records, https://tc39.es/proposal-intl-duration-format/#sec-duration-records
  169. struct DurationRecord {
  170. double years { 0 };
  171. double months { 0 };
  172. double weeks { 0 };
  173. double days { 0 };
  174. double hours { 0 };
  175. double minutes { 0 };
  176. double seconds { 0 };
  177. double milliseconds { 0 };
  178. double microseconds { 0 };
  179. double nanoseconds { 0 };
  180. };
  181. struct DurationInstanceComponent {
  182. double DurationRecord::* value_slot;
  183. DurationFormat::ValueStyle (DurationFormat::*get_style_slot)() const;
  184. void (DurationFormat::*set_style_slot)(DurationFormat::ValueStyle);
  185. DurationFormat::Display (DurationFormat::*get_display_slot)() const;
  186. void (DurationFormat::*set_display_slot)(DurationFormat::Display);
  187. DurationFormat::Unit unit;
  188. ReadonlySpan<StringView> values;
  189. DurationFormat::ValueStyle digital_default;
  190. };
  191. // Table 2: DurationFormat instance internal slots and properties relevant to PartitionDurationFormatPattern, https://tc39.es/proposal-intl-duration-format/#table-partition-duration-format-pattern
  192. // Table 3: Internal slots and property names of DurationFormat instances relevant to Intl.DurationFormat constructor, https://tc39.es/proposal-intl-duration-format/#table-durationformat
  193. static constexpr auto date_values = AK::Array { "long"sv, "short"sv, "narrow"sv };
  194. static constexpr auto time_values = AK::Array { "long"sv, "short"sv, "narrow"sv, "numeric"sv, "2-digit"sv };
  195. static constexpr auto sub_second_values = AK::Array { "long"sv, "short"sv, "narrow"sv, "numeric"sv };
  196. static constexpr auto duration_instances_components = to_array<DurationInstanceComponent>({
  197. { &DurationRecord::years, &DurationFormat::years_style, &DurationFormat::set_years_style, &DurationFormat::years_display, &DurationFormat::set_years_display, DurationFormat::Unit::Years, date_values, DurationFormat::ValueStyle::Short },
  198. { &DurationRecord::months, &DurationFormat::months_style, &DurationFormat::set_months_style, &DurationFormat::months_display, &DurationFormat::set_months_display, DurationFormat::Unit::Months, date_values, DurationFormat::ValueStyle::Short },
  199. { &DurationRecord::weeks, &DurationFormat::weeks_style, &DurationFormat::set_weeks_style, &DurationFormat::weeks_display, &DurationFormat::set_weeks_display, DurationFormat::Unit::Weeks, date_values, DurationFormat::ValueStyle::Short },
  200. { &DurationRecord::days, &DurationFormat::days_style, &DurationFormat::set_days_style, &DurationFormat::days_display, &DurationFormat::set_days_display, DurationFormat::Unit::Days, date_values, DurationFormat::ValueStyle::Short },
  201. { &DurationRecord::hours, &DurationFormat::hours_style, &DurationFormat::set_hours_style, &DurationFormat::hours_display, &DurationFormat::set_hours_display, DurationFormat::Unit::Hours, time_values, DurationFormat::ValueStyle::Numeric },
  202. { &DurationRecord::minutes, &DurationFormat::minutes_style, &DurationFormat::set_minutes_style, &DurationFormat::minutes_display, &DurationFormat::set_minutes_display, DurationFormat::Unit::Minutes, time_values, DurationFormat::ValueStyle::Numeric },
  203. { &DurationRecord::seconds, &DurationFormat::seconds_style, &DurationFormat::set_seconds_style, &DurationFormat::seconds_display, &DurationFormat::set_seconds_display, DurationFormat::Unit::Seconds, time_values, DurationFormat::ValueStyle::Numeric },
  204. { &DurationRecord::milliseconds, &DurationFormat::milliseconds_style, &DurationFormat::set_milliseconds_style, &DurationFormat::milliseconds_display, &DurationFormat::set_milliseconds_display, DurationFormat::Unit::Milliseconds, sub_second_values, DurationFormat::ValueStyle::Numeric },
  205. { &DurationRecord::microseconds, &DurationFormat::microseconds_style, &DurationFormat::set_microseconds_style, &DurationFormat::microseconds_display, &DurationFormat::set_microseconds_display, DurationFormat::Unit::Microseconds, sub_second_values, DurationFormat::ValueStyle::Numeric },
  206. { &DurationRecord::nanoseconds, &DurationFormat::nanoseconds_style, &DurationFormat::set_nanoseconds_style, &DurationFormat::nanoseconds_display, &DurationFormat::set_nanoseconds_display, DurationFormat::Unit::Nanoseconds, sub_second_values, DurationFormat::ValueStyle::Numeric },
  207. });
  208. struct DurationUnitOptions {
  209. DurationFormat::ValueStyle style;
  210. DurationFormat::Display display;
  211. };
  212. struct DurationFormatPart {
  213. StringView type;
  214. String value;
  215. StringView unit;
  216. };
  217. ThrowCompletionOr<DurationRecord> to_duration_record(VM&, Value input);
  218. i8 duration_sign(DurationRecord const&);
  219. ThrowCompletionOr<DurationUnitOptions> get_duration_unit_options(VM&, DurationFormat::Unit unit, Object const& options, DurationFormat::Style base_style, ReadonlySpan<StringView> styles_list, DurationFormat::ValueStyle digital_base, Optional<DurationFormat::ValueStyle> previous_style, bool two_digit_hours);
  220. Crypto::BigFraction compute_fractional_digits(DurationFormat const&, DurationRecord const&);
  221. bool next_unit_fractional(DurationFormat const&, DurationFormat::Unit unit);
  222. Vector<DurationFormatPart> format_numeric_hours(VM&, DurationFormat const&, MathematicalValue const& hours_value, bool sign_displayed);
  223. Vector<DurationFormatPart> format_numeric_minutes(VM&, DurationFormat const&, MathematicalValue const& minutes_value, bool hours_displayed, bool sign_displayed);
  224. Vector<DurationFormatPart> format_numeric_seconds(VM&, DurationFormat const&, MathematicalValue const& seconds_value, bool minutes_displayed, bool sign_displayed);
  225. Vector<DurationFormatPart> format_numeric_units(VM&, DurationFormat const&, DurationRecord const&, DurationFormat::Unit first_numeric_unit, bool sign_displayed);
  226. Vector<DurationFormatPart> list_format_parts(VM&, DurationFormat const&, Vector<Vector<DurationFormatPart>>& partitioned_parts_list);
  227. Vector<DurationFormatPart> partition_duration_format_pattern(VM&, DurationFormat const&, DurationRecord const&);
  228. }