mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 18:02:05 -05:00
3a33b8ea08
It's now possible to edit widget properties inline in the properties window. We're currently relying on the basic GVariant conversion functions to do all the "parsing" but that's not gonna be good enough.
152 lines
3.6 KiB
C++
152 lines
3.6 KiB
C++
#pragma once
|
|
|
|
#include <AK/ByteBuffer.h>
|
|
#include <AK/RetainPtr.h>
|
|
#include <AK/StringImpl.h>
|
|
#include <AK/Traits.h>
|
|
#include <AK/Vector.h>
|
|
#include <AK/StringView.h>
|
|
#include <AK/kstdio.h>
|
|
|
|
namespace AK {
|
|
|
|
class String {
|
|
public:
|
|
~String() { }
|
|
|
|
String() { }
|
|
|
|
String(StringView view)
|
|
: m_impl(StringImpl::create(view.characters(), view.length()))
|
|
{
|
|
}
|
|
|
|
String(const String& other)
|
|
: m_impl(const_cast<String&>(other).m_impl.copy_ref())
|
|
{
|
|
}
|
|
|
|
String(String&& other)
|
|
: m_impl(move(other.m_impl))
|
|
{
|
|
}
|
|
|
|
String(const char* cstring, ShouldChomp shouldChomp = NoChomp)
|
|
: m_impl(StringImpl::create(cstring, shouldChomp))
|
|
{
|
|
}
|
|
|
|
String(const char* cstring, ssize_t length, ShouldChomp shouldChomp = NoChomp)
|
|
: m_impl(StringImpl::create(cstring, length, shouldChomp))
|
|
{
|
|
}
|
|
|
|
String(const StringImpl& impl)
|
|
: m_impl(const_cast<StringImpl&>(impl))
|
|
{
|
|
}
|
|
|
|
String(const StringImpl* impl)
|
|
: m_impl(const_cast<StringImpl*>(impl))
|
|
{
|
|
}
|
|
|
|
String(RetainPtr<StringImpl>&& impl)
|
|
: m_impl(move(impl))
|
|
{
|
|
}
|
|
|
|
String(Retained<StringImpl>&& impl)
|
|
: m_impl(move(impl))
|
|
{
|
|
}
|
|
|
|
int to_int(bool& ok) const;
|
|
unsigned to_uint(bool& ok) const;
|
|
|
|
String to_lowercase() const
|
|
{
|
|
if (!m_impl)
|
|
return String();
|
|
return m_impl->to_lowercase();
|
|
}
|
|
|
|
String to_uppercase() const
|
|
{
|
|
if (!m_impl)
|
|
return String();
|
|
return m_impl->to_uppercase();
|
|
}
|
|
|
|
Vector<String> split(char separator) const;
|
|
String substring(int start, int length) const;
|
|
|
|
Vector<StringView> split_view(char separator) const;
|
|
StringView substring_view(int start, int length) const;
|
|
|
|
bool is_null() const { return !m_impl; }
|
|
bool is_empty() const { return length() == 0; }
|
|
ssize_t length() const { return m_impl ? m_impl->length() : 0; }
|
|
const char* characters() const { return m_impl ? m_impl->characters() : nullptr; }
|
|
char operator[](ssize_t i) const { ASSERT(m_impl); return (*m_impl)[i]; }
|
|
|
|
bool ends_with(const String&) const;
|
|
|
|
bool operator==(const String&) const;
|
|
bool operator!=(const String& other) const { return !(*this == other); }
|
|
bool operator<(const String&) const;
|
|
|
|
String isolated_copy() const;
|
|
|
|
static String empty();
|
|
|
|
StringImpl* impl() { return m_impl.ptr(); }
|
|
const StringImpl* impl() const { return m_impl.ptr(); }
|
|
|
|
String& operator=(String&& other)
|
|
{
|
|
if (this != &other)
|
|
m_impl = move(other.m_impl);
|
|
return *this;
|
|
}
|
|
|
|
String& operator=(const String& other)
|
|
{
|
|
if (this != &other)
|
|
m_impl = const_cast<String&>(other).m_impl.copy_ref();
|
|
return *this;
|
|
}
|
|
|
|
ByteBuffer to_byte_buffer() const;
|
|
static String from_byte_buffer(const ByteBuffer&, ShouldChomp = NoChomp);
|
|
|
|
static String format(const char*, ...);
|
|
|
|
StringView view() const { return { characters(), length() }; }
|
|
|
|
private:
|
|
RetainPtr<StringImpl> m_impl;
|
|
};
|
|
|
|
inline bool StringView::operator==(const String& string) const
|
|
{
|
|
if (string.is_null())
|
|
return !m_characters;
|
|
if (!m_characters)
|
|
return false;
|
|
if (m_length != string.length())
|
|
return false;
|
|
if (m_characters == string.characters())
|
|
return true;
|
|
return !memcmp(m_characters, string.characters(), m_length);
|
|
}
|
|
|
|
template<>
|
|
struct Traits<String> {
|
|
static unsigned hash(const String& s) { return s.impl() ? s.impl()->hash() : 0; }
|
|
static void dump(const String& s) { kprintf("%s", s.characters()); }
|
|
};
|
|
|
|
}
|
|
|
|
using AK::String;
|