2023-01-11 16:26:49 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <AK/Error.h>
|
|
|
|
#include <AK/Format.h>
|
|
|
|
#include <AK/Platform.h>
|
|
|
|
#include <AK/String.h>
|
|
|
|
#include <AK/Traits.h>
|
|
|
|
#include <AK/Types.h>
|
|
|
|
|
|
|
|
namespace AK {
|
|
|
|
|
|
|
|
class FlyString {
|
|
|
|
public:
|
|
|
|
FlyString();
|
|
|
|
~FlyString();
|
|
|
|
|
|
|
|
static ErrorOr<FlyString> from_utf8(StringView);
|
2023-06-11 20:49:02 +03:00
|
|
|
template<typename T>
|
|
|
|
requires(IsOneOf<RemoveCVReference<T>, DeprecatedString, DeprecatedFlyString>)
|
|
|
|
static ErrorOr<String> from_utf8(T&&) = delete;
|
|
|
|
|
2023-02-18 17:54:46 +03:00
|
|
|
FlyString(String const&);
|
2023-02-14 17:21:55 +03:00
|
|
|
FlyString& operator=(String const&);
|
2023-01-11 16:26:49 +03:00
|
|
|
|
|
|
|
FlyString(FlyString const&);
|
|
|
|
FlyString& operator=(FlyString const&);
|
|
|
|
|
|
|
|
FlyString(FlyString&&);
|
|
|
|
FlyString& operator=(FlyString&&);
|
|
|
|
|
|
|
|
[[nodiscard]] bool is_empty() const;
|
|
|
|
[[nodiscard]] unsigned hash() const;
|
2023-09-05 20:55:21 +03:00
|
|
|
[[nodiscard]] u32 ascii_case_insensitive_hash() const;
|
2023-01-11 16:26:49 +03:00
|
|
|
|
|
|
|
explicit operator String() const;
|
|
|
|
String to_string() const;
|
|
|
|
|
|
|
|
[[nodiscard]] Utf8View code_points() const;
|
|
|
|
[[nodiscard]] ReadonlyBytes bytes() const;
|
|
|
|
[[nodiscard]] StringView bytes_as_string_view() const;
|
|
|
|
|
|
|
|
[[nodiscard]] bool operator==(FlyString const& other) const;
|
|
|
|
[[nodiscard]] bool operator==(String const&) const;
|
|
|
|
[[nodiscard]] bool operator==(StringView) const;
|
|
|
|
[[nodiscard]] bool operator==(char const*) const;
|
|
|
|
|
2023-09-05 21:05:54 +03:00
|
|
|
[[nodiscard]] int operator<=>(FlyString const& other) const;
|
|
|
|
|
2023-01-11 16:26:49 +03:00
|
|
|
static void did_destroy_fly_string_data(Badge<Detail::StringData>, StringView);
|
|
|
|
[[nodiscard]] uintptr_t data(Badge<String>) const;
|
|
|
|
|
|
|
|
// This is primarily interesting to unit tests.
|
|
|
|
[[nodiscard]] static size_t number_of_fly_strings();
|
|
|
|
|
2023-03-09 19:45:33 +03:00
|
|
|
// FIXME: Remove these once all code has been ported to FlyString
|
2023-03-04 23:02:58 +03:00
|
|
|
[[nodiscard]] DeprecatedFlyString to_deprecated_fly_string() const;
|
2023-03-09 19:45:33 +03:00
|
|
|
static ErrorOr<FlyString> from_deprecated_fly_string(DeprecatedFlyString const&);
|
2023-06-11 20:49:02 +03:00
|
|
|
template<typename T>
|
|
|
|
requires(IsSame<RemoveCVReference<T>, StringView>)
|
|
|
|
static ErrorOr<String> from_deprecated_fly_string(T&&) = delete;
|
2023-03-04 23:02:58 +03:00
|
|
|
|
2023-03-07 21:53:21 +03:00
|
|
|
// Compare this FlyString against another string with ASCII caseless matching.
|
|
|
|
[[nodiscard]] bool equals_ignoring_ascii_case(FlyString const&) const;
|
2023-09-04 12:49:29 +03:00
|
|
|
[[nodiscard]] bool equals_ignoring_ascii_case(StringView) const;
|
2023-03-07 21:53:21 +03:00
|
|
|
|
2023-03-24 20:25:10 +03:00
|
|
|
template<typename... Ts>
|
|
|
|
[[nodiscard]] ALWAYS_INLINE constexpr bool is_one_of(Ts... strings) const
|
|
|
|
{
|
|
|
|
return (... || this->operator==(forward<Ts>(strings)));
|
|
|
|
}
|
|
|
|
|
2023-01-11 16:26:49 +03:00
|
|
|
private:
|
|
|
|
// This will hold either the pointer to the Detail::StringData it represents or the raw bytes of
|
|
|
|
// an inlined short string.
|
|
|
|
uintptr_t m_data { 0 };
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct Traits<FlyString> : public GenericTraits<FlyString> {
|
|
|
|
static unsigned hash(FlyString const&);
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct Formatter<FlyString> : Formatter<StringView> {
|
|
|
|
ErrorOr<void> format(FormatBuilder&, FlyString const&);
|
|
|
|
};
|
|
|
|
|
2023-09-05 20:55:21 +03:00
|
|
|
struct ASCIICaseInsensitiveFlyStringTraits : public Traits<String> {
|
|
|
|
static unsigned hash(FlyString const& s) { return s.ascii_case_insensitive_hash(); }
|
|
|
|
static bool equals(FlyString const& a, FlyString const& b) { return a.bytes().data() == b.bytes().data() || a.bytes_as_string_view().equals_ignoring_ascii_case(b.bytes_as_string_view()); }
|
|
|
|
};
|
|
|
|
|
2023-01-11 16:26:49 +03:00
|
|
|
}
|
|
|
|
|
2023-08-07 13:07:35 +03:00
|
|
|
[[nodiscard]] ALWAYS_INLINE AK::FlyString operator""_fly_string(char const* cstring, size_t length)
|
2023-02-25 18:14:37 +03:00
|
|
|
{
|
2023-08-07 13:07:35 +03:00
|
|
|
return AK::FlyString::from_utf8(AK::StringView(cstring, length)).release_value();
|
2023-02-25 18:14:37 +03:00
|
|
|
}
|
|
|
|
|
2023-01-11 16:26:49 +03:00
|
|
|
#if USING_AK_GLOBALLY
|
|
|
|
using AK::FlyString;
|
|
|
|
#endif
|