2020-04-02 21:39:21 +00:00
|
|
|
//# This file is a part of toml++ and is subject to the the terms of the MIT license.
|
2021-01-02 15:48:47 +00:00
|
|
|
//# Copyright (c) Mark Gillard <mark.gillard@outlook.com.au>
|
2020-04-02 21:39:21 +00:00
|
|
|
//# See https://github.com/marzer/tomlplusplus/blob/master/LICENSE for the full license text.
|
2020-04-10 16:46:00 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-04-02 21:39:21 +00:00
|
|
|
#pragma once
|
2021-10-24 22:04:23 +00:00
|
|
|
|
2020-04-09 08:13:12 +00:00
|
|
|
//# {{
|
2021-10-23 09:22:41 +00:00
|
|
|
#include "preprocessor.h"
|
2020-06-23 10:04:05 +00:00
|
|
|
#if !TOML_IMPLEMENTATION
|
2021-10-24 22:04:23 +00:00
|
|
|
#error This is an implementation-only header.
|
2020-04-02 21:39:21 +00:00
|
|
|
#endif
|
2020-04-08 13:33:57 +00:00
|
|
|
//# }}
|
2020-04-02 21:39:21 +00:00
|
|
|
|
2021-10-23 09:22:41 +00:00
|
|
|
#include "default_formatter.h"
|
2021-10-24 22:04:23 +00:00
|
|
|
#include "print_to_stream.h"
|
|
|
|
#include "utf8.h"
|
|
|
|
#include "value.h"
|
|
|
|
#include "table.h"
|
|
|
|
#include "array.h"
|
2021-10-24 10:21:32 +00:00
|
|
|
#include "header_start.h"
|
2021-10-25 21:49:17 +00:00
|
|
|
TOML_DISABLE_ARITHMETIC_WARNINGS;
|
2020-06-23 10:04:05 +00:00
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
TOML_NAMESPACE_START
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2020-04-09 08:13:12 +00:00
|
|
|
TOML_EXTERNAL_LINKAGE
|
2021-10-24 22:04:23 +00:00
|
|
|
std::string default_formatter::make_key_segment(const std::string& str) noexcept
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
|
|
|
if (str.empty())
|
2020-07-18 12:10:19 +00:00
|
|
|
return "''"s;
|
2020-04-02 21:39:21 +00:00
|
|
|
else
|
|
|
|
{
|
2020-12-10 11:21:37 +00:00
|
|
|
bool requires_quotes = false;
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::utf8_decoder decoder;
|
2020-12-10 11:21:37 +00:00
|
|
|
for (size_t i = 0; i < str.length() && !requires_quotes; i++)
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
|
|
|
decoder(static_cast<uint8_t>(str[i]));
|
|
|
|
if (decoder.error())
|
2020-12-10 11:21:37 +00:00
|
|
|
requires_quotes = true;
|
2020-04-02 21:39:21 +00:00
|
|
|
else if (decoder.has_code_point())
|
2021-10-24 22:04:23 +00:00
|
|
|
requires_quotes = !impl::is_bare_key_character(decoder.codepoint);
|
2020-04-02 21:39:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-10 11:21:37 +00:00
|
|
|
if (requires_quotes)
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2020-07-18 12:10:19 +00:00
|
|
|
std::string s;
|
2021-10-26 13:49:23 +00:00
|
|
|
s.reserve(str.length() + 2u);
|
2020-07-18 12:10:19 +00:00
|
|
|
s += '"';
|
2020-04-02 21:39:21 +00:00
|
|
|
for (auto c : str)
|
|
|
|
{
|
2020-07-18 12:10:19 +00:00
|
|
|
if TOML_UNLIKELY(c >= '\x00' && c <= '\x1F')
|
2021-10-24 22:04:23 +00:00
|
|
|
s.append(impl::low_character_escape_table[c]);
|
2020-07-18 12:10:19 +00:00
|
|
|
else if TOML_UNLIKELY(c == '\x7F')
|
|
|
|
s.append("\\u007F"sv);
|
|
|
|
else if TOML_UNLIKELY(c == '"')
|
|
|
|
s.append("\\\""sv);
|
2020-04-02 21:39:21 +00:00
|
|
|
else
|
|
|
|
s += c;
|
|
|
|
}
|
2020-07-18 12:10:19 +00:00
|
|
|
s += '"';
|
2020-04-02 21:39:21 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-09 08:13:12 +00:00
|
|
|
TOML_EXTERNAL_LINKAGE
|
2021-10-24 22:04:23 +00:00
|
|
|
size_t default_formatter::count_inline_columns(const node& node) noexcept
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2020-04-08 13:33:57 +00:00
|
|
|
switch (node.type())
|
|
|
|
{
|
|
|
|
case node_type::table:
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2021-10-29 08:12:41 +00:00
|
|
|
auto& tbl = *reinterpret_cast<const table*>(&node);
|
|
|
|
if (tbl.empty())
|
2021-10-26 13:49:23 +00:00
|
|
|
return 2u; // "{}"
|
|
|
|
size_t weight = 3u; // "{ }"
|
2021-10-29 08:12:41 +00:00
|
|
|
for (auto&& [k, v] : tbl)
|
2020-06-08 15:31:23 +00:00
|
|
|
{
|
2021-10-26 13:49:23 +00:00
|
|
|
weight += k.length() + count_inline_columns(v) + 2u; // + ", "
|
2021-10-24 22:04:23 +00:00
|
|
|
if (weight >= line_wrap_cols)
|
2020-06-08 15:31:23 +00:00
|
|
|
break;
|
|
|
|
}
|
2020-04-08 13:33:57 +00:00
|
|
|
return weight;
|
|
|
|
}
|
2020-04-02 21:39:21 +00:00
|
|
|
|
2020-04-08 13:33:57 +00:00
|
|
|
case node_type::array:
|
|
|
|
{
|
2021-10-29 08:12:41 +00:00
|
|
|
auto& arr = *reinterpret_cast<const array*>(&node);
|
|
|
|
if (arr.empty())
|
2021-10-26 13:49:23 +00:00
|
|
|
return 2u; // "[]"
|
|
|
|
size_t weight = 3u; // "[ ]"
|
2021-10-29 08:12:41 +00:00
|
|
|
for (auto& elem : arr)
|
2020-06-08 15:31:23 +00:00
|
|
|
{
|
2021-10-26 13:49:23 +00:00
|
|
|
weight += count_inline_columns(elem) + 2u; // + ", "
|
2021-10-24 22:04:23 +00:00
|
|
|
if (weight >= line_wrap_cols)
|
2020-06-08 15:31:23 +00:00
|
|
|
break;
|
|
|
|
}
|
2020-04-08 13:33:57 +00:00
|
|
|
return weight;
|
|
|
|
}
|
|
|
|
|
|
|
|
case node_type::string:
|
|
|
|
{
|
2021-10-29 08:12:41 +00:00
|
|
|
// todo: proper utf8 decoding?
|
|
|
|
// todo: tab awareness?
|
|
|
|
auto& str = (*reinterpret_cast<const value<std::string>*>(&node)).get();
|
|
|
|
return str.length() + 2u; // + ""
|
2020-04-08 13:33:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
case node_type::integer:
|
|
|
|
{
|
2021-10-29 08:12:41 +00:00
|
|
|
auto val = (*reinterpret_cast<const value<int64_t>*>(&node)).get();
|
|
|
|
if (!val)
|
2021-10-26 13:49:23 +00:00
|
|
|
return 1u;
|
2020-04-08 13:33:57 +00:00
|
|
|
size_t weight = {};
|
2021-10-29 08:12:41 +00:00
|
|
|
if (val < 0)
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2021-10-26 13:49:23 +00:00
|
|
|
weight += 1u;
|
2021-10-29 08:12:41 +00:00
|
|
|
val *= -1;
|
2020-04-02 21:39:21 +00:00
|
|
|
}
|
2021-10-29 08:12:41 +00:00
|
|
|
return weight + static_cast<size_t>(log10(static_cast<double>(val))) + 1u;
|
2020-04-08 13:33:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
case node_type::floating_point:
|
|
|
|
{
|
2021-10-29 08:12:41 +00:00
|
|
|
auto val = (*reinterpret_cast<const value<double>*>(&node)).get();
|
|
|
|
if (val == 0.0)
|
2021-10-26 13:49:23 +00:00
|
|
|
return 3u; // "0.0"
|
|
|
|
size_t weight = 2u; // ".0"
|
2021-10-29 08:12:41 +00:00
|
|
|
if (val < 0.0)
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2021-10-26 13:49:23 +00:00
|
|
|
weight += 1u;
|
2021-10-29 08:12:41 +00:00
|
|
|
val *= -1.0;
|
2020-04-02 21:39:21 +00:00
|
|
|
}
|
2021-10-29 08:12:41 +00:00
|
|
|
return weight + static_cast<size_t>(log10(val)) + 1u;
|
2020-04-08 13:33:57 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-10-26 13:49:23 +00:00
|
|
|
case node_type::boolean: return 5u;
|
2020-04-08 13:33:57 +00:00
|
|
|
case node_type::date: [[fallthrough]];
|
2021-10-26 13:49:23 +00:00
|
|
|
case node_type::time: return 10u;
|
|
|
|
case node_type::date_time: return 30u;
|
2020-04-13 20:23:11 +00:00
|
|
|
case node_type::none: TOML_UNREACHABLE;
|
2021-10-23 09:22:41 +00:00
|
|
|
default: TOML_UNREACHABLE;
|
2020-04-08 13:33:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TOML_UNREACHABLE;
|
2020-04-02 21:39:21 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 08:13:12 +00:00
|
|
|
TOML_EXTERNAL_LINKAGE
|
2021-10-24 22:04:23 +00:00
|
|
|
bool default_formatter::forces_multiline(const node& node, size_t starting_column_bias) noexcept
|
2020-04-02 21:39:21 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
return (count_inline_columns(node) + starting_column_bias) >= line_wrap_cols;
|
2020-04-02 21:39:21 +00:00
|
|
|
}
|
2020-04-10 16:46:00 +00:00
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
TOML_EXTERNAL_LINKAGE
|
|
|
|
void default_formatter::print_pending_table_separator()
|
|
|
|
{
|
|
|
|
if (pending_table_separator_)
|
|
|
|
{
|
|
|
|
base::print_newline(true);
|
|
|
|
base::print_newline(true);
|
|
|
|
pending_table_separator_ = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TOML_EXTERNAL_LINKAGE
|
|
|
|
void default_formatter::print_key_segment(const std::string& str)
|
|
|
|
{
|
|
|
|
if (str.empty())
|
|
|
|
impl::print_to_stream(base::stream(), "''"sv);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bool requires_quotes = false;
|
|
|
|
{
|
|
|
|
impl::utf8_decoder decoder;
|
|
|
|
for (size_t i = 0; i < str.length() && !requires_quotes; i++)
|
|
|
|
{
|
|
|
|
decoder(static_cast<uint8_t>(str[i]));
|
|
|
|
if (decoder.error())
|
|
|
|
requires_quotes = true;
|
|
|
|
else if (decoder.has_code_point())
|
|
|
|
requires_quotes = !impl::is_bare_key_character(decoder.codepoint);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (requires_quotes)
|
|
|
|
{
|
|
|
|
impl::print_to_stream(base::stream(), '"');
|
|
|
|
impl::print_to_stream_with_escapes(base::stream(), str);
|
|
|
|
impl::print_to_stream(base::stream(), '"');
|
|
|
|
}
|
|
|
|
else
|
|
|
|
impl::print_to_stream(base::stream(), str);
|
|
|
|
}
|
|
|
|
base::clear_naked_newline();
|
|
|
|
}
|
|
|
|
|
|
|
|
TOML_EXTERNAL_LINKAGE
|
|
|
|
void default_formatter::print_key_path()
|
|
|
|
{
|
|
|
|
for (const auto& segment : key_path_)
|
|
|
|
{
|
|
|
|
if (std::addressof(segment) > key_path_.data())
|
|
|
|
impl::print_to_stream(base::stream(), '.');
|
|
|
|
impl::print_to_stream(base::stream(), segment);
|
|
|
|
}
|
|
|
|
base::clear_naked_newline();
|
|
|
|
}
|
|
|
|
|
|
|
|
TOML_EXTERNAL_LINKAGE
|
|
|
|
void default_formatter::print_inline(const table& tbl)
|
2020-05-20 10:27:06 +00:00
|
|
|
{
|
|
|
|
if (tbl.empty())
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::print_to_stream(base::stream(), "{}"sv);
|
2020-05-20 10:27:06 +00:00
|
|
|
else
|
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::print_to_stream(base::stream(), "{ "sv);
|
2020-05-20 10:27:06 +00:00
|
|
|
|
|
|
|
bool first = false;
|
2020-06-08 15:31:23 +00:00
|
|
|
for (auto&& [k, v] : tbl)
|
2020-05-20 10:27:06 +00:00
|
|
|
{
|
|
|
|
if (first)
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::print_to_stream(base::stream(), ", "sv);
|
2020-05-20 10:27:06 +00:00
|
|
|
first = true;
|
|
|
|
|
|
|
|
print_key_segment(k);
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::print_to_stream(base::stream(), " = "sv);
|
2020-05-20 10:27:06 +00:00
|
|
|
|
|
|
|
const auto type = v.type();
|
|
|
|
TOML_ASSUME(type != node_type::none);
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case node_type::table: print_inline(*reinterpret_cast<const table*>(&v)); break;
|
|
|
|
case node_type::array: print(*reinterpret_cast<const array*>(&v)); break;
|
2021-10-23 09:22:41 +00:00
|
|
|
default: base::print_value(v, type);
|
2020-05-20 10:27:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::print_to_stream(base::stream(), " }"sv);
|
2020-05-20 10:27:06 +00:00
|
|
|
}
|
|
|
|
base::clear_naked_newline();
|
|
|
|
}
|
2021-05-11 21:48:53 +00:00
|
|
|
|
2020-07-13 18:18:04 +00:00
|
|
|
TOML_EXTERNAL_LINKAGE
|
2021-10-24 22:04:23 +00:00
|
|
|
void default_formatter::print(const array& arr)
|
2020-07-13 18:18:04 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
if (arr.empty())
|
|
|
|
impl::print_to_stream(base::stream(), "[]"sv);
|
|
|
|
else
|
2020-07-13 18:18:04 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
const auto original_indent = base::indent();
|
2021-10-29 08:12:41 +00:00
|
|
|
const auto multiline = forces_multiline(
|
|
|
|
arr,
|
|
|
|
base::indent_columns() * static_cast<size_t>(original_indent < 0 ? 0 : original_indent));
|
2021-10-24 22:04:23 +00:00
|
|
|
impl::print_to_stream(base::stream(), "["sv);
|
|
|
|
if (multiline)
|
|
|
|
{
|
|
|
|
if (original_indent < 0)
|
|
|
|
base::indent(0);
|
2021-10-27 13:03:05 +00:00
|
|
|
if (base::indent_array_elements())
|
|
|
|
base::increase_indent();
|
2021-10-24 22:04:23 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
impl::print_to_stream(base::stream(), ' ');
|
|
|
|
|
|
|
|
for (size_t i = 0; i < arr.size(); i++)
|
|
|
|
{
|
2021-10-26 13:49:23 +00:00
|
|
|
if (i > 0u)
|
2021-10-24 22:04:23 +00:00
|
|
|
{
|
|
|
|
impl::print_to_stream(base::stream(), ',');
|
|
|
|
if (!multiline)
|
|
|
|
impl::print_to_stream(base::stream(), ' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (multiline)
|
|
|
|
{
|
|
|
|
base::print_newline(true);
|
|
|
|
base::print_indent();
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& v = arr[i];
|
|
|
|
const auto type = v.type();
|
|
|
|
TOML_ASSUME(type != node_type::none);
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case node_type::table: print_inline(*reinterpret_cast<const table*>(&v)); break;
|
|
|
|
case node_type::array: print(*reinterpret_cast<const array*>(&v)); break;
|
|
|
|
default: base::print_value(v, type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (multiline)
|
|
|
|
{
|
|
|
|
base::indent(original_indent);
|
|
|
|
base::print_newline(true);
|
|
|
|
base::print_indent();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
impl::print_to_stream(base::stream(), ' ');
|
|
|
|
impl::print_to_stream(base::stream(), "]"sv);
|
2020-07-13 18:18:04 +00:00
|
|
|
}
|
2021-10-24 22:04:23 +00:00
|
|
|
base::clear_naked_newline();
|
2020-07-13 18:18:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TOML_EXTERNAL_LINKAGE
|
2021-10-24 22:04:23 +00:00
|
|
|
void default_formatter::print(const table& tbl)
|
2020-07-13 18:18:04 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
static constexpr auto is_non_inline_array_of_tables = [](auto&& nde) noexcept
|
|
|
|
{
|
|
|
|
auto arr = nde.as_array();
|
2021-10-26 13:49:23 +00:00
|
|
|
return arr && arr->is_array_of_tables() && !arr->template get_as<table>(0u)->is_inline();
|
2021-10-24 22:04:23 +00:00
|
|
|
};
|
2020-07-13 18:18:04 +00:00
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
// values, arrays, and inline tables/table arrays
|
|
|
|
for (auto&& [k, v] : tbl)
|
2020-07-13 18:18:04 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
const auto type = v.type();
|
|
|
|
if ((type == node_type::table && !reinterpret_cast<const table*>(&v)->is_inline())
|
|
|
|
|| (type == node_type::array && is_non_inline_array_of_tables(v)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
pending_table_separator_ = true;
|
|
|
|
base::print_newline();
|
|
|
|
base::print_indent();
|
|
|
|
print_key_segment(k);
|
|
|
|
impl::print_to_stream(base::stream(), " = "sv);
|
|
|
|
TOML_ASSUME(type != node_type::none);
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case node_type::table: print_inline(*reinterpret_cast<const table*>(&v)); break;
|
|
|
|
case node_type::array: print(*reinterpret_cast<const array*>(&v)); break;
|
|
|
|
default: base::print_value(v, type);
|
|
|
|
}
|
2020-07-13 18:18:04 +00:00
|
|
|
}
|
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
// non-inline tables
|
|
|
|
for (auto&& [k, v] : tbl)
|
|
|
|
{
|
|
|
|
const auto type = v.type();
|
|
|
|
if (type != node_type::table || reinterpret_cast<const table*>(&v)->is_inline())
|
|
|
|
continue;
|
|
|
|
auto& child_tbl = *reinterpret_cast<const table*>(&v);
|
|
|
|
|
|
|
|
// we can skip indenting and emitting the headers for tables that only contain other tables
|
|
|
|
// (so we don't over-nest)
|
|
|
|
size_t child_value_count{}; // includes inline tables and non-table arrays
|
|
|
|
size_t child_table_count{};
|
|
|
|
size_t child_table_array_count{};
|
|
|
|
for (auto&& [child_k, child_v] : child_tbl)
|
|
|
|
{
|
|
|
|
(void)child_k;
|
|
|
|
const auto child_type = child_v.type();
|
|
|
|
TOML_ASSUME(child_type != node_type::none);
|
|
|
|
switch (child_type)
|
|
|
|
{
|
|
|
|
case node_type::table:
|
|
|
|
if (reinterpret_cast<const table*>(&child_v)->is_inline())
|
|
|
|
child_value_count++;
|
|
|
|
else
|
|
|
|
child_table_count++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case node_type::array:
|
|
|
|
if (is_non_inline_array_of_tables(child_v))
|
|
|
|
child_table_array_count++;
|
|
|
|
else
|
|
|
|
child_value_count++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: child_value_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bool skip_self = false;
|
2021-10-26 13:49:23 +00:00
|
|
|
if (child_value_count == 0u && (child_table_count > 0u || child_table_array_count > 0u))
|
2021-10-24 22:04:23 +00:00
|
|
|
skip_self = true;
|
|
|
|
|
|
|
|
key_path_.push_back(make_key_segment(k));
|
|
|
|
|
|
|
|
if (!skip_self)
|
|
|
|
{
|
|
|
|
print_pending_table_separator();
|
2021-10-27 13:03:05 +00:00
|
|
|
if (base::indent_sub_tables())
|
|
|
|
base::increase_indent();
|
2021-10-24 22:04:23 +00:00
|
|
|
base::print_indent();
|
|
|
|
impl::print_to_stream(base::stream(), "["sv);
|
|
|
|
print_key_path();
|
|
|
|
impl::print_to_stream(base::stream(), "]"sv);
|
|
|
|
pending_table_separator_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
print(child_tbl);
|
|
|
|
|
|
|
|
key_path_.pop_back();
|
2021-10-27 13:03:05 +00:00
|
|
|
if (!skip_self && base::indent_sub_tables())
|
2021-10-24 22:04:23 +00:00
|
|
|
base::decrease_indent();
|
|
|
|
}
|
|
|
|
|
|
|
|
// table arrays
|
|
|
|
for (auto&& [k, v] : tbl)
|
|
|
|
{
|
|
|
|
if (!is_non_inline_array_of_tables(v))
|
|
|
|
continue;
|
|
|
|
auto& arr = *reinterpret_cast<const array*>(&v);
|
|
|
|
|
2021-10-27 13:03:05 +00:00
|
|
|
if (base::indent_sub_tables())
|
|
|
|
base::increase_indent();
|
2021-10-24 22:04:23 +00:00
|
|
|
key_path_.push_back(make_key_segment(k));
|
|
|
|
|
|
|
|
for (size_t i = 0; i < arr.size(); i++)
|
|
|
|
{
|
|
|
|
print_pending_table_separator();
|
|
|
|
base::print_indent();
|
|
|
|
impl::print_to_stream(base::stream(), "[["sv);
|
|
|
|
print_key_path();
|
|
|
|
impl::print_to_stream(base::stream(), "]]"sv);
|
|
|
|
pending_table_separator_ = true;
|
|
|
|
print(*reinterpret_cast<const table*>(&arr[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
key_path_.pop_back();
|
2021-10-27 13:03:05 +00:00
|
|
|
if (base::indent_sub_tables())
|
|
|
|
base::decrease_indent();
|
2021-10-24 22:04:23 +00:00
|
|
|
}
|
|
|
|
}
|
2020-07-13 18:18:04 +00:00
|
|
|
|
|
|
|
TOML_EXTERNAL_LINKAGE
|
2021-10-24 22:04:23 +00:00
|
|
|
void default_formatter::print()
|
2020-07-13 18:18:04 +00:00
|
|
|
{
|
2021-10-24 22:04:23 +00:00
|
|
|
if (base::dump_failed_parse_result())
|
|
|
|
return;
|
2020-07-13 18:18:04 +00:00
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
switch (auto source_type = base::source().type())
|
|
|
|
{
|
|
|
|
case node_type::table:
|
|
|
|
{
|
|
|
|
auto& tbl = *reinterpret_cast<const table*>(&base::source());
|
|
|
|
if (tbl.is_inline())
|
|
|
|
print_inline(tbl);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
base::decrease_indent(); // so root kvps and tables have the same indent
|
|
|
|
print(tbl);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-07-13 18:18:04 +00:00
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
case node_type::array: print(*reinterpret_cast<const array*>(&base::source())); break;
|
2020-07-13 18:18:04 +00:00
|
|
|
|
2021-10-24 22:04:23 +00:00
|
|
|
default: base::print_value(base::source(), source_type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TOML_NAMESPACE_END;
|
2020-07-18 12:10:19 +00:00
|
|
|
|
2021-10-24 10:21:32 +00:00
|
|
|
#include "header_end.h"
|