summaryrefslogtreecommitdiff
path: root/src/lib/eolian_cxx/grammar/class_declaration.hpp
blob: 172c07b7870025eaa9eaa8584c1c087a291cd5cd (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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
#ifndef EOLIAN_CXX_CLASS_DECLARATION_HH
#define EOLIAN_CXX_CLASS_DECLARATION_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"

namespace efl { namespace eolian { namespace grammar {

struct class_declaration_generator
{
   bool type_traits;
  
   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);

     std::string guard_symbol;

     if(!as_generator(*(string << "_") << string << "_FWD_GUARD")
        .generate(std::back_inserter(guard_symbol)
                  , std::make_tuple(cpp_namespaces, cls.cxx_name), add_upper_case_context(context)))
       return false;

     if(!as_generator(   "#ifndef " << guard_symbol << "\n"
                      << "#define " << guard_symbol << "\n")
        .generate(sink, std::make_tuple(), context))
       return false;
     
     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 " << cxx_class_name << ";\n"
        ).generate(sink, cls.cxx_name, context)) return false;

     auto close_namespace = *(lit("} ")) << "\n";
     if(!as_generator(close_namespace).generate(sink, cpp_namespaces, context)) return false;

     // if(type_traits)
       if(!as_generator
          (
           "namespace efl { namespace eo { template<> struct is_eolian_object< "
           "::" << *(lower_case[string] << "::") << cxx_class_name << "> : ::std::true_type {}; } }\n"
           "namespace efl { namespace eo { template<> struct is_eolian_object< "
           "::" << *(lower_case[string] << "::") << cxx_class_name << "&> : ::std::true_type {}; } }\n"
           "namespace efl { namespace eo { template<> struct is_eolian_object< "
           "::" << *(lower_case[string] << "::") << cxx_class_name << " const> : ::std::true_type {}; } }\n"
           "namespace efl { namespace eo { template<> struct is_eolian_object< "
           "::" << *(lower_case[string] << "::") << cxx_class_name << " const&> : ::std::true_type {}; } }\n"
           ).generate(sink, std::make_tuple
                      (
                       cpp_namespaces, cls.cxx_name, cpp_namespaces, cls.cxx_name
                       , cpp_namespaces, cls.cxx_name, cpp_namespaces, cls.cxx_name
                      ), context)) return false;

     if(!as_generator("#endif\n")
        .generate(sink, std::make_tuple(), context))
       return false;

     
     return true;
   }
};

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

namespace type_traits {
template <>
struct attributes_needed<class_declaration_generator> : std::integral_constant<int, 1> {};
}
      
class_declaration_generator const class_declaration = {true};
class_declaration_generator const class_forward_declaration = {false};
      
} } }

#endif