Add cpp_template_template_parameter

This commit is contained in:
Jonathan Müller 2017-02-01 21:04:53 +01:00
commit edb132981a
7 changed files with 153 additions and 3 deletions

View file

@ -43,6 +43,7 @@ namespace cppast
template_type_parameter_t,
non_type_template_parameter_t,
template_template_parameter_t,
alias_template_t,
@ -54,6 +55,9 @@ namespace cppast
/// \returns Whether or not a given entity kind is a C++ type.
bool is_type(cpp_entity_kind kind) noexcept;
/// \returns Whether or not a given entity kind is a C++ template.
bool is_template(cpp_entity_kind kind) noexcept;
} // namespace cppast
#endif // CPPAST_CPP_ENTITY_KIND_HPP_INCLUDED

View file

@ -36,8 +36,7 @@ namespace cppast
/// \effects Registers the namespace in the [cppast::cpp_entity_index](),
/// using the given [cppast::cpp_entity_id]().
/// \returns The finished namespace.
std::unique_ptr<cpp_namespace> finish(const cpp_entity_index& idx,
cpp_entity_id id) noexcept
std::unique_ptr<cpp_namespace> finish(const cpp_entity_index& idx, cpp_entity_id id)
{
idx.register_entity(std::move(id), type_safe::ref(*namespace_));
return std::move(namespace_);

View file

@ -5,7 +5,10 @@
#ifndef CPPAST_CPP_TEMPLATE_PARAMETER_HPP_INCLUDED
#define CPPAST_CPP_TEMPLATE_PARAMETER_HPP_INCLUDED
#include <type_safe/optional.hpp>
#include <cppast/cpp_entity.hpp>
#include <cppast/cpp_entity_container.hpp>
#include <cppast/cpp_variable_base.hpp>
namespace cppast
@ -124,7 +127,7 @@ namespace cppast
cpp_template_type_parameter_ref parameter_;
};
/// A [cppast::cpp_entity]() modelling a non-type template parameter.
/// A [cppast::cpp_entity]() modelling a C++ non-type template parameter.
class cpp_non_type_template_parameter final : public cpp_template_parameter,
public cpp_variable_base
{
@ -146,6 +149,94 @@ namespace cppast
cpp_entity_kind do_get_entity_kind() const noexcept override;
};
/// \exclude
namespace detail
{
struct cpp_template_ref_predicate
{
bool operator()(const cpp_entity& e);
};
} // namespace detail
class cpp_template;
/// A reference to a [cppast::cpp_template]().
using cpp_template_ref = basic_cpp_entity_ref<cpp_template, detail::cpp_template_ref_predicate>;
/// A [cppast::cpp_entity]() modelling a C++ template template parameter.
class cpp_template_template_parameter final
: public cpp_template_parameter,
public cpp_entity_container<cpp_template_template_parameter, cpp_template_parameter>
{
public:
/// Builds a [cppast::cpp_template_template_parameter]().
class builder
{
public:
/// \effects Sets the name and whether it is variadic.
builder(std::string name, bool variadic)
: parameter_(new cpp_template_template_parameter(std::move(name), variadic))
{
}
/// \effects Sets the keyword,
/// default is [cpp_template_keyword::keyword_class]().
void keyword(cpp_template_keyword kw)
{
parameter_->keyword_ = kw;
}
/// \effects Adds a parameter to the template.
void add_parameter(std::unique_ptr<cpp_template_parameter> param)
{
parameter_->add_child(std::move(param));
}
/// \effects Sets the default template.
void default_template(cpp_template_ref templ)
{
parameter_->default_ = std::move(templ);
}
/// \effects Registers the parameter in the [cppast::cpp_entity_index](),
/// using the given [cppast::cpp_entity_id]().
/// \returns The finished parameter.
std::unique_ptr<cpp_template_template_parameter> finish(const cpp_entity_index& idx,
cpp_entity_id id)
{
idx.register_entity(std::move(id), type_safe::ref(*parameter_));
return std::move(parameter_);
}
private:
std::unique_ptr<cpp_template_template_parameter> parameter_;
};
/// \returns The keyword used in the template parameter.
cpp_template_keyword keyword() const noexcept
{
return keyword_;
}
/// \returns A [ts::optional]() that is the default template.
type_safe::optional<cpp_template_ref> default_template() const noexcept
{
return default_;
}
private:
cpp_template_template_parameter(std::string name, bool variadic)
: cpp_template_parameter(std::move(name), variadic),
keyword_(cpp_template_keyword::keyword_class)
{
}
cpp_entity_kind do_get_entity_kind() const noexcept override;
type_safe::optional<cpp_template_ref> default_;
cpp_template_keyword keyword_;
};
} // namespace cppast
#endif // CPPAST_CPP_TEMPLATE_PARAMETER_HPP_INCLUDED

