Chris@16
|
1 /*=============================================================================
|
Chris@16
|
2 Copyright (c) 2001-2007 Joel de Guzman
|
Chris@16
|
3
|
Chris@16
|
4 Distributed under the Boost Software License, Version 1.0. (See accompanying
|
Chris@16
|
5 file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
6 ==============================================================================*/
|
Chris@16
|
7 #ifndef PHOENIX_STATEMENT_DETAIL_SWITCH_HPP
|
Chris@16
|
8 #define PHOENIX_STATEMENT_DETAIL_SWITCH_HPP
|
Chris@16
|
9
|
Chris@16
|
10 #include <boost/spirit/home/phoenix/core/nothing.hpp>
|
Chris@16
|
11 #include <boost/fusion/include/vector.hpp>
|
Chris@16
|
12 #include <boost/fusion/include/as_vector.hpp>
|
Chris@16
|
13 #include <boost/fusion/include/push_back.hpp>
|
Chris@16
|
14 #include <boost/fusion/include/push_front.hpp>
|
Chris@16
|
15 #include <boost/fusion/include/begin.hpp>
|
Chris@16
|
16 #include <boost/fusion/include/size.hpp>
|
Chris@16
|
17 #include <boost/fusion/include/value_of.hpp>
|
Chris@16
|
18 #include <boost/fusion/include/is_sequence.hpp>
|
Chris@16
|
19 #include <boost/mpl/identity.hpp>
|
Chris@16
|
20 #include <boost/mpl/bool.hpp>
|
Chris@16
|
21 #include <boost/mpl/eval_if.hpp>
|
Chris@16
|
22 #include <boost/mpl/if.hpp>
|
Chris@16
|
23
|
Chris@16
|
24 namespace boost { namespace phoenix
|
Chris@16
|
25 {
|
Chris@16
|
26
|
Chris@16
|
27 template <typename Actor, typename K, K Value>
|
Chris@16
|
28 struct switch_case;
|
Chris@16
|
29
|
Chris@16
|
30 template <typename Actor>
|
Chris@16
|
31 struct default_case;
|
Chris@16
|
32
|
Chris@16
|
33 namespace detail
|
Chris@16
|
34 {
|
Chris@16
|
35 template <typename T>
|
Chris@16
|
36 struct is_default_case : mpl::bool_<T::is_default> {};
|
Chris@16
|
37
|
Chris@16
|
38 template <typename A0, typename A1>
|
Chris@16
|
39 struct compose_case_a
|
Chris@16
|
40 {
|
Chris@16
|
41 // here, A0 and A1 are both switch cases
|
Chris@16
|
42 typedef typename
|
Chris@16
|
43 mpl::if_<
|
Chris@16
|
44 is_default_case<A1>
|
Chris@16
|
45 , fusion::vector<actor<A1>, actor<A0> >
|
Chris@16
|
46 , fusion::vector<actor<A0>, actor<A1> >
|
Chris@16
|
47 >::type
|
Chris@16
|
48 type;
|
Chris@16
|
49
|
Chris@16
|
50 static type
|
Chris@16
|
51 eval(A0 const& _0, A1 const& _1, mpl::false_)
|
Chris@16
|
52 {
|
Chris@16
|
53 return type(_0, _1);
|
Chris@16
|
54 }
|
Chris@16
|
55
|
Chris@16
|
56 static type
|
Chris@16
|
57 eval(A0 const& _0, A1 const& _1, mpl::true_)
|
Chris@16
|
58 {
|
Chris@16
|
59 return type(_1, _0);
|
Chris@16
|
60 }
|
Chris@16
|
61
|
Chris@16
|
62 static type
|
Chris@16
|
63 eval(A0 const& _0, A1 const& _1)
|
Chris@16
|
64 {
|
Chris@16
|
65 return eval(_0, _1, is_default_case<A1>());
|
Chris@16
|
66 }
|
Chris@16
|
67 };
|
Chris@16
|
68
|
Chris@16
|
69 template <typename Seq, typename Case>
|
Chris@16
|
70 struct compose_case_b
|
Chris@16
|
71 {
|
Chris@16
|
72 typedef typename fusion::result_of::as_vector<
|
Chris@16
|
73 typename mpl::eval_if<
|
Chris@16
|
74 is_default_case<Case>
|
Chris@16
|
75 , fusion::result_of::push_front<Seq const, actor<Case> >
|
Chris@16
|
76 , fusion::result_of::push_back<Seq const, actor<Case> >
|
Chris@16
|
77 >::type>::type
|
Chris@16
|
78 type;
|
Chris@16
|
79
|
Chris@16
|
80 static type
|
Chris@16
|
81 eval(Seq const& seq, Case const& case_, mpl::false_)
|
Chris@16
|
82 {
|
Chris@16
|
83 return fusion::as_vector(
|
Chris@16
|
84 fusion::push_back(seq, actor<Case>(case_)));
|
Chris@16
|
85 }
|
Chris@16
|
86
|
Chris@16
|
87 static type
|
Chris@16
|
88 eval(Seq const& seq, Case const& case_, mpl::true_)
|
Chris@16
|
89 {
|
Chris@16
|
90 return fusion::as_vector(
|
Chris@16
|
91 fusion::push_front(seq, actor<Case>(case_)));
|
Chris@16
|
92 }
|
Chris@16
|
93
|
Chris@16
|
94 static type
|
Chris@16
|
95 eval(Seq const& seq, Case const& case_)
|
Chris@16
|
96 {
|
Chris@16
|
97 return eval(seq, case_, is_default_case<Case>());
|
Chris@16
|
98 }
|
Chris@16
|
99 };
|
Chris@16
|
100
|
Chris@16
|
101 template <typename Cases>
|
Chris@16
|
102 struct ensure_default
|
Chris@16
|
103 {
|
Chris@16
|
104 typedef
|
Chris@16
|
105 is_default_case<
|
Chris@16
|
106 typename fusion::result_of::value_of<
|
Chris@16
|
107 typename fusion::result_of::begin<Cases>::type
|
Chris@16
|
108 >::type
|
Chris@16
|
109 >
|
Chris@16
|
110 is_default_case_;
|
Chris@16
|
111
|
Chris@16
|
112 typedef typename
|
Chris@16
|
113 mpl::eval_if<
|
Chris@16
|
114 is_default_case_
|
Chris@16
|
115 , mpl::identity<Cases>
|
Chris@16
|
116 , fusion::result_of::push_front<
|
Chris@16
|
117 Cases const, actor<default_case<actor<null_actor> > > >
|
Chris@16
|
118 >::type
|
Chris@16
|
119 type;
|
Chris@16
|
120
|
Chris@16
|
121 static type
|
Chris@16
|
122 eval(Cases const& cases, mpl::false_);
|
Chris@16
|
123
|
Chris@16
|
124 static type
|
Chris@16
|
125 eval(Cases const& cases, mpl::true_)
|
Chris@16
|
126 {
|
Chris@16
|
127 return cases;
|
Chris@16
|
128 }
|
Chris@16
|
129
|
Chris@16
|
130 static type
|
Chris@16
|
131 eval(Cases const& cases)
|
Chris@16
|
132 {
|
Chris@16
|
133 return eval(cases, is_default_case_());
|
Chris@16
|
134 }
|
Chris@16
|
135 };
|
Chris@16
|
136
|
Chris@16
|
137 template <typename Cond, typename Cases>
|
Chris@16
|
138 struct switch_composite
|
Chris@16
|
139 {
|
Chris@16
|
140 BOOST_STATIC_ASSERT(fusion::traits::is_sequence<Cases>::value);
|
Chris@16
|
141 typedef ensure_default<Cases> ensure_default_;
|
Chris@16
|
142
|
Chris@16
|
143 typedef typename
|
Chris@16
|
144 fusion::result_of::as_vector<
|
Chris@16
|
145 typename fusion::result_of::push_front<
|
Chris@16
|
146 typename ensure_default_::type, Cond>::type
|
Chris@16
|
147 >::type
|
Chris@16
|
148 tuple_type;
|
Chris@16
|
149
|
Chris@16
|
150 typedef
|
Chris@16
|
151 composite<
|
Chris@16
|
152 detail::switch_eval<fusion::result_of::size<tuple_type>::value-2>
|
Chris@16
|
153 , tuple_type>
|
Chris@16
|
154 type;
|
Chris@16
|
155
|
Chris@16
|
156 static type
|
Chris@16
|
157 eval(Cond const& cond, Cases const& cases)
|
Chris@16
|
158 {
|
Chris@16
|
159 return fusion::as_vector(
|
Chris@16
|
160 fusion::push_front(ensure_default_::eval(cases), cond));
|
Chris@16
|
161 }
|
Chris@16
|
162 };
|
Chris@16
|
163
|
Chris@16
|
164 template <typename Cond, typename Cases>
|
Chris@16
|
165 struct switch_composite_actor
|
Chris@16
|
166 {
|
Chris@16
|
167 typedef actor<typename switch_composite<Cond, Cases>::type> type;
|
Chris@16
|
168 };
|
Chris@16
|
169 }
|
Chris@16
|
170 }}
|
Chris@16
|
171
|
Chris@16
|
172 #endif
|