#pragma once #include "toml_print_to_stream.h" namespace toml { enum class format_flags : uint8_t { none, always_print_as_inline = 1, quote_dates_and_times = 2 }; [[nodiscard]] constexpr format_flags operator & (format_flags lhs, format_flags rhs) noexcept { return static_cast(impl::unwrap_enum(lhs) & impl::unwrap_enum(rhs)); } [[nodiscard]] constexpr format_flags operator | (format_flags lhs, format_flags rhs) noexcept { return static_cast( impl::unwrap_enum(lhs) | impl::unwrap_enum(rhs) ); } } namespace toml::impl { template class formatter { private: const toml::node* source_; format_flags flags_; int indent_; bool naked_newline_; std::basic_ostream* stream_ = nullptr; protected: [[nodiscard]] const toml::node& source() const noexcept { return *source_; } [[nodiscard]] format_flags flags() const noexcept { return flags_; } [[nodiscard]] std::basic_ostream& stream() const noexcept { return *stream_; } static constexpr size_t indent_columns = 4; static constexpr toml::string_view indent_string = TOML_STRING_PREFIX(" "sv); [[nodiscard]] int indent() const noexcept { return indent_; } void indent(int level) noexcept { indent_ = level; } void increase_indent() noexcept { indent_++; } void decrease_indent() noexcept { indent_--; } void clear_naked_newline() noexcept { naked_newline_ = false; } void attach(std::basic_ostream& stream) noexcept { indent_ = 0; naked_newline_ = true; stream_ = &stream; } void detach() noexcept { stream_ = nullptr; } void print_newline(bool force = false) TOML_MAY_THROW { if (!naked_newline_ || force) { print_to_stream('\n', *stream_); naked_newline_ = true; } } void print_indent() TOML_MAY_THROW { for (int i = 0; i < indent_; i++) { print_to_stream(indent_string, *stream_); naked_newline_ = false; } } void print_quoted_string(toml::string_view str) TOML_MAY_THROW { if (str.empty()) print_to_stream("\"\""sv, *stream_); else { print_to_stream('"', *stream_); print_to_stream_with_escapes(str, *stream_); print_to_stream('"', *stream_); } naked_newline_ = false; } template void print(const value& val) TOML_MAY_THROW { if constexpr (std::is_same_v) { print_quoted_string(val.get()); } else { static constexpr auto is_date_time = std::is_same_v || std::is_same_v || std::is_same_v; if constexpr (is_date_time) { if ((flags_ & format_flags::quote_dates_and_times) != format_flags::none) print_to_stream('"', *stream_); } *stream_ << val; if constexpr (is_date_time) { if ((flags_ & format_flags::quote_dates_and_times) != format_flags::none) print_to_stream('"', *stream_); } naked_newline_ = false; } } void print(const node& val_node, node_type type) TOML_MAY_THROW { switch (type) { case node_type::string: print(*reinterpret_cast*>(&val_node)); break; case node_type::integer: print(*reinterpret_cast*>(&val_node)); break; case node_type::floating_point: print(*reinterpret_cast*>(&val_node)); break; case node_type::boolean: print(*reinterpret_cast*>(&val_node)); break; case node_type::date: print(*reinterpret_cast*>(&val_node)); break; case node_type::time: print(*reinterpret_cast*>(&val_node)); break; case node_type::date_time: print(*reinterpret_cast*>(&val_node)); break; TOML_NO_DEFAULT_CASE; } } formatter(const toml::node& source, format_flags flags) noexcept : source_{ &source }, flags_{ flags } {} }; }