summaryrefslogtreecommitdiff
path: root/src/lib/eolian_cxx/grammar/class_implementation.hpp
blob: b2578b5bf2856ef9cd2520a17356fc0cc661c5e4 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
#ifndef EOLIAN_CXX_CLASS_IMPLEMENTATION_HH
#define EOLIAN_CXX_CLASS_IMPLEMENTATION_HH

#include "grammar/generator.hpp"
#include "grammar/klass_def.hpp"

#include "grammar/string.hpp"
#include "grammar/indentation.hpp"
#include "grammar/list.hpp"
#include "grammar/alternative.hpp"
#include "grammar/type.hpp"
#include "grammar/parameter.hpp"
#include "grammar/function_definition.hpp"
#include "grammar/namespace.hpp"
#include "grammar/type_impl.hpp"
#include "grammar/attribute_reorder.hpp"

namespace efl { namespace eolian { namespace grammar {

struct class_implementation_generator
{
   template <typename OutputIterator, typename Context>
   bool generate(OutputIterator sink, attributes::klass_def const& cls, Context const& ctx) const
   {
     std::vector<std::string> cpp_namespaces = attributes::cpp_namespaces(cls.namespaces);
     auto base_class_name = *(lower_case[string] << "::") << string;
     auto class_name = *(lit("::") << lower_case[string]) << "::" << string;
     return as_generator
       (
#ifndef USE_EOCXX_INHERIT_ONLY
        (namespaces
         [*function_definition(get_klass_name(cls))]
         << "\n"
       )).generate(sink, std::make_tuple(cls.namespaces, cls.functions), ctx)
       && as_generator
       (
#endif
        attribute_reorder<0, 1, 2, 3, 2, 3, 2, 3, 2, 3, 2, 3, 2, 3, 2, 3, 2, 3, 2, 3>
        (
         "namespace eo_cxx {\n"
         << namespaces
         [
          *function_definition(get_klass_name(cls))
          << "inline " << base_class_name << "::operator " << class_name << "() const { return *static_cast< "
            << class_name << " const*>(static_cast<void const*>(this)); }\n"
          << "inline " << base_class_name << "::operator " << class_name << "&() { return *static_cast< "
            << class_name << "*>(static_cast<void*>(this)); }\n"
          << "inline " << base_class_name << "::operator " << class_name << " const&() const { return *static_cast< "
            << class_name << " const*>(static_cast<void const*>(this)); }\n"
         ]
         << "}\n\n"
         )).generate(sink, std::make_tuple(cls.namespaces, cls.functions, cpp_namespaces, cls.cxx_name), ctx);
   }
};

template <>
struct is_eager_generator<class_implementation_generator> : std::true_type {};
template <>
struct is_generator<class_implementation_generator> : std::true_type {};

namespace type_traits {
template <>
struct attributes_needed<class_implementation_generator> : std::integral_constant<int, 1> {};
}
      
class_implementation_generator const class_implementation = {};
      
} } }

#endif