efl/src/lib/eolian_cxx/grammar/base_class_definition.hpp

122 lines
4.3 KiB
C++

#ifndef EOLIAN_CXX_BASE_CLASS_DEFINITION_HH
#define EOLIAN_CXX_BASE_CLASS_DEFINITION_HH
#include "grammar/generator.hpp"
#include "grammar/klass_def.hpp"
#include "grammar/indentation.hpp"
#include "grammar/list.hpp"
#include "grammar/alternative.hpp"
#include "grammar/type.hpp"
#include "grammar/parameter.hpp"
#include "grammar/function_declaration.hpp"
#include "grammar/namespace.hpp"
#include "grammar/case.hpp"
#include "grammar/address_of.hpp"
#include "grammar/attribute_reorder.hpp"
#include "grammar/part_declaration.hpp"
namespace efl { namespace eolian { namespace grammar {
struct base_class_definition_generator
{
template <typename OutputIterator, typename Context>
bool generate(OutputIterator sink, attributes::klass_def const& cls, Context const& context) const
{
std::vector<std::string> cpp_namespaces = attributes::cpp_namespaces(cls.namespaces);
// static_assert(std::is_same<OutputIterator, void>::value, "");
auto open_namespace = *("namespace " << string << " { ") << "\n";
if(!as_generator(open_namespace).generate
(sink, cpp_namespaces, add_lower_case_context(context))) return false;
if(!as_generator
(
"struct " << string << " {\n"
).generate(sink, cls.cxx_name, context)) return false;
if(!as_generator(*(function_declaration(get_klass_name(cls))))
.generate(sink, cls.functions, context)) return false;
if(!as_generator(*(part_declaration << ";\n"))
.generate(sink, cls.parts, context)) return false;
// static Efl_Class const* _eo_class();
std::string suffix;
switch(cls.type)
{
case attributes::class_type::regular:
case attributes::class_type::abstract_:
suffix = "CLASS";
break;
case attributes::class_type::mixin:
suffix = "MIXIN";
break;
case attributes::class_type::interface_:
suffix = "INTERFACE";
break;
}
if(!as_generator
(
scope_tab << "static Efl_Class const* _eo_class()\n"
<< scope_tab << "{\n"
<< scope_tab << scope_tab << "return "
).generate(sink, attributes::unused, context)) return false;
if(!as_generator
(*(lower_case[string] << "_") << string << "_" << string)
.generate(sink, std::make_tuple(cls.namespaces, cls.eolian_name, suffix), add_upper_case_context(context)))
return false;
if(!as_generator(";\n" << scope_tab << "}\n").generate(sink, attributes::unused, context)) return false;
if(!as_generator(
scope_tab << "Eo* _eo_ptr() const { return *(reinterpret_cast<Eo **>"
<< "(const_cast<" << string << " *>(this))); }\n"
).generate(sink, cls.cxx_name, context))
return false;
// operator ::ns::Class_Name() const;
// operator ::ns::Class_Name&();
// operator ::ns::Class_Name const&() const;
auto class_name = *(lit("::") << lower_case[string]) << "::" << string;
if(!as_generator
(
attribute_reorder<0, 1, 0, 1, 0, 1, 0, 1>
(
scope_tab << "operator " << class_name << "() const;\n"
<< scope_tab << "operator " << class_name << "&();\n"
<< scope_tab << "operator " << class_name << " const&() const;\n"
)).generate(sink, std::make_tuple(cpp_namespaces, cls.cxx_name), context))
return false;
// /// @cond LOCAL
if(!as_generator(scope_tab << "/// @cond LOCAL\n").generate(sink, attributes::unused, context)) return false;
if(!as_generator(address_of).generate(sink, cls, context)) return false;
// /// @endcond
if(!as_generator(scope_tab << "/// @endcond\n").generate(sink, attributes::unused, context)) return false;
if(!as_generator("};\n").generate(sink, attributes::unused, context)) return false;
auto close_namespace = *(lit("} ")) << "\n";
if(!as_generator(close_namespace).generate(sink, cpp_namespaces, context)) return false;
return true;
}
};
template <>
struct is_eager_generator<base_class_definition_generator> : std::true_type {};
template <>
struct is_generator<base_class_definition_generator> : std::true_type {};
namespace type_traits {
template <>
struct attributes_needed<base_class_definition_generator> : std::integral_constant<int, 1> {};
}
base_class_definition_generator const base_class_definition = {};
} } }
#endif