summaryrefslogtreecommitdiff
path: root/core/property_list.hpp
blob: 3fd5ee93c862478633b107b5d427b021c3bc0547 (plain)
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
90
91
92
93
94
#ifndef RAA4DT8yVWXBMqy1rxYiM12MKFA
#define RAA4DT8yVWXBMqy1rxYiM12MKFA

#include <boost/mpl/assert.hpp>
#include <boost/mpl/empty.hpp>
#include <boost/mpl/front.hpp>
#include <boost/mpl/list.hpp>
#include <boost/mpl/logical.hpp>
#include <boost/mpl/not.hpp>
#include <boost/mpl/pop_front.hpp>
#include <boost/static_assert.hpp>
#include <boost/type_traits/is_same.hpp>

#include "property_instance.hpp"
#include "template_helpers.hpp"

namespace PLImpl {

  using namespace boost::mpl;

  // forward decls
  template <typename Haystack, typename Needle> struct PL_Contains;


  /// PL -  Property List

  //  template <typename Head, bool doWriteResults, typename Tail>
  template <typename List>
  struct PL {
    // compile time data structure (types)
    typedef typename front<List>::type     head_t;
    typedef typename pop_front<List>::type tail_t;

    typedef typename head_t::first prop_t;
    typedef typename head_t::second doWriteResult_t;
    typedef typename if_<empty<tail_t>,
			 PL<Void>,
			 PL<tail_t>
			 >::type next_t;

    typedef PropertyInstance<prop_t, doWriteResult_t::value > data_t;

    // access methods
    //   template params:
    //     - PropComp: the list of all properties
    //     - Action: the action class to call
    template<typename PropComp, typename Action>
    inline typename Action::result_t call(PropComp &pc, Action &action) {
      typename Action::template local_state_t<prop_t, doWriteResult_t::value> state;
      action.pre(pc, data, state);
      tail.call<PropComp, Action>(pc, action);
      action.post(pc, data, state);
      return action.result;
    }

    // data members
    data_t data;
    next_t tail;

    // integrity checks:
    //   1. each property may be used only once
    BOOST_MPL_ASSERT(( not_< PL_Contains< next_t, prop_t > > )); 
    // TODO (beauty): dependencies (once they are implemented anywhere)
  };

  // the empty tail
  template<>
  struct PL<Void> {
    typedef Void prop_t;
    typedef PL<Void> tail_t;
    typedef Void data_t;

    template<typename PropComp, typename Action>
    inline typename Action::result_t call(PropComp &pc, Action &action) {
      return typename Action::result_t();
    }
  };

  /// PL Template Helpers
  template <typename Haystack, typename Needle>
  struct PL_Contains : boost::mpl::or_< 
    boost::is_same<typename Haystack::prop_t, Needle>,
    PL_Contains<typename Haystack::next_t, Needle>
    > { };

  template <typename Needle>
  struct PL_Contains<PL<Void>, Needle> : boost::mpl::false_ { };

};

using PLImpl::PL;
using PLImpl::PL_Contains;

#endif // RAA4DT8yVWXBMqy1rxYiM12MKFA
contact: Jan Huwald // Impressum