View file

@ -64,6 +64,8 @@ const char* cppast::to_string(cpp_entity_kind kind) noexcept
return "template type parameter";
case cpp_entity_kind::non_type_template_parameter_t:
return "non type template parameter";
case cpp_entity_kind::template_template_parameter_t:
return "template template parameter";
case cpp_entity_kind::alias_template_t:
return "alias template";
@ -104,6 +106,7 @@ bool cppast::is_type(cpp_entity_kind kind) noexcept
case cpp_entity_kind::destructor_t:
case cpp_entity_kind::template_type_parameter_t:
case cpp_entity_kind::non_type_template_parameter_t:
case cpp_entity_kind::template_template_parameter_t:
case cpp_entity_kind::alias_template_t:
case cpp_entity_kind::count:
break;
@ -111,3 +114,41 @@ bool cppast::is_type(cpp_entity_kind kind) noexcept
return false;
}
bool cppast::is_template(cpp_entity_kind kind) noexcept
{
switch (kind)
{
case cpp_entity_kind::alias_template_t:
return true;
case cpp_entity_kind::file_t:
case cpp_entity_kind::language_linkage_t:
case cpp_entity_kind::namespace_t:
case cpp_entity_kind::namespace_alias_t:
case cpp_entity_kind::using_directive_t:
case cpp_entity_kind::using_declaration_t:
case cpp_entity_kind::type_alias_t:
case cpp_entity_kind::enum_t:
case cpp_entity_kind::enum_value_t:
case cpp_entity_kind::class_t:
case cpp_entity_kind::access_specifier_t:
case cpp_entity_kind::base_class_t:
case cpp_entity_kind::variable_t:
case cpp_entity_kind::member_variable_t:
case cpp_entity_kind::bitfield_t:
case cpp_entity_kind::function_parameter_t:
case cpp_entity_kind::function_t:
case cpp_entity_kind::member_function_t:
case cpp_entity_kind::conversion_op_t:
case cpp_entity_kind::constructor_t:
case cpp_entity_kind::destructor_t:
case cpp_entity_kind::template_type_parameter_t:
case cpp_entity_kind::non_type_template_parameter_t:
case cpp_entity_kind::template_template_parameter_t:
case cpp_entity_kind::count:
break;
}
return false;
}

View file

@ -3,6 +3,7 @@
// found in the top-level directory of this distribution.
#include <cppast/cpp_template_parameter.hpp>
#include <cppast/cpp_entity_kind.hpp>
using namespace cppast;
@ -55,3 +56,13 @@ cpp_entity_kind cpp_non_type_template_parameter::do_get_entity_kind() const noex
{
return cpp_entity_kind::non_type_template_parameter_t;
}
bool detail::cpp_template_ref_predicate::operator()(const cpp_entity& e)
{
return is_template(e.kind());
}
cpp_entity_kind cpp_template_template_parameter::do_get_entity_kind() const noexcept
{
return cpp_entity_kind::template_template_parameter_t;
}

View file

@ -101,6 +101,7 @@ bool cppast::is_valid(const cpp_type& type) noexcept
case cpp_type_kind::builtin:
case cpp_type_kind::user_defined:
case cpp_type_kind::template_parameter:
case cpp_type_kind::unexposed:
// no further check required/possible
break;

View file

@ -14,6 +14,7 @@
#include <cppast/cpp_language_linkage.hpp>
#include <cppast/cpp_member_function.hpp>
#include <cppast/cpp_namespace.hpp>
#include <cppast/cpp_template_parameter.hpp>
using namespace cppast;
@ -58,6 +59,8 @@ bool detail::visit(const cpp_entity& e, detail::visitor_callback_t cb, void* fun
return handle_container<cpp_conversion_op>(e, cb, functor);
case cpp_entity_kind::constructor_t:
return handle_container<cpp_constructor>(e, cb, functor);
case cpp_entity_kind::template_template_parameter_t:
return handle_container<cpp_template_template_parameter>(e, cb, functor);
case cpp_entity_kind::alias_template_t:
return handle_container<cpp_alias_template>(e, cb, functor);