2024-11-06 06:46:35 +08:00

64 lines
1.3 KiB
C++

#pragma once
inline size_t str_len(const char *s) { return strlen(s); }
inline size_t str_len(const wchar_t *s) { return wcslen(s); }
template <class CharT>
struct TextUnion
{
enum
{
ShortTextCapacity = 0x10 / sizeof(CharT)
};
union
{
const CharT *text; // 0x0
CharT chars[ShortTextCapacity];
};
size_t size, // 0x10
capacity;
bool isValid() const
{
if (size <= 0 || size > capacity)
return false;
const CharT *t = getText();
return Engine::isAddressWritable(t, size) && str_len(t) == size;
}
const CharT *getText() const
{
return capacity < ShortTextCapacity ? chars : text;
}
void setText(const CharT *_text, size_t _size)
{
if (_size < ShortTextCapacity)
::memcpy(chars, _text, (_size + 1) * sizeof(CharT));
else
text = _text;
capacity = size = _size;
}
void setLongText(const CharT *_text, size_t _size)
{
text = _text;
size = _size;
capacity = max(ShortTextCapacity, _size);
}
void setText(const std::basic_string<CharT> &text)
{
setText((const CharT *)text.c_str(), text.size());
}
void setLongText(const std::basic_string<CharT> &text)
{
setLongText((const CharT *)text.c_str(), text.size());
}
};
using TextUnionA = TextUnion<char>;
using TextUnionW = TextUnion<wchar_t>;
// EOF