1 | /*
|
---|
2 | * Action_impl_header.hpp
|
---|
3 | *
|
---|
4 | * Created on: Aug 25, 2010
|
---|
5 | * Author: heber
|
---|
6 | */
|
---|
7 |
|
---|
8 | // include config.h
|
---|
9 | #ifdef HAVE_CONFIG_H
|
---|
10 | #include <config.h>
|
---|
11 | #endif
|
---|
12 |
|
---|
13 | #include <boost/preprocessor/cat.hpp>
|
---|
14 | #include <boost/preprocessor/comparison/equal.hpp>
|
---|
15 | #include <boost/preprocessor/comparison/not_equal.hpp>
|
---|
16 | #include <boost/preprocessor/control/expr_if.hpp>
|
---|
17 | #include <boost/preprocessor/control/if.hpp>
|
---|
18 | #include <boost/preprocessor/debug/assert.hpp>
|
---|
19 | #include <boost/preprocessor/iteration/local.hpp>
|
---|
20 | #include <boost/preprocessor/list/adt.hpp>
|
---|
21 | #include <boost/preprocessor/punctuation/comma_if.hpp>
|
---|
22 | #include <boost/preprocessor/punctuation/paren.hpp>
|
---|
23 | #include <boost/preprocessor/repetition/repeat.hpp>
|
---|
24 | #include <boost/preprocessor/seq/elem.hpp>
|
---|
25 | #include <boost/preprocessor/seq/push_back.hpp>
|
---|
26 | #include <boost/preprocessor/seq/seq.hpp>
|
---|
27 | #include <boost/preprocessor/seq/size.hpp>
|
---|
28 | #include <boost/preprocessor/seq/transform.hpp>
|
---|
29 |
|
---|
30 | #include <iostream>
|
---|
31 | #include <typeinfo>
|
---|
32 |
|
---|
33 | #include "Actions/ActionSequence.hpp"
|
---|
34 | #include "Actions/ActionTraits.hpp"
|
---|
35 |
|
---|
36 | #include "Parameters/Parameter.hpp"
|
---|
37 |
|
---|
38 | // some derived names: if CATEGORY is not given, we don't prefix with it
|
---|
39 | #ifdef CATEGORY
|
---|
40 | #define ACTION BOOST_PP_CAT(CATEGORY, BOOST_PP_CAT(ACTIONNAME, Action))
|
---|
41 | #define COMMAND BOOST_PP_CAT(CATEGORY, ACTIONNAME)
|
---|
42 | #define PARAMS BOOST_PP_CAT(CATEGORY, BOOST_PP_CAT(ACTIONNAME, Parameters))
|
---|
43 | #else
|
---|
44 | #define ACTION BOOST_PP_CAT(ACTIONNAME, Action)
|
---|
45 | #define COMMAND ACTIONNAME
|
---|
46 | #define PARAMS BOOST_PP_CAT(ACTIONNAME, Parameters)
|
---|
47 | #endif
|
---|
48 | // check if no lists given
|
---|
49 | #ifndef paramtypes
|
---|
50 | #define MAXPARAMTYPES 0
|
---|
51 | #else
|
---|
52 | #define MAXPARAMTYPES BOOST_PP_SEQ_SIZE(paramtypes)
|
---|
53 | #endif
|
---|
54 | #ifndef paramdefaults
|
---|
55 | #define MAXPARAMDEFAULTS 0
|
---|
56 | // this is required for valid_print "else part"
|
---|
57 | #define sequencer(z,n,data) \
|
---|
58 | BOOST_PP_SEQ_PUSH_BACK( data, NOPARAM_DEFAULT)
|
---|
59 | #define paramdefaults BOOST_PP_REPEAT( MAXPARAMTYPES, sequencer, BOOST_PP_SEQ_NIL )
|
---|
60 | #else
|
---|
61 | #define MAXPARAMDEFAULTS BOOST_PP_SEQ_SIZE(paramdefaults)
|
---|
62 | #endif
|
---|
63 | #define PARAM_DEFAULT(x) \
|
---|
64 | (x, BOOST_PP_NIL)
|
---|
65 |
|
---|
66 | // check user has given name and category
|
---|
67 | #ifndef ACTIONNAME
|
---|
68 | ERROR: No "ACTIONNAME" defined in: __FILE__
|
---|
69 | #endif
|
---|
70 |
|
---|
71 | // calculate numbers and check whether all have same size
|
---|
72 | #ifdef paramtokens
|
---|
73 | BOOST_PP_ASSERT_MSG(BOOST_PP_EQUAL(MAXPARAMTYPES, BOOST_PP_SEQ_SIZE(paramtokens)),\
|
---|
74 | ERROR: There are not the same number of "paramtokens" and "paramtypes" in: __FILE__ \
|
---|
75 | )
|
---|
76 | #endif
|
---|
77 | #ifdef paramreferences
|
---|
78 | BOOST_PP_ASSERT_MSG(BOOST_PP_EQUAL(MAXPARAMTYPES, BOOST_PP_SEQ_SIZE(paramreferences)),\
|
---|
79 | ERROR: There are not the same number of "paramtokens" and "paramreferences" in: __FILE__ \
|
---|
80 | )
|
---|
81 | #endif
|
---|
82 | #ifdef paramdescriptions
|
---|
83 | BOOST_PP_ASSERT_MSG(BOOST_PP_EQUAL(MAXPARAMTYPES, BOOST_PP_SEQ_SIZE(paramdescriptions)),\
|
---|
84 | ERROR: There are not the same number of "paramtokens" and "paramdescriptions" in: __FILE__ \
|
---|
85 | )
|
---|
86 | #endif
|
---|
87 |
|
---|
88 | // check for mandatory defines
|
---|
89 | #ifndef DESCRIPTION
|
---|
90 | BOOST_PP_ASSERT_MSG(0, \
|
---|
91 | "ERROR: Description is mandatory for Actions, here for ACTION " \
|
---|
92 | )
|
---|
93 | #endif
|
---|
94 |
|
---|
95 | // check if paramdefaults is given, otherwise fill list with NOPARAM_DEFAULT
|
---|
96 | // this does not work: paramdefaults has to be completely defined before
|
---|
97 | // being used within option_print (used as an array there and not as
|
---|
98 | // some function call still to be expanded)
|
---|
99 | //#define paramdefaults (NOPARAM_DEFAULT)
|
---|
100 | //#define tempvalue(z,n,value)
|
---|
101 | // BOOST_PP_CAT(value,(NOPARAM_DEFAULT))
|
---|
102 | //BOOST_PP_REPEAT(tempvalue, MAXPARAMTYPES, paramdefaults)
|
---|
103 | //#undef tempvalue
|
---|
104 | //#else
|
---|
105 |
|
---|
106 | // if present, check if correct number of arguments
|
---|
107 | #ifdef paramdefaults
|
---|
108 | BOOST_PP_ASSERT_MSG(BOOST_PP_EQUAL(MAXPARAMTYPES, BOOST_PP_SEQ_SIZE(paramdefaults)),\
|
---|
109 | ERROR: There are not the same number of "paramtokens" and "paramdefaults" in: __FILE__ \
|
---|
110 | )
|
---|
111 | #endif
|
---|
112 |
|
---|
113 | // print a list of type ref followed by a separator, i.e. "Parameter<int> i;"
|
---|
114 | #define type_print(z,n,TYPELIST, VARLIST, separator) \
|
---|
115 | Parameter < \
|
---|
116 | BOOST_PP_SEQ_ELEM(n, TYPELIST) \
|
---|
117 | > \
|
---|
118 | BOOST_PP_SEQ_ELEM(n, VARLIST) \
|
---|
119 | separator
|
---|
120 |
|
---|
121 | // print a list of type ref followed, i.e. "int i, double position"
|
---|
122 | #define type_list(z,n,TYPELIST, VARLIST) \
|
---|
123 | BOOST_PP_COMMA_IF(n)\
|
---|
124 | BOOST_PP_SEQ_ELEM(n, TYPELIST) \
|
---|
125 | BOOST_PP_SEQ_ELEM(n, VARLIST)
|
---|
126 |
|
---|
127 | // prints Options.insert
|
---|
128 | #define option_print(z,n,unused, unused2) \
|
---|
129 | tester = Options. insert (\
|
---|
130 | std::pair< std::string, OptionTrait *> ( \
|
---|
131 | BOOST_PP_SEQ_ELEM(n, paramtokens), \
|
---|
132 | new OptionTrait(\
|
---|
133 | BOOST_PP_SEQ_ELEM(n, paramtokens), \
|
---|
134 | &typeid( BOOST_PP_SEQ_ELEM(n, paramtypes) ), \
|
---|
135 | BOOST_PP_SEQ_ELEM(n, paramdescriptions) \
|
---|
136 | BOOST_PP_COMMA_IF( BOOST_PP_NOT( BOOST_PP_LIST_IS_NIL( BOOST_PP_SEQ_ELEM(n, paramdefaults) ) ) ) \
|
---|
137 | BOOST_PP_EXPR_IF( \
|
---|
138 | BOOST_PP_NOT( BOOST_PP_LIST_IS_NIL( BOOST_PP_SEQ_ELEM(n, paramdefaults) ) ), \
|
---|
139 | toString BOOST_PP_LPAREN() \
|
---|
140 | BOOST_PP_LIST_FIRST( BOOST_PP_SEQ_ELEM(n, paramdefaults) )) \
|
---|
141 | BOOST_PP_RPAREN() \
|
---|
142 | )\
|
---|
143 | )\
|
---|
144 | ); \
|
---|
145 | ASSERT(tester.second, "ActionTrait<ACTION>::ActionTrait<ACTION>() option token present twice!");
|
---|
146 |
|
---|
147 | namespace MoleCuilder {
|
---|
148 |
|
---|
149 | void COMMAND(
|
---|
150 | #if defined paramtypes && defined paramreferences
|
---|
151 | #define BOOST_PP_LOCAL_MACRO(n) type_list(~, n, paramtypes, paramreferences)
|
---|
152 | #define BOOST_PP_LOCAL_LIMITS (0, MAXPARAMTYPES-1)
|
---|
153 | #include BOOST_PP_LOCAL_ITERATE()
|
---|
154 | #endif
|
---|
155 | );
|
---|
156 |
|
---|
157 | class ACTION;
|
---|
158 |
|
---|
159 | template <>
|
---|
160 | class ActionTraits<ACTION> : public ActionTrait {
|
---|
161 | public:
|
---|
162 | ActionTraits() :
|
---|
163 | #ifndef SHORTFORM
|
---|
164 | ActionTrait(OptionTrait(TOKEN, &typeid(void), DESCRIPTION, std::string()))
|
---|
165 | #else
|
---|
166 | ActionTrait(OptionTrait(TOKEN, &typeid(void), DESCRIPTION, std::string(), SHORTFORM))
|
---|
167 | #endif /* SHORTFORM */
|
---|
168 | {
|
---|
169 | // initialize remainder of action info
|
---|
170 | #ifdef MENUNAME
|
---|
171 | MenuTitle = MENUNAME;
|
---|
172 | #endif
|
---|
173 | #ifdef MENUPOSITION
|
---|
174 | MenuPosition = MENUPOSITION;
|
---|
175 | #endif
|
---|
176 |
|
---|
177 | // initialize action's options
|
---|
178 | std::pair< ActionTrait::options_iterator, bool > tester;
|
---|
179 | #ifdef paramtokens
|
---|
180 | #define BOOST_PP_LOCAL_MACRO(n) option_print(~, n, ~, )
|
---|
181 | #define BOOST_PP_LOCAL_LIMITS (0, MAXPARAMTYPES-1)
|
---|
182 | #include BOOST_PP_LOCAL_ITERATE()
|
---|
183 | #endif
|
---|
184 | // associate action's short form also with option
|
---|
185 | #ifdef SHORTFORM
|
---|
186 | if (Options.find(TOKEN) != Options.end())
|
---|
187 | Options[TOKEN]->setShortForm(std::string(SHORTFORM));
|
---|
188 | #endif /* SHORTFORM */
|
---|
189 | //std::cout << "ActionTraits<" << BOOST_PP_STRINGIZE(ACTION) << ">::ActionTraits() on instance " << this << " with " << getName() << ", type " << getTypeName() << " and description " << getDescription() << std::endl;
|
---|
190 | }
|
---|
191 |
|
---|
192 | virtual ~ActionTraits() {
|
---|
193 | //std::cout << "ActionTraits<" << BOOST_PP_STRINGIZE(ACTION) << ">::~ActionTraits() on instance " << this << " with name " << getName() << " called." << std::endl;
|
---|
194 | }
|
---|
195 | };
|
---|
196 |
|
---|
197 | class ACTION : public MakroAction {
|
---|
198 | #if defined paramtypes && defined paramreferences
|
---|
199 | friend void COMMAND(
|
---|
200 | #define BOOST_PP_LOCAL_MACRO(n) type_list(~, n, paramtypes, paramreferences)
|
---|
201 | #define BOOST_PP_LOCAL_LIMITS (0, MAXPARAMTYPES-1)
|
---|
202 | #include BOOST_PP_LOCAL_ITERATE()
|
---|
203 | );
|
---|
204 | # else
|
---|
205 | void COMMAND();
|
---|
206 | #endif
|
---|
207 |
|
---|
208 | public:
|
---|
209 | ACTION();
|
---|
210 | virtual ~ACTION();
|
---|
211 |
|
---|
212 | // must be called after all primitive actions are present
|
---|
213 | void prepare(ActionRegistry &AR);
|
---|
214 | // must be called before alle primitive actions are removed
|
---|
215 | void unprepare(ActionRegistry &AR);
|
---|
216 |
|
---|
217 | bool canUndo();
|
---|
218 | bool shouldUndo();
|
---|
219 |
|
---|
220 | struct PARAMS : ActionParameters {
|
---|
221 | //!> constructor for class PARAMS, setting valid ranges
|
---|
222 | PARAMS();
|
---|
223 | //!> copy constructor for class PARAMS, setting valid ranges
|
---|
224 | PARAMS(const PARAMS &p);
|
---|
225 | #if defined paramtypes && defined paramreferences
|
---|
226 | #define BOOST_PP_LOCAL_MACRO(n) type_print(~, n, paramtypes, paramreferences, ;)
|
---|
227 | #define BOOST_PP_LOCAL_LIMITS (0, MAXPARAMTYPES-1)
|
---|
228 | #include BOOST_PP_LOCAL_ITERATE()
|
---|
229 | #endif
|
---|
230 | } params;
|
---|
231 |
|
---|
232 | protected:
|
---|
233 | virtual Dialog * fillDialog(Dialog*);
|
---|
234 |
|
---|
235 | void startTimer() const;
|
---|
236 | void endTimer() const;
|
---|
237 |
|
---|
238 | private:
|
---|
239 | //!> flag to check whether actions have been prepared
|
---|
240 | static bool isPrepared;
|
---|
241 | //!> sequence of actions for this macro action
|
---|
242 | static ActionSequence actions;
|
---|
243 |
|
---|
244 | private:
|
---|
245 | virtual Action::state_ptr performCall();
|
---|
246 | virtual Action::state_ptr performUndo(Action::state_ptr);
|
---|
247 | virtual Action::state_ptr performRedo(Action::state_ptr);
|
---|
248 | };
|
---|
249 |
|
---|
250 | }
|
---|
251 |
|
---|
252 | #undef paramvalids
|
---|
253 | #undef paramtypes
|
---|
254 | #undef paramtokens
|
---|
255 | #undef paramreferences
|
---|
256 | #undef paramdescriptions
|
---|
257 | #undef paramdefaults
|
---|
258 | #undef MAXPARAMTYPES
|
---|
259 | #undef MAXPARAMDEFAULTS
|
---|
260 | #undef statetypes
|
---|
261 | #undef statereferences
|
---|
262 | #undef MAXSTATETYPES
|
---|
263 | #undef PARAM_DEFAULT
|
---|
264 |
|
---|
265 | #undef option_print
|
---|
266 | #undef sequencer
|
---|
267 | #undef type_print
|
---|
268 | #undef type_list
|
---|
269 |
|
---|
270 | #undef ACTION
|
---|
271 | #undef COMMAND
|
---|
272 | #undef COMMANDFULL
|
---|
273 | #undef PARAMS
|
---|
274 |
|
---|
275 | #undef ACTIONNAME
|
---|
276 | #undef CATEGORY
|
---|
277 | #undef MENUNAME
|
---|
278 | #undef MENUPOSITION
|
---|
279 | #undef TOKEN
|
---|
280 |
|
---|
281 | #undef DESCRIPTION
|
---|
282 | #undef SHORTFORM
|
---|