| [97ebf8] | 1 | /* | 
|---|
|  | 2 | * MapOfActions.hpp | 
|---|
|  | 3 | * | 
|---|
|  | 4 | *  Created on: 10.05.2010 | 
|---|
|  | 5 | *      Author: heber | 
|---|
|  | 6 | */ | 
|---|
|  | 7 |  | 
|---|
|  | 8 | #ifndef MAPOFACTIONS_HPP_ | 
|---|
|  | 9 | #define MAPOFACTIONS_HPP_ | 
|---|
|  | 10 |  | 
|---|
|  | 11 | #include <boost/program_options.hpp> | 
|---|
| [0bb05a] | 12 | #include <boost/lexical_cast.hpp> | 
|---|
| [326bbe] | 13 |  | 
|---|
| [97ebf8] | 14 | #include <map> | 
|---|
|  | 15 | #include <set> | 
|---|
| [d02e07] | 16 | #include <vector> | 
|---|
| [ab9a27] | 17 | #include <typeinfo> | 
|---|
| [97ebf8] | 18 |  | 
|---|
| [ab9a27] | 19 | #include "Exceptions/IllegalTypeException.hpp" | 
|---|
| [4e145c] | 20 | #include "Exceptions/MissingValueException.hpp" | 
|---|
|  | 21 | #include "Helpers/Assert.hpp" | 
|---|
| [97ebf8] | 22 | #include "Patterns/Singleton.hpp" | 
|---|
|  | 23 |  | 
|---|
| [4e145c] | 24 | class MapOfActionsTest; | 
|---|
|  | 25 |  | 
|---|
| [d02e07] | 26 | class Box; | 
|---|
|  | 27 | class atom; | 
|---|
|  | 28 | class element; | 
|---|
|  | 29 | class molecule; | 
|---|
|  | 30 | class Vector; | 
|---|
|  | 31 |  | 
|---|
| [97ebf8] | 32 | namespace po = boost::program_options; | 
|---|
|  | 33 |  | 
|---|
| [ab9a27] | 34 | using boost::lexical_cast; | 
|---|
|  | 35 |  | 
|---|
| [326bbe] | 36 | /** Central class for adding functionality to the code. | 
|---|
|  | 37 | * | 
|---|
|  | 38 | * In Molecuilder everything that can be done - such as adding atoms, | 
|---|
|  | 39 | * translating molecules, saving bind information - is an Action. | 
|---|
|  | 40 | * | 
|---|
|  | 41 | * In order to reference Action's with what the user sees, this class is the | 
|---|
|  | 42 | * mediator. | 
|---|
|  | 43 | * | 
|---|
|  | 44 | * An Action is described to the user by: | 
|---|
|  | 45 | * -# a name (this is the most important information) | 
|---|
|  | 46 | * -# a description | 
|---|
|  | 47 | * -# a shortform (single letter for use on the command line) | 
|---|
|  | 48 | * -# a text menu it resides in | 
|---|
|  | 49 | * -# the type of its argument | 
|---|
|  | 50 | * -# the command line category | 
|---|
|  | 51 | * | 
|---|
|  | 52 | * The Action::NAME is the most important information because every Action | 
|---|
|  | 53 | * registers itself automatically with the ActionRegistry and can be retrieved | 
|---|
|  | 54 | * therefrom and from this MapOfActions simply by knowing its name alone. | 
|---|
|  | 55 | * | 
|---|
|  | 56 | * In the constructor of MapOfActions all this is set. | 
|---|
|  | 57 | * | 
|---|
|  | 58 | * Note that Action will require input from the user. This is done via class | 
|---|
|  | 59 | * Query. | 
|---|
|  | 60 | * | 
|---|
|  | 61 | * And note also that MapOfActions actually contains more than just all | 
|---|
|  | 62 | * Actions: There are a number of names that actually are just arguments to | 
|---|
|  | 63 | * actions (e.g. "output-file"). | 
|---|
|  | 64 | * | 
|---|
|  | 65 | * <h1> Howto add an Action</h1> | 
|---|
|  | 66 | * | 
|---|
|  | 67 | * Let us assume your new action (class) is called SuperDuperAction, consisting | 
|---|
|  | 68 | * of two files SuperDuperAction.cpp and SuperDuperAction.hpp. | 
|---|
|  | 69 | * | 
|---|
|  | 70 | * Furthermore, let's say you Action needs two values: a double value as a | 
|---|
|  | 71 | * upper threshold and a string which is the name of the output file. | 
|---|
|  | 72 | * | 
|---|
|  | 73 | * <h2> Command Line preliminaries </h2> | 
|---|
|  | 74 | * | 
|---|
|  | 75 | * You have to decide whether (for the command line) it makes sense to have an | 
|---|
|  | 76 | * extra argument requesting the arguments, or one should be the argument of | 
|---|
|  | 77 | * your action. I.e. your action name is "super-duper", then the use may | 
|---|
|  | 78 | * call your action like this: | 
|---|
|  | 79 | * | 
|---|
|  | 80 | * ./molecuilder --super-duper 4 --output-file test.dat | 
|---|
|  | 81 | * | 
|---|
|  | 82 | * Here, we have picked the threshold as the value for your action and the | 
|---|
|  | 83 | * name of the output file is given by an additional argument. Of course, | 
|---|
|  | 84 | * it can be the other way round or by two arguments such as here: | 
|---|
|  | 85 | * | 
|---|
|  | 86 | * ./molecuilder --super-duper --threshold 4 --output-file test.dat | 
|---|
|  | 87 | * | 
|---|
|  | 88 | * It depends on what possible arguments are already there (don't make up | 
|---|
|  | 89 | * new ones if present ones actually make sense for your action) and which | 
|---|
|  | 90 | * argument is more natural or closer to what your action does. | 
|---|
|  | 91 | * | 
|---|
|  | 92 | * <h2> Menu preliminaries </h2> | 
|---|
|  | 93 | * | 
|---|
|  | 94 | * Whatever you decide, your action will need some Query dialogs to request | 
|---|
|  | 95 | * the necessary information from the user, either via a command line | 
|---|
|  | 96 | * argument (--output-file) via a text dialog (referenced by "output-file") | 
|---|
|  | 97 | * or via a graphical dialog (same reference). And therein, the names | 
|---|
|  | 98 | * of the arguments have to re-appear. | 
|---|
|  | 99 | * | 
|---|
|  | 100 | * Then, the following steps have to be done to incorporate your Action: | 
|---|
|  | 101 | * -# create a unique name for your action (no capital letters) to reference | 
|---|
|  | 102 | *    it, this name has to appear in the file SuperDuperAction.cpp, e.g. | 
|---|
|  | 103 | *    "super-duper" | 
|---|
|  | 104 | * -# pick names the other required arguments, best if they are already | 
|---|
|  | 105 | *    present in the MapOfActions. They have to appear in Query's in the | 
|---|
|  | 106 | *    code of your Action. | 
|---|
|  | 107 | * -# With this name create entries in the following maps for the action | 
|---|
|  | 108 | *    name and for each names of a desired addtional argument if not present: | 
|---|
|  | 109 | *   -# DescriptionMap, a catchy description of what your action does | 
|---|
|  | 110 | *   -# TypeMap, see MapOfActions::OptionTypes for possible types of the single | 
|---|
|  | 111 | *      argument it takes. | 
|---|
|  | 112 | *   -# MenuContainsActionMap, in which menu should your action appear | 
|---|
|  | 113 | *   -# ShortFormMap (optional), single letter for command line call | 
|---|
|  | 114 | *   -# DefaultValueMap (optional), the default value (always a string) | 
|---|
|  | 115 | * -# add to one of the command line sets by the following categories | 
|---|
|  | 116 | *   -# generic - generic options (i.e. not one of the others) | 
|---|
|  | 117 | *   -# config - action/argument only considers internal bevahior, user | 
|---|
|  | 118 | *      does not have to see it while still having full functionality | 
|---|
|  | 119 | *   -# hidden - this should be hidden from the user | 
|---|
|  | 120 | *   -# visible - this should be visible to the user | 
|---|
|  | 121 | *   -# inputfile - this should only be parsed from an input file, not | 
|---|
|  | 122 | *      from command line | 
|---|
|  | 123 | * -# add to a menu, i.e. make an entry in MenuContainsActionMap. | 
|---|
|  | 124 | * -# add header file SuperDuperAction.hpp to MapOfActions.cpp and instantiate | 
|---|
|  | 125 | *    your action in populateMenu() (mind the sorting: 1. menu, | 
|---|
|  | 126 | *    2. alphabetical) | 
|---|
|  | 127 | * | 
|---|
|  | 128 | *  And that's. | 
|---|
|  | 129 | * | 
|---|
|  | 130 | *  Now, your action can be called from the command line, within the text | 
|---|
|  | 131 | *  menu and the graphical user interface. | 
|---|
|  | 132 | * | 
|---|
|  | 133 | */ | 
|---|
| [97ebf8] | 134 | class MapOfActions : public Singleton<MapOfActions> { | 
|---|
|  | 135 | friend class Singleton<MapOfActions>; | 
|---|
|  | 136 | friend class MapOfActionsTest; | 
|---|
|  | 137 | public: | 
|---|
| [0b0a20] | 138 | enum OptionTypes { None, Boolean, Integer, ListOfIntegers, Double, ListOfDoubles, String, ListOfStrings, Vector, ListOfVectors, Box, Molecule, ListOfMolecules, Atom, ListOfAtoms, Element, ListOfElements }; | 
|---|
| [97ebf8] | 139 |  | 
|---|
|  | 140 | // getter for the action descriptions and short forms | 
|---|
| [326bbe] | 141 | std::string getDescription(std::string actionname); | 
|---|
|  | 142 | std::string getKeyAndShortForm(std::string actionname); | 
|---|
|  | 143 | std::string getShortForm(std::string actionname); | 
|---|
|  | 144 | std::map <std::string, std::string> getShortFormToActionMap(); | 
|---|
| [97ebf8] | 145 |  | 
|---|
|  | 146 | void AddOptionsToParser(); | 
|---|
|  | 147 |  | 
|---|
|  | 148 | // check presence and getter for action type | 
|---|
| [326bbe] | 149 | bool hasValue(std::string actionname); | 
|---|
|  | 150 | bool isShortFormPresent(std::string shortform); | 
|---|
| [ab9a27] | 151 | std::string  getValueType(std::string actionname); | 
|---|
| [326bbe] | 152 |  | 
|---|
|  | 153 | std::set<std::string> generic; | 
|---|
|  | 154 | std::set<std::string> config; | 
|---|
|  | 155 | std::set<std::string> hidden; | 
|---|
|  | 156 | std::set<std::string> visible; | 
|---|
|  | 157 | std::set<std::string> inputfile; | 
|---|
|  | 158 |  | 
|---|
|  | 159 | std::multimap <std::string, std::string> MenuContainsActionMap; | 
|---|
| [b2531f] | 160 | std::map <std::string, std::pair<std::string,std::string> > MenuDescription; | 
|---|
| [97ebf8] | 161 |  | 
|---|
| [326bbe] | 162 | // instantiates and puts all known actions into the ActionRegistry | 
|---|
|  | 163 | void populateActions(); | 
|---|
| [97ebf8] | 164 |  | 
|---|
| [d02e07] | 165 | void queryCurrentValue(const char * name, class atom * &_T); | 
|---|
|  | 166 | void queryCurrentValue(const char * name, class element * &_T); | 
|---|
|  | 167 | void queryCurrentValue(const char * name, class molecule * &_T); | 
|---|
|  | 168 | void queryCurrentValue(const char * name, class Box &_T); | 
|---|
|  | 169 | void queryCurrentValue(const char * name, class Vector &_T); | 
|---|
| [0b0a20] | 170 | void queryCurrentValue(const char * name, std::vector<atom *>&_T); | 
|---|
| [d02e07] | 171 | void queryCurrentValue(const char * name, std::vector<element *>&_T); | 
|---|
| [0b0a20] | 172 | void queryCurrentValue(const char * name, std::vector<molecule *>&_T); | 
|---|
| [ab9a27] | 173 | template<typename T> void queryCurrentValue(const char * name, T &_T) | 
|---|
|  | 174 | { | 
|---|
| [4e145c] | 175 | if (typeid( T ) == *TypeMap[name]) { // constructor of type_info is private, hence can only store by ref or ptr | 
|---|
|  | 176 | if (CurrentValue.find(name) == CurrentValue.end()) | 
|---|
|  | 177 | throw MissingValueException(__FILE__, __LINE__); | 
|---|
| [ab9a27] | 178 | _T = lexical_cast<T>(CurrentValue[name].c_str()); | 
|---|
| [4e145c] | 179 | CurrentValue.erase(name); | 
|---|
|  | 180 | } else | 
|---|
| [ab9a27] | 181 | throw IllegalTypeException(__FILE__,__LINE__); | 
|---|
|  | 182 | } | 
|---|
| [0b0a20] | 183 | template<typename T> void queryCurrentValue(const char * name, std::vector<T> &_T) | 
|---|
|  | 184 | { | 
|---|
|  | 185 | T temp; | 
|---|
|  | 186 | if (typeid( std::vector<T> ) == *TypeMap[name]) { // constructor of type_info is private, hence can only store by ref or ptr | 
|---|
|  | 187 | if (CurrentValue.find(name) == CurrentValue.end()) | 
|---|
|  | 188 | throw MissingValueException(__FILE__, __LINE__); | 
|---|
|  | 189 | std::istringstream stream(CurrentValue[name]); | 
|---|
|  | 190 | CurrentValue.erase(name); | 
|---|
|  | 191 | while (!stream.fail()) { | 
|---|
|  | 192 | stream >> temp >> std::ws; | 
|---|
|  | 193 | _T.push_back(temp); | 
|---|
|  | 194 | } | 
|---|
|  | 195 | } else | 
|---|
|  | 196 | throw IllegalTypeException(__FILE__,__LINE__); | 
|---|
|  | 197 | } | 
|---|
| [ab9a27] | 198 |  | 
|---|
| [d02e07] | 199 | void setCurrentValue(const char * name, class atom * &_T); | 
|---|
|  | 200 | void setCurrentValue(const char * name, class element * &_T); | 
|---|
|  | 201 | void setCurrentValue(const char * name, class molecule * &_T); | 
|---|
|  | 202 | void setCurrentValue(const char * name, class Box &_T); | 
|---|
|  | 203 | void setCurrentValue(const char * name, class Vector &_T); | 
|---|
| [0b0a20] | 204 | void setCurrentValue(const char * name, std::vector<atom *>&_T); | 
|---|
| [d02e07] | 205 | void setCurrentValue(const char * name, std::vector<element *>&_T); | 
|---|
| [0b0a20] | 206 | void setCurrentValue(const char * name, std::vector<molecule *>&_T); | 
|---|
| [ab9a27] | 207 | template<class T> void setCurrentValue(const char * name, T &_T) | 
|---|
|  | 208 | { | 
|---|
| [7cd469] | 209 | std::ostringstream stream; | 
|---|
|  | 210 | if (typeid( T ) == *TypeMap[name]) {  // constructor of type_info is private, hence can only store by ref or ptr | 
|---|
|  | 211 | stream << _T; | 
|---|
|  | 212 | CurrentValue[name] = stream.str(); | 
|---|
|  | 213 | } else | 
|---|
| [ab9a27] | 214 | throw IllegalTypeException(__FILE__,__LINE__); | 
|---|
|  | 215 | } | 
|---|
| [0b0a20] | 216 | template<class T> void setCurrentValue(const char * name, std::vector<T> &_T) | 
|---|
|  | 217 | { | 
|---|
|  | 218 | std::ostringstream stream; | 
|---|
|  | 219 | if (typeid( std::vector<T> ) == *TypeMap[name]) {  // constructor of type_info is private, hence can only store by ref or ptr | 
|---|
|  | 220 | std::ostringstream stream; | 
|---|
| [85f35e] | 221 | for (typename std::vector<T>::iterator iter = _T.begin(); iter != _T.end(); ++iter) { | 
|---|
| [0b0a20] | 222 | stream << (*iter) << " "; | 
|---|
|  | 223 | } | 
|---|
|  | 224 | CurrentValue[name] = stream.str(); | 
|---|
|  | 225 | } else | 
|---|
|  | 226 | throw IllegalTypeException(__FILE__,__LINE__); | 
|---|
|  | 227 | } | 
|---|
| [ab9a27] | 228 |  | 
|---|
|  | 229 |  | 
|---|
| [97ebf8] | 230 | private: | 
|---|
|  | 231 | // private constructor and destructor | 
|---|
|  | 232 | MapOfActions(); | 
|---|
|  | 233 | virtual ~MapOfActions(); | 
|---|
|  | 234 |  | 
|---|
|  | 235 | // lookup list from our configs to the ones of CommandLineParser | 
|---|
| [326bbe] | 236 | std::map< std::set<std::string> *, po::options_description *> CmdParserLookup; | 
|---|
| [97ebf8] | 237 |  | 
|---|
|  | 238 | // map of the action names and their description | 
|---|
| [ab9a27] | 239 | std::map<std::string, std::string> CurrentValue; | 
|---|
| [326bbe] | 240 | std::map<std::string, std::string> DescriptionMap; | 
|---|
|  | 241 | std::map<std::string, std::string> ShortFormMap; | 
|---|
| [ab9a27] | 242 | std::map<std::string, const std::type_info * > TypeMap; | 
|---|
|  | 243 | std::map<const std::type_info *, enum OptionTypes > TypeEnumMap; | 
|---|
| [97ebf8] | 244 | }; | 
|---|
|  | 245 |  | 
|---|
|  | 246 |  | 
|---|
|  | 247 | #endif /* MAPOFACTIONS_HPP_ */ | 
|---|