DurationFormat.cpp 2.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. /*
  2. * Copyright (c) 2024, Tim Flynn <trflynn89@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #define AK_DONT_REPLACE_STD
  7. #include <AK/Array.h>
  8. #include <AK/CharacterTypes.h>
  9. #include <AK/GenericLexer.h>
  10. #include <LibUnicode/DurationFormat.h>
  11. #include <LibUnicode/ICU.h>
  12. #include <LibUnicode/NumberFormat.h>
  13. #include <unicode/measfmt.h>
  14. #include <unicode/measunit.h>
  15. #include <unicode/measure.h>
  16. namespace Unicode {
  17. static constexpr bool is_not_ascii_digit(u32 code_point)
  18. {
  19. return !is_ascii_digit(code_point);
  20. }
  21. DigitalFormat digital_format(StringView locale)
  22. {
  23. UErrorCode status = U_ZERO_ERROR;
  24. auto locale_data = LocaleData::for_locale(locale);
  25. if (!locale_data.has_value())
  26. return {};
  27. if (auto const& digital_format = locale_data->digital_format(); digital_format.has_value())
  28. return *digital_format;
  29. RoundingOptions rounding_options;
  30. rounding_options.type = RoundingType::SignificantDigits;
  31. rounding_options.min_significant_digits = 1;
  32. rounding_options.max_significant_digits = 2;
  33. auto number_formatter = NumberFormat::create(locale, "latn"sv, {}, rounding_options);
  34. auto icu_locale = adopt_own(*locale_data->locale().clone());
  35. icu_locale->setUnicodeKeywordValue("nu", "latn", status);
  36. if (icu_failure(status))
  37. return {};
  38. icu::MeasureFormat measurement_formatter(*icu_locale, UMEASFMT_WIDTH_NUMERIC, status);
  39. if (icu_failure(status))
  40. return {};
  41. auto measures = to_array<icu::Measure>({
  42. { 1, icu::MeasureUnit::createHour(status), status },
  43. { 22, icu::MeasureUnit::createMinute(status), status },
  44. { 33, icu::MeasureUnit::createSecond(status), status },
  45. });
  46. if (icu_failure(status))
  47. return {};
  48. icu::UnicodeString formatted;
  49. icu::FieldPosition position;
  50. measurement_formatter.formatMeasures(measures.data(), static_cast<i32>(measures.size()), formatted, position, status);
  51. if (icu_failure(status))
  52. return {};
  53. auto hours_minutes_seconds = icu_string_to_string(formatted);
  54. GenericLexer lexer { hours_minutes_seconds };
  55. DigitalFormat digital_format;
  56. auto hours = lexer.consume_while(is_ascii_digit);
  57. digital_format.uses_two_digit_hours = hours.length() == 2;
  58. auto hours_minutes_separator = lexer.consume_while(is_not_ascii_digit);
  59. digital_format.hours_minutes_separator = MUST(String::from_utf8(hours_minutes_separator));
  60. lexer.ignore_while(is_ascii_digit);
  61. auto minutes_seconds_separator = lexer.consume_while(is_not_ascii_digit);
  62. digital_format.minutes_seconds_separator = MUST(String::from_utf8(minutes_seconds_separator));
  63. locale_data->set_digital_format(move(digital_format));
  64. return *locale_data->digital_format();
  65. }
  66. }