mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-31 13:20:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			157 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2024-2025, Tim Flynn <trflynn89@ladybird.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/NonnullOwnPtr.h>
 | |
| #include <LibUnicode/ICU.h>
 | |
| #include <LibUnicode/ListFormat.h>
 | |
| 
 | |
| #include <unicode/listformatter.h>
 | |
| 
 | |
| namespace Unicode {
 | |
| 
 | |
| ListFormatType list_format_type_from_string(StringView list_format_type)
 | |
| {
 | |
|     if (list_format_type == "conjunction"sv)
 | |
|         return ListFormatType::Conjunction;
 | |
|     if (list_format_type == "disjunction"sv)
 | |
|         return ListFormatType::Disjunction;
 | |
|     if (list_format_type == "unit"sv)
 | |
|         return ListFormatType::Unit;
 | |
|     VERIFY_NOT_REACHED();
 | |
| }
 | |
| 
 | |
| StringView list_format_type_to_string(ListFormatType list_format_type)
 | |
| {
 | |
|     switch (list_format_type) {
 | |
|     case ListFormatType::Conjunction:
 | |
|         return "conjunction"sv;
 | |
|     case ListFormatType::Disjunction:
 | |
|         return "disjunction"sv;
 | |
|     case ListFormatType::Unit:
 | |
|         return "unit"sv;
 | |
|     }
 | |
|     VERIFY_NOT_REACHED();
 | |
| }
 | |
| 
 | |
| static constexpr UListFormatterType icu_list_format_type(ListFormatType type)
 | |
| {
 | |
|     switch (type) {
 | |
|     case ListFormatType::Conjunction:
 | |
|         return ULISTFMT_TYPE_AND;
 | |
|     case ListFormatType::Disjunction:
 | |
|         return ULISTFMT_TYPE_OR;
 | |
|     case ListFormatType::Unit:
 | |
|         return ULISTFMT_TYPE_UNITS;
 | |
|     }
 | |
|     VERIFY_NOT_REACHED();
 | |
| }
 | |
| 
 | |
| static constexpr UListFormatterWidth icu_list_format_width(Style style)
 | |
| {
 | |
|     switch (style) {
 | |
|     case Style::Long:
 | |
|         return ULISTFMT_WIDTH_WIDE;
 | |
|     case Style::Short:
 | |
|         return ULISTFMT_WIDTH_SHORT;
 | |
|     case Style::Narrow:
 | |
|         return ULISTFMT_WIDTH_NARROW;
 | |
|     }
 | |
|     VERIFY_NOT_REACHED();
 | |
| }
 | |
| 
 | |
| static constexpr StringView icu_list_format_field_to_string(i32 field)
 | |
| {
 | |
|     switch (field) {
 | |
|     case ULISTFMT_LITERAL_FIELD:
 | |
|         return "literal"sv;
 | |
|     case ULISTFMT_ELEMENT_FIELD:
 | |
|         return "element"sv;
 | |
|     }
 | |
|     VERIFY_NOT_REACHED();
 | |
| }
 | |
| 
 | |
| class ListFormatImpl : public ListFormat {
 | |
| public:
 | |
|     ListFormatImpl(NonnullOwnPtr<icu::ListFormatter> formatter)
 | |
|         : m_formatter(move(formatter))
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     virtual ~ListFormatImpl() override = default;
 | |
| 
 | |
|     virtual Utf16String format(ReadonlySpan<Utf16String> list) const override
 | |
|     {
 | |
|         UErrorCode status = U_ZERO_ERROR;
 | |
| 
 | |
|         auto formatted = format_list_impl(list);
 | |
|         if (!formatted.has_value())
 | |
|             return {};
 | |
| 
 | |
|         auto formatted_string = formatted->toTempString(status);
 | |
|         if (icu_failure(status))
 | |
|             return {};
 | |
| 
 | |
|         return icu_string_to_utf16_string(formatted_string);
 | |
|     }
 | |
| 
 | |
|     virtual Vector<Partition> format_to_parts(ReadonlySpan<Utf16String> list) const override
 | |
|     {
 | |
|         UErrorCode status = U_ZERO_ERROR;
 | |
| 
 | |
|         auto formatted = format_list_impl(list);
 | |
|         if (!formatted.has_value())
 | |
|             return {};
 | |
| 
 | |
|         auto formatted_string = formatted->toTempString(status);
 | |
|         if (icu_failure(status))
 | |
|             return {};
 | |
| 
 | |
|         Vector<Partition> result;
 | |
| 
 | |
|         icu::ConstrainedFieldPosition position;
 | |
|         position.constrainCategory(UFIELD_CATEGORY_LIST);
 | |
| 
 | |
|         while (static_cast<bool>(formatted->nextPosition(position, status)) && icu_success(status)) {
 | |
|             auto type = icu_list_format_field_to_string(position.getField());
 | |
|             auto part = formatted_string.tempSubStringBetween(position.getStart(), position.getLimit());
 | |
| 
 | |
|             result.empend(type, icu_string_to_utf16_string(part));
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     Optional<icu::FormattedList> format_list_impl(ReadonlySpan<Utf16String> list) const
 | |
|     {
 | |
|         UErrorCode status = U_ZERO_ERROR;
 | |
| 
 | |
|         auto icu_list = icu_string_list(list);
 | |
| 
 | |
|         auto formatted_list = m_formatter->formatStringsToValue(icu_list.data(), static_cast<i32>(icu_list.size()), status);
 | |
|         if (icu_failure(status))
 | |
|             return {};
 | |
| 
 | |
|         return formatted_list;
 | |
|     }
 | |
| 
 | |
|     NonnullOwnPtr<icu::ListFormatter> m_formatter;
 | |
| };
 | |
| 
 | |
| NonnullOwnPtr<ListFormat> ListFormat::create(StringView locale, ListFormatType type, Style style)
 | |
| {
 | |
|     UErrorCode status = U_ZERO_ERROR;
 | |
| 
 | |
|     auto locale_data = LocaleData::for_locale(locale);
 | |
|     VERIFY(locale_data.has_value());
 | |
| 
 | |
|     auto formatter = adopt_own(*icu::ListFormatter::createInstance(locale_data->locale(), icu_list_format_type(type), icu_list_format_width(style), status));
 | |
|     VERIFY(icu_success(status));
 | |
| 
 | |
|     return adopt_own(*new ListFormatImpl(move(formatter)));
 | |
| }
 | |
| 
 | |
| }
 | 
