| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2024-10-04 13:19:50 +02:00
										 |  |  |  * Copyright (c) 2018-2021, Andreas Kling <andreas@ladybird.org> | 
					
						
							| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-15 14:56:37 +02:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-30 00:17:54 +03:00
										 |  |  | #include <AK/Assertions.h>
 | 
					
						
							|  |  |  | #include <AK/Checked.h>
 | 
					
						
							| 
									
										
										
										
											2024-02-22 16:26:37 +01:00
										 |  |  | #include <AK/Concepts.h>
 | 
					
						
							| 
									
										
										
										
											2022-10-22 15:38:21 +02:00
										 |  |  | #include <AK/EnumBits.h>
 | 
					
						
							| 
									
										
										
										
											2020-02-14 21:41:10 +01:00
										 |  |  | #include <AK/Forward.h>
 | 
					
						
							| 
									
										
										
										
											2021-11-10 11:05:21 +01:00
										 |  |  | #include <AK/Optional.h>
 | 
					
						
							| 
									
										
										
										
											2020-07-27 14:15:37 +02:00
										 |  |  | #include <AK/Span.h>
 | 
					
						
							| 
									
										
										
										
											2020-02-14 21:41:10 +01:00
										 |  |  | #include <AK/StdLibExtras.h>
 | 
					
						
							| 
									
										
										
										
											2025-06-26 19:06:46 -04:00
										 |  |  | #include <AK/StringConversions.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-14 15:21:50 +02:00
										 |  |  | #include <AK/StringHash.h>
 | 
					
						
							| 
									
										
										
										
											2020-02-26 15:25:24 +08:00
										 |  |  | #include <AK/StringUtils.h>
 | 
					
						
							| 
									
										
										
										
											2019-04-16 02:39:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace AK { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-15 14:56:37 +02:00
										 |  |  | class StringView { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2021-01-10 16:29:28 -07:00
										 |  |  |     ALWAYS_INLINE constexpr StringView() = default; | 
					
						
							| 
									
										
										
										
											2025-05-10 16:10:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |     ALWAYS_INLINE constexpr StringView(char const* characters, size_t length) | 
					
						
							| 
									
										
										
										
											2019-05-28 11:53:16 +02:00
										 |  |  |         : m_characters(characters) | 
					
						
							|  |  |  |         , m_length(length) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-27 22:41:38 +04:30
										 |  |  |         if (!is_constant_evaluated()) | 
					
						
							| 
									
										
										
										
											2023-03-07 14:28:21 +00:00
										 |  |  |             VERIFY(!Checked<uintptr_t>::addition_would_overflow(reinterpret_cast<uintptr_t>(characters), length)); | 
					
						
							| 
									
										
										
										
											2019-05-28 11:53:16 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-10 16:10:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |     ALWAYS_INLINE StringView(unsigned char const* characters, size_t length) | 
					
						
							| 
									
										
										
										
											2023-03-07 14:28:21 +00:00
										 |  |  |         : m_characters(reinterpret_cast<char const*>(characters)) | 
					
						
							| 
									
										
										
										
											2019-05-28 11:53:16 +02:00
										 |  |  |         , m_length(length) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-03-07 14:28:21 +00:00
										 |  |  |         VERIFY(!Checked<uintptr_t>::addition_would_overflow(reinterpret_cast<uintptr_t>(characters), length)); | 
					
						
							| 
									
										
										
										
											2019-05-28 11:53:16 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-10 16:10:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 11:43:41 +02:00
										 |  |  |     ALWAYS_INLINE StringView(ReadonlyBytes bytes) | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |         : m_characters(reinterpret_cast<char const*>(bytes.data())) | 
					
						
							| 
									
										
										
										
											2020-08-19 11:43:41 +02:00
										 |  |  |         , m_length(bytes.size()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-29 12:03:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |     StringView(ByteBuffer const&); | 
					
						
							| 
									
										
										
										
											2022-12-06 20:39:11 +00:00
										 |  |  |     StringView(String const&); | 
					
						
							| 
									
										
										
										
											2023-01-28 08:26:22 -05:00
										 |  |  |     StringView(FlyString const&); | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     StringView(ByteString const&); | 
					
						
							| 
									
										
										
										
											2019-04-15 14:56:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-10 23:26:22 +02:00
										 |  |  |     explicit StringView(ByteBuffer&&) = delete; | 
					
						
							| 
									
										
										
										
											2022-12-06 20:39:11 +00:00
										 |  |  |     explicit StringView(String&&) = delete; | 
					
						
							| 
									
										
										
										
											2023-01-28 08:26:22 -05:00
										 |  |  |     explicit StringView(FlyString&&) = delete; | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     explicit StringView(ByteString&&) = delete; | 
					
						
							| 
									
										
										
										
											2021-09-04 17:53:43 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-01 16:49:30 +02:00
										 |  |  |     template<OneOf<String, FlyString, ByteString, ByteBuffer> StringType> | 
					
						
							| 
									
										
										
										
											2023-01-28 08:22:54 -05:00
										 |  |  |     StringView& operator=(StringType&&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-10 16:30:36 -04:00
										 |  |  |     [[nodiscard]] constexpr bool is_null() const { return m_characters == nullptr; } | 
					
						
							| 
									
										
										
										
											2021-04-21 21:19:39 -06:00
										 |  |  |     [[nodiscard]] constexpr bool is_empty() const { return m_length == 0; } | 
					
						
							| 
									
										
										
										
											2020-07-27 14:15:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 15:21:50 +02:00
										 |  |  |     [[nodiscard]] constexpr char const* characters_without_null_termination() const { return m_characters; } | 
					
						
							| 
									
										
										
										
											2021-04-21 21:19:39 -06:00
										 |  |  |     [[nodiscard]] constexpr size_t length() const { return m_length; } | 
					
						
							| 
									
										
										
										
											2020-07-27 14:15:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-11 01:30:53 -07:00
										 |  |  |     [[nodiscard]] ReadonlyBytes bytes() const { return { m_characters, m_length }; } | 
					
						
							| 
									
										
										
										
											2020-07-27 14:15:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-26 00:38:49 +01:00
										 |  |  |     constexpr char const& operator[](size_t index) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!is_constant_evaluated()) | 
					
						
							|  |  |  |             VERIFY(index < m_length); | 
					
						
							|  |  |  |         return m_characters[index]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-10 16:13:29 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-03 16:43:01 -05:00
										 |  |  |     using ConstIterator = SimpleIterator<StringView const, char const>; | 
					
						
							| 
									
										
										
										
											2020-09-06 21:14:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-11 01:30:53 -07:00
										 |  |  |     [[nodiscard]] constexpr ConstIterator begin() const { return ConstIterator::begin(*this); } | 
					
						
							|  |  |  |     [[nodiscard]] constexpr ConstIterator end() const { return ConstIterator::end(*this); } | 
					
						
							| 
									
										
										
										
											2019-04-15 14:56:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 15:21:50 +02:00
										 |  |  |     [[nodiscard]] constexpr unsigned hash() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (is_empty()) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         return string_hash(characters_without_null_termination(), length()); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-24 22:31:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     [[nodiscard]] bool starts_with(StringView, CaseSensitivity = CaseSensitivity::CaseSensitive) const; | 
					
						
							|  |  |  |     [[nodiscard]] bool ends_with(StringView, CaseSensitivity = CaseSensitivity::CaseSensitive) const; | 
					
						
							| 
									
										
										
										
											2021-04-11 01:30:53 -07:00
										 |  |  |     [[nodiscard]] bool starts_with(char) const; | 
					
						
							|  |  |  |     [[nodiscard]] bool ends_with(char) const; | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     [[nodiscard]] bool matches(StringView mask, CaseSensitivity = CaseSensitivity::CaseInsensitive) const; | 
					
						
							|  |  |  |     [[nodiscard]] bool matches(StringView mask, Vector<MaskSpan>&, CaseSensitivity = CaseSensitivity::CaseInsensitive) const; | 
					
						
							| 
									
										
										
										
											2021-04-11 01:30:53 -07:00
										 |  |  |     [[nodiscard]] bool contains(char) const; | 
					
						
							| 
									
										
										
										
											2022-09-12 16:31:16 +02:00
										 |  |  |     [[nodiscard]] bool contains(u32) const; | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     [[nodiscard]] bool contains(StringView, CaseSensitivity = CaseSensitivity::CaseSensitive) const; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     [[nodiscard]] bool equals_ignoring_ascii_case(StringView) const; | 
					
						
							| 
									
										
										
										
											2025-04-06 08:39:05 -04:00
										 |  |  |     [[nodiscard]] bool is_ascii() const; | 
					
						
							| 
									
										
										
										
											2019-09-12 06:13:07 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     [[nodiscard]] StringView trim(StringView characters, TrimMode mode = TrimMode::Both) const { return StringUtils::trim(*this, characters, mode); } | 
					
						
							| 
									
										
										
										
											2021-04-11 01:30:53 -07:00
										 |  |  |     [[nodiscard]] StringView trim_whitespace(TrimMode mode = TrimMode::Both) const { return StringUtils::trim_whitespace(*this, mode); } | 
					
						
							| 
									
										
										
										
											2020-09-20 18:05:04 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-06 10:19:35 -04:00
										 |  |  |     [[nodiscard]] String to_ascii_lowercase_string() const; | 
					
						
							|  |  |  |     [[nodiscard]] String to_ascii_uppercase_string() const; | 
					
						
							|  |  |  |     [[nodiscard]] String to_ascii_titlecase_string() const; | 
					
						
							| 
									
										
										
										
											2021-07-01 13:45:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-10 16:30:36 -04:00
										 |  |  |     [[nodiscard]] Optional<size_t> find(char needle, size_t start = 0) const { return StringUtils::find(*this, needle, start); } | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     [[nodiscard]] Optional<size_t> find(StringView needle, size_t start = 0) const { return StringUtils::find(*this, needle, start); } | 
					
						
							| 
									
										
										
										
											2021-07-01 14:58:37 +02:00
										 |  |  |     [[nodiscard]] Optional<size_t> find_last(char needle) const { return StringUtils::find_last(*this, needle); } | 
					
						
							| 
									
										
										
										
											2022-12-15 21:20:14 +00:00
										 |  |  |     [[nodiscard]] Optional<size_t> find_last(StringView needle) const { return StringUtils::find_last(*this, needle); } | 
					
						
							| 
									
										
										
										
											2022-09-30 21:19:53 +02:00
										 |  |  |     [[nodiscard]] Optional<size_t> find_last_not(char needle) const { return StringUtils::find_last_not(*this, needle); } | 
					
						
							| 
									
										
										
										
											2021-01-12 23:28:45 +03:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 11:05:21 +01:00
										 |  |  |     [[nodiscard]] Vector<size_t> find_all(StringView needle) const; | 
					
						
							| 
									
										
										
										
											2021-07-01 17:00:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 18:12:21 +02:00
										 |  |  |     using SearchDirection = StringUtils::SearchDirection; | 
					
						
							| 
									
										
										
										
											2021-10-31 14:53:22 -06:00
										 |  |  |     [[nodiscard]] Optional<size_t> find_any_of(StringView needles, SearchDirection direction = SearchDirection::Forward) const { return StringUtils::find_any_of(*this, needles, direction); } | 
					
						
							| 
									
										
										
										
											2021-07-01 15:01:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-21 21:19:39 -06:00
										 |  |  |     [[nodiscard]] constexpr StringView substring_view(size_t start, size_t length) const | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-27 22:41:38 +04:30
										 |  |  |         if (!is_constant_evaluated()) | 
					
						
							|  |  |  |             VERIFY(start + length <= m_length); | 
					
						
							| 
									
										
										
										
											2021-04-21 21:19:39 -06:00
										 |  |  |         return { m_characters + start, length }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [[nodiscard]] constexpr StringView substring_view(size_t start) const | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-10-24 23:29:42 +02:00
										 |  |  |         if (!is_constant_evaluated()) | 
					
						
							|  |  |  |             VERIFY(start <= length()); | 
					
						
							| 
									
										
										
										
											2021-04-21 21:19:39 -06:00
										 |  |  |         return substring_view(start, length() - start); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-22 15:38:21 +02:00
										 |  |  |     [[nodiscard]] Vector<StringView> split_view(char, SplitBehavior = SplitBehavior::Nothing) const; | 
					
						
							|  |  |  |     [[nodiscard]] Vector<StringView> split_view(StringView, SplitBehavior = SplitBehavior::Nothing) const; | 
					
						
							| 
									
										
										
										
											2019-08-04 11:44:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-22 15:38:21 +02:00
										 |  |  |     [[nodiscard]] Vector<StringView> split_view_if(Function<bool(char)> const& predicate, SplitBehavior = SplitBehavior::Nothing) const; | 
					
						
							| 
									
										
										
										
											2021-04-12 07:54:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-18 18:35:36 +02:00
										 |  |  |     [[nodiscard]] StringView find_last_split_view(char separator) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         auto begin = find_last(separator); | 
					
						
							|  |  |  |         if (!begin.has_value()) | 
					
						
							|  |  |  |             return *this; | 
					
						
							|  |  |  |         return substring_view(begin.release_value() + 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 05:31:37 +03:00
										 |  |  |     [[nodiscard]] StringView find_first_split_view(char separator) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         auto needle_begin = find(separator); | 
					
						
							|  |  |  |         if (!needle_begin.has_value()) | 
					
						
							|  |  |  |             return *this; | 
					
						
							|  |  |  |         return substring_view(0, needle_begin.release_value()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  |     template<typename Callback> | 
					
						
							|  |  |  |     auto for_each_split_view(char separator, SplitBehavior split_behavior, Callback callback) const | 
					
						
							| 
									
										
										
										
											2022-01-09 02:26:45 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-08-26 11:00:23 +00:00
										 |  |  |         StringView separator_view { &separator, 1 }; | 
					
						
							|  |  |  |         return for_each_split_view(separator_view, split_behavior, callback); | 
					
						
							| 
									
										
										
										
											2022-01-09 02:26:45 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  |     template<typename Callback> | 
					
						
							|  |  |  |     auto for_each_split_view(StringView separator, SplitBehavior split_behavior, Callback callback) const | 
					
						
							| 
									
										
										
										
											2022-01-09 02:26:45 -08:00
										 |  |  |     { | 
					
						
							|  |  |  |         VERIFY(!separator.is_empty()); | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  |         // FIXME: This can't go in the template header since declval won't allow the incomplete StringView type.
 | 
					
						
							| 
									
										
										
										
											2024-02-22 16:26:37 +01:00
										 |  |  |         using CallbackReturn = InvokeResult<Callback, StringView>; | 
					
						
							|  |  |  |         constexpr auto ReturnsErrorOr = FallibleFunction<Callback, StringView>; | 
					
						
							|  |  |  |         // FIXME: We might need a concept for this...
 | 
					
						
							|  |  |  |         constexpr auto ReturnsIterationDecision = []() -> bool { | 
					
						
							|  |  |  |             if constexpr (ReturnsErrorOr) | 
					
						
							|  |  |  |                 return IsSame<typename CallbackReturn::ResultType, IterationDecision>; | 
					
						
							|  |  |  |             return IsSame<CallbackReturn, IterationDecision>; | 
					
						
							|  |  |  |         }(); | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  |         using ReturnType = Conditional<ReturnsErrorOr, ErrorOr<void>, void>; | 
					
						
							|  |  |  |         return [&]() -> ReturnType { | 
					
						
							|  |  |  |             if (is_empty()) | 
					
						
							|  |  |  |                 return ReturnType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             StringView view { *this }; | 
					
						
							|  |  |  |             auto maybe_separator_index = find(separator); | 
					
						
							|  |  |  |             bool keep_empty = has_flag(split_behavior, SplitBehavior::KeepEmpty); | 
					
						
							|  |  |  |             bool keep_separator = has_flag(split_behavior, SplitBehavior::KeepTrailingSeparator); | 
					
						
							|  |  |  |             while (maybe_separator_index.has_value()) { | 
					
						
							|  |  |  |                 auto separator_index = maybe_separator_index.value(); | 
					
						
							|  |  |  |                 auto part_with_separator = view.substring_view(0, separator_index + separator.length()); | 
					
						
							|  |  |  |                 if (keep_empty || separator_index > 0) { | 
					
						
							|  |  |  |                     auto part = part_with_separator; | 
					
						
							|  |  |  |                     if (!keep_separator) | 
					
						
							|  |  |  |                         part = part_with_separator.substring_view(0, separator_index); | 
					
						
							| 
									
										
										
										
											2024-02-22 16:26:37 +01:00
										 |  |  |                     if constexpr (ReturnsErrorOr) { | 
					
						
							|  |  |  |                         if constexpr (ReturnsIterationDecision) { | 
					
						
							|  |  |  |                             if (TRY(callback(part)) == IterationDecision::Break) | 
					
						
							|  |  |  |                                 return ReturnType(); | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             TRY(callback(part)); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         if constexpr (ReturnsIterationDecision) { | 
					
						
							|  |  |  |                             if (callback(part) == IterationDecision::Break) | 
					
						
							|  |  |  |                                 return ReturnType(); | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             callback(part); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 view = view.substring_view_starting_after_substring(part_with_separator); | 
					
						
							|  |  |  |                 maybe_separator_index = view.find(separator); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (keep_empty || !view.is_empty()) { | 
					
						
							|  |  |  |                 if constexpr (ReturnsErrorOr) | 
					
						
							|  |  |  |                     TRY(callback(view)); | 
					
						
							| 
									
										
										
										
											2022-10-22 16:31:59 +02:00
										 |  |  |                 else | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  |                     callback(view); | 
					
						
							| 
									
										
										
										
											2022-10-22 16:31:59 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-02-04 17:54:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return ReturnType(); | 
					
						
							|  |  |  |         }(); | 
					
						
							| 
									
										
										
										
											2022-01-09 02:26:45 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 07:42:33 -05:00
										 |  |  |     // Create a Vector of StringViews split by line endings. As of CommonMark
 | 
					
						
							|  |  |  |     // 0.29, the spec defines a line ending as "a newline (U+000A), a carriage
 | 
					
						
							|  |  |  |     // return (U+000D) not followed by a newline, or a carriage return and a
 | 
					
						
							|  |  |  |     // following newline.".
 | 
					
						
							| 
									
										
										
										
											2024-03-08 11:27:48 -05:00
										 |  |  |     enum class ConsiderCarriageReturn { | 
					
						
							|  |  |  |         No, | 
					
						
							|  |  |  |         Yes, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     [[nodiscard]] Vector<StringView> lines(ConsiderCarriageReturn = ConsiderCarriageReturn::Yes) const; | 
					
						
							| 
									
										
										
										
											2024-03-08 09:57:35 -05:00
										 |  |  |     [[nodiscard]] size_t count_lines(ConsiderCarriageReturn = ConsiderCarriageReturn::Yes) const; | 
					
						
							| 
									
										
										
										
											2019-12-02 07:42:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-13 16:30:55 +03:00
										 |  |  |     // Create a new substring view of this string view, starting either at the beginning of
 | 
					
						
							|  |  |  |     // the given substring view, or after its end, and continuing until the end of this string
 | 
					
						
							|  |  |  |     // view (that is, for the remaining part of its length). For example,
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     //    StringView str { "foobar" };
 | 
					
						
							|  |  |  |     //    StringView substr = str.substring_view(1, 2);  // "oo"
 | 
					
						
							|  |  |  |     //    StringView substr_from = str.substring_view_starting_from_substring(subst);  // "oobar"
 | 
					
						
							|  |  |  |     //    StringView substr_after = str.substring_view_starting_after_substring(subst);  // "bar"
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     // Note that this only works if the string view passed as an argument is indeed a substring
 | 
					
						
							|  |  |  |     // view of this string view, such as one created by substring_view() and split_view(). It
 | 
					
						
							|  |  |  |     // does not work for arbitrary strings; for example declaring substr in the example above as
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     //     StringView substr { "oo" };
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     // would not work.
 | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     [[nodiscard]] StringView substring_view_starting_from_substring(StringView substring) const; | 
					
						
							|  |  |  |     [[nodiscard]] StringView substring_view_starting_after_substring(StringView substring) const; | 
					
						
							| 
									
										
										
										
											2019-06-13 16:30:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-29 02:52:20 +02:00
										 |  |  |     [[nodiscard]] bool copy_characters_to_buffer(char* buffer, size_t buffer_size) const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |     constexpr bool operator==(char const* cstring) const | 
					
						
							| 
									
										
										
										
											2019-06-07 19:22:58 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-08 18:30:40 +02:00
										 |  |  |         if (is_null()) | 
					
						
							| 
									
										
										
										
											2021-11-06 14:12:16 -06:00
										 |  |  |             return cstring == nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-08 18:30:40 +02:00
										 |  |  |         if (!cstring) | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:13:57 +01:00
										 |  |  |         // NOTE: `m_characters` is not guaranteed to be null-terminated, but `cstring` is.
 | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |         char const* cp = cstring; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:13:57 +01:00
										 |  |  |         for (size_t i = 0; i < m_length; ++i) { | 
					
						
							| 
									
										
										
										
											2021-11-06 14:12:16 -06:00
										 |  |  |             if (*cp == '\0') | 
					
						
							| 
									
										
										
										
											2021-02-24 22:10:32 +01:00
										 |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:13:57 +01:00
										 |  |  |             if (m_characters[i] != *(cp++)) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-11-06 14:12:16 -06:00
										 |  |  |         return *cp == '\0'; | 
					
						
							| 
									
										
										
										
											2019-06-07 19:22:58 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-23 14:13:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-10 18:52:14 +02:00
										 |  |  |     constexpr bool operator==(char const c) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_length == 1 && *m_characters == c; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     bool operator==(ByteString const&) const; | 
					
						
							| 
									
										
										
										
											2019-04-16 02:39:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |     [[nodiscard]] constexpr int compare(StringView other) const | 
					
						
							| 
									
										
										
										
											2019-08-15 14:07:23 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-15 13:08:15 +00:00
										 |  |  |         if (m_length == 0 && other.m_length == 0) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-30 16:53:49 +01:00
										 |  |  |         if (m_characters == nullptr) | 
					
						
							|  |  |  |             return other.m_characters ? -1 : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (other.m_characters == nullptr) | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         size_t rlen = min(m_length, other.m_length); | 
					
						
							|  |  |  |         int c = __builtin_memcmp(m_characters, other.m_characters, rlen); | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |         if (c == 0) { | 
					
						
							|  |  |  |             if (length() < other.length()) | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |             if (length() == other.length()) | 
					
						
							|  |  |  |                 return 0; | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return c; | 
					
						
							| 
									
										
										
										
											2019-08-15 14:07:23 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |     constexpr bool operator==(StringView other) const | 
					
						
							| 
									
										
										
										
											2019-08-15 14:07:23 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |         return length() == other.length() && compare(other) == 0; | 
					
						
							| 
									
										
										
										
											2019-08-15 14:07:23 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |     constexpr bool operator!=(StringView other) const | 
					
						
							| 
									
										
										
										
											2020-08-09 21:55:32 -04:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |         return length() != other.length() || compare(other) != 0; | 
					
						
							| 
									
										
										
										
											2020-08-09 21:55:32 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-29 15:49:33 +01:00
										 |  |  |     constexpr bool operator<(StringView other) const { return compare(other) < 0; } | 
					
						
							|  |  |  |     constexpr bool operator<=(StringView other) const { return compare(other) <= 0; } | 
					
						
							|  |  |  |     constexpr bool operator>(StringView other) const { return compare(other) > 0; } | 
					
						
							|  |  |  |     constexpr bool operator>=(StringView other) const { return compare(other) >= 0; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     [[nodiscard]] ByteString to_byte_string() const; | 
					
						
							| 
									
										
										
										
											2020-05-06 18:53:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 00:24:24 +02:00
										 |  |  |     [[nodiscard]] bool is_whitespace() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return StringUtils::is_whitespace(*this); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-03 02:56:02 +03:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     [[nodiscard]] ByteString replace(StringView needle, StringView replacement, ReplaceMode) const; | 
					
						
							| 
									
										
										
										
											2025-05-10 16:30:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 00:24:24 +02:00
										 |  |  |     [[nodiscard]] size_t count(StringView needle) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return StringUtils::count(*this, needle); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-11 01:02:24 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-14 21:55:36 -04:00
										 |  |  |     [[nodiscard]] size_t count(char needle) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return StringUtils::count(*this, needle); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 14:16:29 +04:30
										 |  |  |     template<typename... Ts> | 
					
						
							| 
									
										
										
										
											2021-07-31 19:27:25 -04:00
										 |  |  |     [[nodiscard]] ALWAYS_INLINE constexpr bool is_one_of(Ts&&... strings) const | 
					
						
							| 
									
										
										
										
											2020-10-28 01:28:11 +03:30
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-04 14:16:29 +04:30
										 |  |  |         return (... || this->operator==(forward<Ts>(strings))); | 
					
						
							| 
									
										
										
										
											2020-10-28 01:28:11 +03:30
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-20 23:52:36 +01:00
										 |  |  |     template<typename... Ts> | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     [[nodiscard]] ALWAYS_INLINE constexpr bool is_one_of_ignoring_ascii_case(Ts&&... strings) const | 
					
						
							| 
									
										
										
										
											2022-03-20 23:52:36 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         return (... || | 
					
						
							|  |  |  |                 [this, &strings]() -> bool { | 
					
						
							|  |  |  |             if constexpr (requires(Ts a) { a.view()->StringView; }) | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |                 return this->equals_ignoring_ascii_case(forward<Ts>(strings.view())); | 
					
						
							| 
									
										
										
										
											2022-03-20 23:52:36 +01:00
										 |  |  |             else | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |                 return this->equals_ignoring_ascii_case(forward<Ts>(strings)); | 
					
						
							| 
									
										
										
										
											2022-03-20 23:52:36 +01:00
										 |  |  |         }()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-23 11:35:03 +13:00
										 |  |  |     template<Arithmetic T> | 
					
						
							| 
									
										
										
										
											2025-08-06 07:09:37 -04:00
										 |  |  |     Optional<T> to_number(TrimWhitespace trim_whitespace = TrimWhitespace::Yes, int base = 10) const | 
					
						
							| 
									
										
										
										
											2023-12-23 11:35:03 +13:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-08-06 07:09:37 -04:00
										 |  |  |         return parse_number<T>(*this, trim_whitespace, base); | 
					
						
							| 
									
										
										
										
											2023-12-23 11:35:03 +13:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-15 14:56:37 +02:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     friend class ByteString; | 
					
						
							| 
									
										
										
										
											2022-04-01 20:58:27 +03:00
										 |  |  |     char const* m_characters { nullptr }; | 
					
						
							| 
									
										
										
										
											2019-12-09 17:45:40 +01:00
										 |  |  |     size_t m_length { 0 }; | 
					
						
							| 
									
										
										
										
											2019-04-15 14:56:37 +02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-04-16 02:39:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 10:48:56 -07:00
										 |  |  | template<> | 
					
						
							| 
									
										
										
										
											2023-11-08 20:29:12 +01:00
										 |  |  | struct Traits<StringView> : public DefaultTraits<StringView> { | 
					
						
							| 
									
										
										
										
											2024-01-25 20:42:53 +01:00
										 |  |  |     using PeekType = StringView; | 
					
						
							|  |  |  |     using ConstPeekType = StringView; | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |     static unsigned hash(StringView s) { return s.hash(); } | 
					
						
							| 
									
										
										
										
											2020-10-08 10:48:56 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-14 08:40:53 -03:00
										 |  |  | struct CaseInsensitiveASCIIStringViewTraits : public Traits<StringView> { | 
					
						
							| 
									
										
										
										
											2022-01-10 11:47:23 -05:00
										 |  |  |     static unsigned hash(StringView s) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (s.is_empty()) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         return case_insensitive_string_hash(s.characters_without_null_termination(), s.length()); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     static bool equals(StringView const& a, StringView const& b) { return a.equals_ignoring_ascii_case(b); } | 
					
						
							| 
									
										
										
										
											2022-01-10 11:47:23 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-16 02:39:16 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-19 11:44:39 -04:00
										 |  |  | // FIXME: Remove this when clang on BSD distributions fully support consteval (specifically in the context of default parameter initialization).
 | 
					
						
							|  |  |  | //        Note that this is fixed in clang-15, but is not yet picked up by all downstream distributions.
 | 
					
						
							|  |  |  | //        See: https://github.com/llvm/llvm-project/issues/48230
 | 
					
						
							| 
									
										
										
										
											2023-06-27 19:31:58 -06:00
										 |  |  | //        Additionally, oss-fuzz currently ships an llvm-project commit that is a pre-release of 15.0.0.
 | 
					
						
							|  |  |  | //        See: https://github.com/google/oss-fuzz/issues/9989
 | 
					
						
							| 
									
										
										
										
											2023-07-19 03:56:12 -06:00
										 |  |  | //        Android currently doesn't ship clang-15 in any NDK
 | 
					
						
							|  |  |  | #if defined(AK_OS_BSD_GENERIC) || defined(OSS_FUZZ) || defined(AK_OS_ANDROID)
 | 
					
						
							| 
									
										
										
										
											2022-03-17 14:53:10 -04:00
										 |  |  | #    define AK_STRING_VIEW_LITERAL_CONSTEVAL constexpr
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #    define AK_STRING_VIEW_LITERAL_CONSTEVAL consteval
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-17 18:27:39 +02:00
										 |  |  | [[nodiscard]] ALWAYS_INLINE AK_STRING_VIEW_LITERAL_CONSTEVAL AK::StringView operator""sv(char const* cstring, size_t length) | 
					
						
							| 
									
										
										
										
											2021-02-24 04:50:00 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     return AK::StringView(cstring, length); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-26 12:18:30 +01:00
										 |  |  | #if USING_AK_GLOBALLY
 | 
					
						
							| 
									
										
										
										
											2023-03-14 08:40:53 -03:00
										 |  |  | using AK::CaseInsensitiveASCIIStringViewTraits; | 
					
						
							| 
									
										
										
										
											2019-04-16 02:39:16 +02:00
										 |  |  | using AK::StringView; | 
					
						
							| 
									
										
										
										
											2022-11-26 12:18:30 +01:00
										 |  |  | #endif
 |