/* * Copyright (c) 2021-2023, Tim Flynn * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include namespace AK { using Utf16Data = Vector; ErrorOr utf8_to_utf16(StringView); ErrorOr utf8_to_utf16(Utf8View const&); ErrorOr utf32_to_utf16(Utf32View const&); ErrorOr code_point_to_utf16(Utf16Data&, u32); class Utf16View; class Utf16CodePointIterator { friend class Utf16View; public: Utf16CodePointIterator() = default; ~Utf16CodePointIterator() = default; bool operator==(Utf16CodePointIterator const& other) const { return (m_ptr == other.m_ptr) && (m_remaining_code_units == other.m_remaining_code_units); } Utf16CodePointIterator& operator++(); u32 operator*() const; size_t length_in_code_units() const; private: Utf16CodePointIterator(u16 const* ptr, size_t length) : m_ptr(ptr) , m_remaining_code_units(length) { } u16 const* m_ptr { nullptr }; size_t m_remaining_code_units { 0 }; }; class Utf16View { public: using Iterator = Utf16CodePointIterator; static bool is_high_surrogate(u16); static bool is_low_surrogate(u16); static u32 decode_surrogate_pair(u16 high_surrogate, u16 low_surrogate); Utf16View() = default; ~Utf16View() = default; explicit Utf16View(ReadonlySpan code_units) : m_code_units(code_units) { } template Utf16View(char16_t const (&code_units)[Size]) : m_code_units( reinterpret_cast(&code_units[0]), code_units[Size - 1] == u'\0' ? Size - 1 : Size) { } bool operator==(Utf16View const& other) const { return m_code_units == other.m_code_units; } enum class AllowInvalidCodeUnits { Yes, No, }; ErrorOr to_byte_string(AllowInvalidCodeUnits = AllowInvalidCodeUnits::No) const; ErrorOr to_utf8(AllowInvalidCodeUnits = AllowInvalidCodeUnits::No) const; bool is_null() const { return m_code_units.is_null(); } bool is_empty() const { return m_code_units.is_empty(); } size_t length_in_code_units() const { return m_code_units.size(); } size_t length_in_code_points() const; Utf16CodePointIterator begin() const { return { begin_ptr(), m_code_units.size() }; } Utf16CodePointIterator end() const { return { end_ptr(), 0 }; } u16 const* data() const { return m_code_units.data(); } u16 code_unit_at(size_t index) const; u32 code_point_at(size_t index) const; size_t code_point_offset_of(size_t code_unit_offset) const; size_t code_unit_offset_of(size_t code_point_offset) const; size_t code_unit_offset_of(Utf16CodePointIterator const&) const; Utf16View substring_view(size_t code_unit_offset, size_t code_unit_length) const; Utf16View substring_view(size_t code_unit_offset) const { return substring_view(code_unit_offset, length_in_code_units() - code_unit_offset); } Utf16View unicode_substring_view(size_t code_point_offset, size_t code_point_length) const; Utf16View unicode_substring_view(size_t code_point_offset) const { return unicode_substring_view(code_point_offset, length_in_code_points() - code_point_offset); } bool starts_with(Utf16View const&) const; bool validate(size_t& valid_code_units) const; bool validate() const { size_t valid_code_units; return validate(valid_code_units); } bool equals_ignoring_case(Utf16View const&) const; private: u16 const* begin_ptr() const { return m_code_units.data(); } u16 const* end_ptr() const { return begin_ptr() + m_code_units.size(); } size_t calculate_length_in_code_points() const; ReadonlySpan m_code_units; mutable Optional m_length_in_code_points; }; } template<> struct AK::Formatter : Formatter { ErrorOr format(FormatBuilder& builder, AK::Utf16View const& value) { return builder.builder().try_append(value); } }; #if USING_AK_GLOBALLY using AK::Utf16Data; using AK::Utf16View; #endif