//# This file is a part of toml++ and is subject to the the terms of the MIT license. //# Copyright (c) 2019-2020 Mark Gillard //# See https://github.com/marzer/tomlplusplus/blob/master/LICENSE for the full license text. // SPDX-License-Identifier: MIT #pragma once //# {{ #include "toml_preprocessor.h" #if !TOML_IMPLEMENTATION #error This is an implementation-only header. #endif //# }} #include "toml_table.h" #include "toml_node_view.h" TOML_NAMESPACE_START { #if TOML_LIFETIME_HOOKS TOML_EXTERNAL_LINKAGE void table::lh_ctor() noexcept { TOML_TABLE_CREATED; } TOML_EXTERNAL_LINKAGE void table::lh_dtor() noexcept { TOML_TABLE_DESTROYED; } #endif TOML_EXTERNAL_LINKAGE table::table() noexcept { #if TOML_LIFETIME_HOOKS lh_ctor(); #endif } TOML_EXTERNAL_LINKAGE table::table(const table& other) noexcept : node{ std::move(other) }, inline_{ other.inline_ } { for (auto&& [k, v] : other) map.emplace_hint(map.end(), k, impl::make_node(v)); #if TOML_LIFETIME_HOOKS lh_ctor(); #endif } TOML_EXTERNAL_LINKAGE table::table(table&& other) noexcept : node{ std::move(other) }, map{ std::move(other.map) }, inline_{ other.inline_ } { #if TOML_LIFETIME_HOOKS lh_ctor(); #endif } TOML_EXTERNAL_LINKAGE table& table::operator= (const table& rhs) noexcept { if (&rhs != this) { node::operator=(rhs); map.clear(); for (auto&& [k, v] : rhs) map.emplace_hint(map.end(), k, impl::make_node(v)); inline_ = rhs.inline_; } return *this; } TOML_EXTERNAL_LINKAGE table& table::operator= (table&& rhs) noexcept { if (&rhs != this) { node::operator=(std::move(rhs)); map = std::move(rhs.map); inline_ = rhs.inline_; } return *this; } TOML_EXTERNAL_LINKAGE table::~table() noexcept { #if TOML_LIFETIME_HOOKS lh_dtor(); #endif } TOML_EXTERNAL_LINKAGE table::table(impl::table_init_pair* pairs, size_t count) noexcept { for (size_t i = 0; i < count; i++) { if (!pairs[i].value) // empty node_views continue; map.insert_or_assign( std::move(pairs[i].key), std::move(pairs[i].value) ); } } #define TOML_MEMBER_ATTR(attr) TOML_EXTERNAL_LINKAGE TOML_ATTR(attr) TOML_MEMBER_ATTR(const) node_type table::type() const noexcept { return node_type::table; } TOML_MEMBER_ATTR(const) bool table::is_table() const noexcept { return true; } TOML_MEMBER_ATTR(const) bool table::is_array() const noexcept { return false; } TOML_MEMBER_ATTR(const) bool table::is_value() const noexcept { return false; } TOML_MEMBER_ATTR(const) const table* table::as_table() const noexcept { return this; } TOML_MEMBER_ATTR(const) table* table::as_table() noexcept { return this; } TOML_MEMBER_ATTR(pure) bool table::is_inline() const noexcept { return inline_; } TOML_EXTERNAL_LINKAGE void table::is_inline(bool val) noexcept { inline_ = val; } TOML_EXTERNAL_LINKAGE table::const_iterator table::begin() const noexcept { return { map.begin() }; } TOML_EXTERNAL_LINKAGE table::const_iterator table::end() const noexcept { return { map.end() }; } TOML_EXTERNAL_LINKAGE table::const_iterator table::cbegin() const noexcept { return { map.cbegin() }; } TOML_EXTERNAL_LINKAGE table::const_iterator table::cend() const noexcept { return { map.cend() }; } TOML_EXTERNAL_LINKAGE table::iterator table::begin() noexcept { return { map.begin() }; } TOML_EXTERNAL_LINKAGE table::iterator table::end() noexcept { return { map.end() }; } TOML_MEMBER_ATTR(pure) bool table::empty() const noexcept { return map.empty(); } TOML_MEMBER_ATTR(pure) size_t table::size() const noexcept { return map.size(); } TOML_EXTERNAL_LINKAGE void table::clear() noexcept { map.clear(); } #undef TOML_MEMBER_ATTR TOML_EXTERNAL_LINKAGE bool table::is_homogeneous(node_type ntype) const noexcept { if (map.empty()) return false; if (ntype == node_type::none) ntype = map.cbegin()->second->type(); for (const auto& [k, v] : map) { (void)k; if (v->type() != ntype) return false; } return true; } namespace impl { template TOML_INTERNAL_LINKAGE bool table_is_homogeneous(T& map, node_type ntype, U& first_nonmatch) noexcept { if (map.empty()) { first_nonmatch = {}; return false; } if (ntype == node_type::none) ntype = map.cbegin()->second->type(); for (const auto& [k, v] : map) { (void)k; if (v->type() != ntype) { first_nonmatch = v.get(); return false; } } return true; } } TOML_EXTERNAL_LINKAGE bool table::is_homogeneous(node_type ntype, toml::node*& first_nonmatch) noexcept { return impl::table_is_homogeneous(map, ntype, first_nonmatch); } TOML_EXTERNAL_LINKAGE bool table::is_homogeneous(node_type ntype, const toml::node*& first_nonmatch) const noexcept { return impl::table_is_homogeneous(map, ntype, first_nonmatch); } TOML_EXTERNAL_LINKAGE node_view table::operator[] (std::string_view key) noexcept { return node_view{ this->get(key) }; } TOML_EXTERNAL_LINKAGE node_view table::operator[] (std::string_view key) const noexcept { return node_view{ this->get(key) }; } TOML_EXTERNAL_LINKAGE table::iterator table::erase(iterator pos) noexcept { return { map.erase(pos.raw_) }; } TOML_EXTERNAL_LINKAGE table::iterator table::erase(const_iterator pos) noexcept { return { map.erase(pos.raw_) }; } TOML_EXTERNAL_LINKAGE table::iterator table::erase(const_iterator first, const_iterator last) noexcept { return { map.erase(first.raw_, last.raw_) }; } TOML_EXTERNAL_LINKAGE bool table::erase(std::string_view key) noexcept { if (auto it = map.find(key); it != map.end()) { map.erase(it); return true; } return false; } TOML_EXTERNAL_LINKAGE node* table::get(std::string_view key) noexcept { return do_get(map, key); } TOML_EXTERNAL_LINKAGE const node* table::get(std::string_view key) const noexcept { return do_get(map, key); } TOML_EXTERNAL_LINKAGE table::iterator table::find(std::string_view key) noexcept { return { map.find(key) }; } TOML_EXTERNAL_LINKAGE table::const_iterator table::find(std::string_view key) const noexcept { return { map.find(key) }; } TOML_EXTERNAL_LINKAGE bool table::contains(std::string_view key) const noexcept { return do_contains(map, key); } #if TOML_WINDOWS_COMPAT TOML_EXTERNAL_LINKAGE node_view table::operator[] (std::wstring_view key) noexcept { return node_view{ this->get(key) }; } TOML_EXTERNAL_LINKAGE node_view table::operator[] (std::wstring_view key) const noexcept { return node_view{ this->get(key) }; } TOML_EXTERNAL_LINKAGE bool table::erase(std::wstring_view key) noexcept { return erase(impl::narrow(key)); } TOML_EXTERNAL_LINKAGE node* table::get(std::wstring_view key) noexcept { return get(impl::narrow(key)); } TOML_EXTERNAL_LINKAGE const node* table::get(std::wstring_view key) const noexcept { return get(impl::narrow(key)); } TOML_EXTERNAL_LINKAGE table::iterator table::find(std::wstring_view key) noexcept { return find(impl::narrow(key)); } TOML_EXTERNAL_LINKAGE table::const_iterator table::find(std::wstring_view key) const noexcept { return find(impl::narrow(key)); } TOML_EXTERNAL_LINKAGE bool table::contains(std::wstring_view key) const noexcept { return contains(impl::narrow(key)); } #endif // TOML_WINDOWS_COMPAT TOML_API TOML_EXTERNAL_LINKAGE bool operator == (const table& lhs, const table& rhs) noexcept { if (&lhs == &rhs) return true; if (lhs.map.size() != rhs.map.size()) return false; for (auto l = lhs.map.begin(), r = rhs.map.begin(), e = lhs.map.end(); l != e; l++, r++) { if (l->first != r->first) return false; const auto lhs_type = l->second->type(); const node& rhs_ = *r->second; const auto rhs_type = rhs_.type(); if (lhs_type != rhs_type) return false; const bool equal = l->second->visit([&](const auto& lhs_) noexcept { return lhs_ == *reinterpret_cast*>(&rhs_); }); if (!equal) return false; } return true; } TOML_API TOML_EXTERNAL_LINKAGE bool operator != (const table& lhs, const table& rhs) noexcept { return !(lhs == rhs); } } TOML_NAMESPACE_END