13#include "libxr_def.hpp"
18template <Print::Text Source,
typename... Args>
28template <
typename... Values>
37template <
typename T,
bool IsEnum = std::is_enum_v<T>>
46 using Type = std::underlying_type_t<T>;
55static constexpr bool runtime_string_always_false =
false;
57inline constexpr size_t runtime_string_unbounded_capacity =
58 std::numeric_limits<size_t>::max();
59inline constexpr size_t runtime_string_max_field_width =
60 std::numeric_limits<uint8_t>::max();
75[[nodiscard]]
constexpr size_t RuntimeStringFieldCapacity(Print::FormatPackKind pack)
77 constexpr size_t width = runtime_string_max_field_width;
78 constexpr size_t precision = runtime_string_max_field_width;
82 case Print::FormatPackKind::I32:
83 return 1U + precision;
84 case Print::FormatPackKind::I64:
85 return 1U + precision;
86 case Print::FormatPackKind::U32:
87 return 2U + precision;
88 case Print::FormatPackKind::U64:
89 return 2U + precision;
90 case Print::FormatPackKind::Pointer:
91 return 2U + (precision > 2U *
sizeof(uintptr_t) ? precision
92 : 2U *
sizeof(uintptr_t));
93 case Print::FormatPackKind::Character:
95 case Print::FormatPackKind::StringView:
96 return runtime_string_unbounded_capacity;
97 case Print::FormatPackKind::F32:
98 return 1U +
static_cast<size_t>(std::numeric_limits<float>::max_exponent10) +
100 case Print::FormatPackKind::F64:
101 return 1U +
static_cast<size_t>(std::numeric_limits<double>::max_exponent10) +
103 case Print::FormatPackKind::LongDouble:
105 static_cast<size_t>(std::numeric_limits<long double>::max_exponent10) +
109 return runtime_string_unbounded_capacity;
112[[nodiscard]]
constexpr bool RuntimeStringAddCapacity(
size_t& total,
size_t value)
114 if (value == runtime_string_unbounded_capacity ||
115 runtime_string_unbounded_capacity - total < value)
123template <
typename Built>
124[[nodiscard]]
consteval size_t RuntimeStringMaxFormattedSize(
size_t source_size)
126 size_t total = source_size;
127 for (
const auto& argument : Built::ArgumentList())
129 if (!RuntimeStringAddCapacity(total, RuntimeStringFieldCapacity(argument.pack)))
131 return runtime_string_unbounded_capacity;
152 using Decayed = std::remove_cvref_t<T>;
153 using Normalized =
typename RuntimeStringNormalized<Decayed>::Type;
155 static constexpr bool is_text =
156 std::is_array_v<Decayed> || std::is_same_v<Decayed, char*> ||
157 std::is_same_v<Decayed, const char*> || std::is_same_v<Decayed, std::string> ||
158 std::is_same_v<Decayed, std::string_view>;
160 static constexpr bool is_pointer =
161 std::is_pointer_v<Decayed> &&
162 !std::is_function_v<std::remove_pointer_t<Decayed>>;
164 static constexpr bool has_static_capacity =
165 !is_text && (std::is_same_v<Decayed, std::nullptr_t> || is_pointer ||
166 std::is_integral_v<Normalized> ||
167 std::is_floating_point_v<Normalized>);
176template <
typename... Args>
179 template <
typename... CallArgs>
180 static constexpr bool matches =
182 typename RuntimeStringArgumentTraits<CallArgs>::Decayed...>,
192 std::string_view view;
207inline constexpr bool runtime_string_is_view_v =
216 char* data =
nullptr;
220 [[nodiscard]]
ErrorCode Write(std::string_view chunk)
222 if (capacity - size < chunk.size())
228 std::memcpy(data + size, chunk.data(), chunk.size());
229 size += chunk.size();
253template <
Print::Text Source =
"",
typename... Args>
257 static_assert((std::is_same_v<
258 Args,
typename Detail::RuntimeStringArgumentTraits<Args>::Decayed> &&
260 "LibXR::RuntimeStringView argument types must be unqualified "
263 "LibXR::RuntimeStringView formatted arguments cannot contain "
264 "runtime strings; use RuntimeStringView<> for text concatenation");
274 :
data_(other.data_),
279 other.data_ =
nullptr;
289 requires(Source.
Size() == 0 &&
sizeof...(Args) == 0)
297 requires(Source.
Size() == 0 &&
sizeof...(Args) == 0)
305 requires(Source.
Size() == 0 &&
sizeof...(Args) == 0)
311 template <
typename CharPtr>
312 requires(Source.
Size() == 0 &&
sizeof...(Args) == 0 &&
313 (std::is_same_v<CharPtr, char*> || std::is_same_v<CharPtr, const char*>))
321 requires(Source.
Size() == 0 &&
sizeof...(Args) == 0)
323 static_cast<void>(
AssignCopy(
static_cast<const char*
>(text)));
327 template <
typename First,
typename Second,
typename... Rest>
329 requires(Source.
Size() == 0 &&
sizeof...(Args) == 0)
331 static_cast<void>(
AssignConcat(std::forward<First>(first),
332 std::forward<Second>(second),
333 std::forward<Rest>(rest)...));
340 template <
typename... CallArgs>
342 requires((Source.
Size() != 0 ||
sizeof...(Args) != 0) &&
346 "LibXR::RuntimeStringView format argument types do not match "
349 static constexpr size_t max_size =
350 Detail::RuntimeStringMaxFormattedSize<Built>(Source.
Size());
351 static_assert(max_size != Detail::runtime_string_unbounded_capacity,
352 "LibXR::RuntimeStringView cannot retain unbounded formatted "
356 return Print::FormatTo<Source>(sink, std::forward<CallArgs>(args)...);
364 template <
typename... CallArgs>
366 requires((Source.
Size() != 0 ||
sizeof...(Args) != 0) &&
370 "LibXR::RuntimeStringView printf argument types do not match "
373 static constexpr size_t max_size =
374 Detail::RuntimeStringMaxFormattedSize<Built>(Source.Size());
375 static_assert(max_size != Detail::runtime_string_unbounded_capacity,
376 "LibXR::RuntimeStringView cannot retain unbounded formatted "
380 return Print::PrintfTo<Source>(sink, std::forward<CallArgs>(args)...);
385 [[nodiscard]]
constexpr std::string_view
View()
const
387 return data_ ==
nullptr ? std::string_view{} : std::string_view(
data_,
size_);
393 [[nodiscard]]
constexpr size_t Size()
const {
return size_; }
395 [[nodiscard]]
constexpr bool Empty()
const {
return size_ == 0; }
399 [[nodiscard]]
constexpr operator std::string_view()
const {
return View(); }
401 [[nodiscard]]
operator const char*()
const {
return CStr(); }
411 if (
data_ !=
nullptr)
439 if (payload_size == std::numeric_limits<size_t>::max())
447 if (
data_ !=
nullptr)
451 if (payload_size == 0)
456 data_ =
new (std::nothrow)
char[payload_size + 1U];
457 if (
data_ ==
nullptr)
459 ASSERT(
data_ !=
nullptr);
479 std::memcpy(
data_, text.data(), text.size());
481 if (
data_ !=
nullptr)
483 data_[text.size()] =
'\0';
498 :
AssignCopy(std::string_view(text, std::strlen(text)));
510 template <
typename... Parts>
513 size_t total_size = 0;
516 auto count = [&](
auto&& part)
527 status = text.status;
529 else if (std::numeric_limits<size_t>::max() - total_size < text.view.size())
535 total_size += text.view.size();
538 (count(std::forward<Parts>(parts)), ...);
551 auto copy = [&](
auto&& part)
555 if (!text.view.empty())
557 std::memcpy(
data_ + offset, text.view.data(), text.view.size());
558 offset += text.view.size();
561 (copy(std::forward<Parts>(parts)), ...);
563 if (
data_ !=
nullptr)
565 data_[total_size] =
'\0';
582 template <
typename WriteFn>
585 if (
data_ ==
nullptr)
600 if (
data_ !=
nullptr)
602 data_[sink.size] =
'\0';
615 size_t bound)
noexcept
618 while (size < bound && text[size] !=
'\0')
632 template <
typename T>
635 using Bare = std::remove_cvref_t<T>;
636 if constexpr (Detail::runtime_string_is_view_v<T>)
638 const auto& retained = text;
643 else if constexpr (std::is_array_v<Bare> &&
644 std::is_same_v<std::remove_cv_t<std::remove_extent_t<Bare>>,
char>)
646 return {.view = std::string_view(text,
BoundedTextLength(text, std::extent_v<Bare>))};
648 else if constexpr (std::is_same_v<Bare, std::string_view>)
650 return {.view = text};
652 else if constexpr (std::is_same_v<Bare, std::string>)
654 return {.view = std::string_view(text.data(), text.size())};
656 else if constexpr (std::is_same_v<Bare, std::nullptr_t>)
660 else if constexpr (std::is_same_v<Bare, char*> || std::is_same_v<Bare, const char*>)
662 return text ==
nullptr
665 .view = std::string_view(text, std::strlen(text))};
669 static_assert(Detail::runtime_string_always_false<T>,
670 "RuntimeStringView constructor only accepts text-like parts. "
671 "Use Reformat() or Reprintf() for numeric formatting.");
692template <
typename First,
typename Second,
typename... Rest>
static consteval bool Matches()
判断 Args... 是否与启用的转换项精确匹配 / Return whether Args... exactly match the enabled conversions
constexpr RuntimeStringView()=default
Constructs an empty valid view. / 构造一个空的有效视图。
size_t capacity_
不含结尾 NUL 的已探测/分配容量。 / Probed/allocated payload capacity excluding the trailing NUL.
constexpr ErrorCode Status() const
Returns the latest operation status. / 返回最近一次操作状态。
ErrorCode status_
最近一次构造或重写状态。 / Status of the latest construction or rewrite.
ErrorCode SetFailure(ErrorCode status)
记录最近一次失败,并在已有存储上保留一个有效的空 C 字符串。
RuntimeStringView(First &&first, Second &&second, Rest &&... rest)
Concatenates text parts into retained storage. / 拼接多个文本片段到保留存储。
RuntimeStringView(const char(&text)[N])
Copies one bounded const char array as text. / 按文本语义拷贝一个有界只读字符数组。
ErrorCode AssignCopy(const char *text)
C 字符串入口负责空指针检查,再进入统一的文本拷贝路径。
ErrorCode Reformat(CallArgs &&... args)
使用绑定的 brace-style 格式重写当前内容。
ErrorCode AssignFormatted(size_t max_size, WriteFn &&write)
格式化重写入口;首次调用按编译期上界分配,后续调用只覆盖已有存储。
RuntimeStringView(std::nullptr_t text)
Preserves the old bare-nullptr entry for runtime null checks. / 保留裸 nullptr 入口以维持运行期空指针检查语义。
constexpr size_t Size() const
Returns the text size excluding the trailing NUL. / 返回不含结尾 NUL 的文本长度。
static constexpr size_t BoundedTextLength(const char *text, size_t bound) noexcept
在已知边界内查找文本长度;遇到首个 \0 截断,否则使用整个数组长度。
ErrorCode AssignConcat(Parts &&... parts)
拼接构造的两遍流程:先校验并统计所有片段,再一次性写入。
const char * CStr() const
Returns a NUL-terminated C string. / 返回 NUL 结尾 C 字符串。
RuntimeStringView(CharPtr text)
Copies a NUL-terminated C-string pointer into retained storage. / 拷贝 NUL 结尾 C 字符串指针到保留存储。
RuntimeStringView(RuntimeStringView &&other) noexcept
Move-constructs by taking the retained storage handle. / 移动构造并接管保留存储句柄。
ErrorCode AssignCopy(std::string_view text)
构造期文本拷贝入口;空字符串保持零分配。
size_t size_
不含结尾 NUL 的当前可见文本长度。 / Current visible payload size excluding the trailing NUL.
char * data_
长期保留的 NUL 结尾存储;对象析构时不释放。 / Retained NUL-terminated storage; not released by the destructor.
ErrorCode Reprintf(CallArgs &&... args)
使用绑定的 printf-style 格式重写当前内容。
RuntimeStringView(std::string_view text)
Copies text into retained storage. / 拷贝文本到保留存储。
constexpr std::string_view View() const
Returns the current read-only view. / 返回当前只读视图。
RuntimeStringView(char(&text)[N])
Copies one bounded mutable char array as text. / 按文本语义拷贝一个有界可变字符数组。
ErrorCode EnsureCapacity(size_t payload_size)
为首个非空结果分配保留存储,已有存储不会再次扩容。
static constexpr Detail::RuntimeStringTextPart NormalizeConcatPart(T &&text)
把拼接构造支持的输入统一归一化为只读文本片段;普通拼接只接受文本类输入, 数值格式化必须显式走 Reformat() 或 Reprintf()。
constexpr bool Empty() const
Returns whether the current visible text is empty. / 返回当前可见文本是否为空。
@ OUT_OF_RANGE
超出范围 | Out of range
@ NO_MEM
内存不足 | Insufficient memory
@ PTR_NULL
空指针 | Null pointer
@ OK
操作成功 | Operation successful
@ ARG_ERR
参数错误 | Argument error
RuntimeStringView(std::string_view) -> RuntimeStringView<>
单参数文本构造推导为普通保留字符串。 / Single text argument deduces a plain retained string.
Compile-time traits for formatted retained-string arguments.
Checks that a rewrite call uses exactly the argument types bound in RuntimeStringView<Source,...
Sink used by the second formatting pass to fill retained storage.
Normalizes enum arguments to their underlying type for capacity probes.
Normalized text part used by the copy/concatenation path.
Type-list tag used only for exact argument-list comparison.
作为 printf 格式源的结构化字符串字面量包装 / Structural literal wrapper used as the NTTP source for printf formats
constexpr size_t Size() const
返回不含结尾零字节的格式串长度 / Return the format length without the terminating zero byte