Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/core/CommandExecutor.cc @ 1420

Last change on this file since 1420 was 1349, checked in by rgrieder, 17 years ago
  • merged input branch back to trunk
File size: 62.4 KB
RevLine 
[947]1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
[1056]3 *                    > www.orxonox.net <
[947]4 *
5 *
6 *   License notice:
7 *
8 *   This program is free software; you can redistribute it and/or
9 *   modify it under the terms of the GNU General Public License
10 *   as published by the Free Software Foundation; either version 2
11 *   of the License, or (at your option) any later version.
12 *
13 *   This program is distributed in the hope that it will be useful,
14 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 *   GNU General Public License for more details.
17 *
18 *   You should have received a copy of the GNU General Public License
19 *   along with this program; if not, write to the Free Software
20 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
21 *
22 *   Author:
23 *      Fabian 'x3n' Landau
24 *   Co-authors:
25 *      ...
26 *
27 */
28
29#include "CommandExecutor.h"
30#include "ConsoleCommand.h"
31#include "util/String.h"
[1030]32#include "util/Convert.h"
[1349]33#include "util/SubString.h"
[947]34#include "Identifier.h"
35#include "Language.h"
36#include "Debug.h"
37#include "Executor.h"
38#include "ConfigValueContainer.h"
[1214]39#include "TclBind.h"
[947]40
41#define COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE "set"
42#define COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY "tset"
43#define COMMAND_EXECUTOR_KEYWORD_SET_KEYBIND "bind"
44
45namespace orxonox
46{
47    ConsoleCommandShortcutGeneric(keyword1, createExecutor((FunctorStatic*)0, "set", AccessLevel::User));
48    ConsoleCommandShortcutGeneric(keyword2, createExecutor((FunctorStatic*)0, "tset", AccessLevel::User));
49    ConsoleCommandShortcutGeneric(keyword3, createExecutor((FunctorStatic*)0, "bind", AccessLevel::User));
50
[993]51    ConsoleCommandShortcutExtern(exec, AccessLevel::None);
[994]52    ConsoleCommandShortcutExtern(echo, AccessLevel::None);
[953]53
[994]54    ConsoleCommandShortcutExtern(read, AccessLevel::None);
55    ConsoleCommandShortcutExtern(append, AccessLevel::None);
56    ConsoleCommandShortcutExtern(write, AccessLevel::None);
57
[993]58    void exec(const std::string& filename)
59    {
60        static std::set<std::string> executingFiles;
61
62        std::set<std::string>::const_iterator it = executingFiles.find(filename);
63        if (it != executingFiles.end())
64        {
65            COUT(1) << "Error: Recurring exec command in \"" << filename << "\". Stopped execution." << std::endl;
66            return;
67        }
68
69        // Open the file
70        std::ifstream file;
71        file.open(filename.c_str(), std::fstream::in);
72
73        if (!file.is_open())
74        {
75            COUT(1) << "Error: Couldn't execute file \"" << filename << "\"." << std::endl;
76            return;
77        }
78
79        executingFiles.insert(filename);
80
81        // Iterate through the file and put the lines into the CommandExecutor
82        char line[1024];
83        while (file.good() && !file.eof())
84        {
85            file.getline(line, 1024);
86            CommandExecutor::execute(line);
87        }
88
89        executingFiles.erase(filename);
[994]90        file.close();
[993]91    }
92
[994]93    std::string echo(const std::string& text)
94    {
95        return text;
96    }
[993]97
[994]98    void write(const std::string& filename, const std::string& text)
99    {
100        std::ofstream file;
101        file.open(filename.c_str(), std::fstream::out);
102
103        if (!file.is_open())
104        {
105            COUT(1) << "Error: Couldn't write to file \"" << filename << "\"." << std::endl;
106            return;
107        }
108
109        file << text << std::endl;
110        file.close();
111    }
112
113    void append(const std::string& filename, const std::string& text)
114    {
115        std::ofstream file;
116        file.open(filename.c_str(), std::fstream::app);
117
118        if (!file.is_open())
119        {
120            COUT(1) << "Error: Couldn't append to file \"" << filename << "\"." << std::endl;
121            return;
122        }
123
124        file << text << std::endl;
125        file.close();
126    }
127
128    std::string read(const std::string& filename)
129    {
130        std::ifstream file;
131        file.open(filename.c_str(), std::fstream::in);
132
133        if (!file.is_open())
134        {
135            COUT(1) << "Error: Couldn't read from file \"" << filename << "\"." << std::endl;
136            return "";
137        }
138
139        std::string output = "";
140        char line[1024];
141        while (file.good() && !file.eof())
142        {
143            file.getline(line, 1024);
144            output += line;
145            output += "\n";
146        }
147
148        file.close();
149
150        return output;
151    }
152
153
[953]154    ///////////////////////
155    // CommandEvaluation //
156    ///////////////////////
[955]157    CommandEvaluation::CommandEvaluation()
158    {
159        this->processedCommand_ = "";
160        this->additionalParameter_ = "";
161
162        this->functionclass_ = 0;
163        this->configvalueclass_ = 0;
164        this->shortcut_ = 0;
165        this->function_ = 0;
166        this->configvalue_ = 0;
167        this->key_ = 0;
168
169        this->errorMessage_ = "";
170        this->state_ = CS_Uninitialized;
[967]171
172        this->bEvaluatedParams_ = false;
173        this->evaluatedExecutor_ = 0;
[955]174    }
175
[1349]176    KeybindMode::Enum CommandEvaluation::getKeybindMode()
[953]177    {
178        if (this->state_ == CS_Shortcut_Params || this->state_ == CS_Shortcut_Finished)
179        {
180//            if (this->shortcut_ != 0)
181//                return this->shortcut_->getKeybindMode();
182        }
183        else if (this->state_ == CS_Function_Params || this->state_ == CS_Function_Finished)
184        {
185//            if (this->function_ != 0)
186//                return this->function_->getKeybindMode();
187        }
188        else if (this->state_ == CS_ConfigValueType || this->state_ == CS_ConfigValueFinished)
189        {
190//            return KeybindMode::onPress;
191        }
192        else if (this->state_ == CS_KeybindCommand || this->state_ == CS_KeybindFinished)
193        {
194//            return KeybindMode::onPress;
195        }
196        else
197        {
198//            return KeybindMode::onPress;
199        }
[1059]200        // FIXME: Had to insert a return statement
[1349]201        return (KeybindMode::Enum)0;
[953]202    }
203
204    bool CommandEvaluation::isValid() const
205    {
206        if (this->state_ == CS_Shortcut_Params || this->state_ == CS_Shortcut_Finished)
207        {
[972]208            return this->shortcut_;
[953]209        }
210        else if (this->state_ == CS_Function_Params || this->state_ == CS_Function_Finished)
211        {
[972]212            return (this->functionclass_ && this->function_);
[953]213        }
214        else if (this->state_ == CS_ConfigValueType || this->state_ == CS_ConfigValueFinished)
215        {
[972]216            return (this->configvalueclass_ && this->configvalue_);
[953]217        }
218        else if (this->state_ == CS_KeybindCommand || this->state_ == CS_KeybindFinished)
219        {
[972]220            return this->key_;
[953]221        }
222        else
223        {
224            return false;
225        }
226    }
227
[967]228    void CommandEvaluation::evaluateParams()
229    {
230        this->bEvaluatedParams_ = false;
231        this->evaluatedExecutor_ = 0;
[953]232
[967]233        for (unsigned int i = 0; i < MAX_FUNCTOR_ARGUMENTS; i++)
234            this->param_[i] = MT_null;
235
236        if (this->state_ == CS_Shortcut_Params || this->state_ == CS_Shortcut_Finished)
237        {
[972]238            if (this->shortcut_)
[967]239            {
[1214]240                if (this->tokens_.size() <= 1)
[967]241                {
[1214]242                    if (this->shortcut_->evaluate(this->getAdditionalParameter(), this->param_, " "))
243                    {
244                        this->bEvaluatedParams_ = true;
245                        this->evaluatedExecutor_ = this->shortcut_;
246                    }
[967]247                }
[1214]248                else if (this->tokens_.size() > 1)
249                {
250                    if (this->shortcut_->evaluate(this->tokens_.subSet(1).join() + this->getAdditionalParameter(), this->param_, " "))
251                    {
252                        this->bEvaluatedParams_ = true;
253                        this->evaluatedExecutor_ = this->shortcut_;
254                    }
255                }
[967]256            }
257        }
258        else if (this->state_ == CS_Function_Params || this->state_ == CS_Function_Finished)
259        {
[972]260            if (this->function_)
[967]261            {
[1214]262                if (this->tokens_.size() <= 2)
[967]263                {
[1214]264                    if (this->function_->evaluate(this->getAdditionalParameter(), this->param_, " "))
265                    {
266                        this->bEvaluatedParams_ = true;
267                        this->evaluatedExecutor_ = this->function_;
268                    }
[967]269                }
[1214]270                else if (this->tokens_.size() > 2)
271                {
272                    if (this->function_->evaluate(this->tokens_.subSet(2).join() + this->getAdditionalParameter(), this->param_, " "))
273                    {
274                        this->bEvaluatedParams_ = true;
275                        this->evaluatedExecutor_ = this->function_;
276                    }
277                }
[967]278            }
279        }
280    }
281
282    void CommandEvaluation::setEvaluatedParameter(unsigned int index, MultiTypeMath param)
283    {
284        if (index >= 0 && index < MAX_FUNCTOR_ARGUMENTS)
285            this->param_[index] = param;
286    }
287
288    MultiTypeMath CommandEvaluation::getEvaluatedParameter(unsigned int index) const
289    {
290        if (index >= 0 && index < MAX_FUNCTOR_ARGUMENTS)
291            return this->param_[index];
292
293        return MT_null;
294    }
295
[1214]296    bool CommandEvaluation::hasReturnvalue() const
297    {
298        if (this->state_ == CS_Shortcut_Params || this->state_ == CS_Shortcut_Finished)
299        {
300            if (this->shortcut_)
301                return this->shortcut_->hasReturnvalue();
302        }
303        else if (this->state_ == CS_Function_Params || this->state_ == CS_Function_Finished)
304        {
305            if (this->function_)
306                return this->function_->hasReturnvalue();
307        }
308
309        return MT_null;
310    }
311
[994]312    MultiTypeMath CommandEvaluation::getReturnvalue() const
313    {
314        if (this->state_ == CS_Shortcut_Params || this->state_ == CS_Shortcut_Finished)
315        {
316            if (this->shortcut_)
317                return this->shortcut_->getReturnvalue();
318        }
319        else if (this->state_ == CS_Function_Params || this->state_ == CS_Function_Finished)
320        {
321            if (this->function_)
322                return this->function_->getReturnvalue();
323        }
[967]324
[994]325        return MT_null;
326    }
327
328
[953]329    /////////////////////
330    // CommandExecutor //
331    /////////////////////
[949]332    CommandExecutor& CommandExecutor::getInstance()
333    {
334        static CommandExecutor instance;
335        return instance;
336    }
[947]337
[952]338    CommandEvaluation& CommandExecutor::getEvaluation()
339    {
340        return CommandExecutor::getInstance().evaluation_;
341    }
342
[1214]343    const CommandEvaluation& CommandExecutor::getLastEvaluation()
344    {
345        return CommandExecutor::getInstance().evaluation_;
346    }
347
[1001]348    Executor& CommandExecutor::addConsoleCommandShortcut(ExecutorStatic* executor)
[947]349    {
[951]350        CommandExecutor::getInstance().consoleCommandShortcuts_[executor->getName()] = executor;
351        CommandExecutor::getInstance().consoleCommandShortcuts_LC_[getLowercase(executor->getName())] = executor;
[1001]352        return (*executor);
[947]353    }
354
355    /**
356        @brief Returns the executor of a console command shortcut with given name.
357        @brief name The name of the requested console command shortcut
358        @return The executor of the requested console command shortcut
359    */
360    ExecutorStatic* CommandExecutor::getConsoleCommandShortcut(const std::string& name)
361    {
[951]362        std::map<std::string, ExecutorStatic*>::const_iterator it = CommandExecutor::getInstance().consoleCommandShortcuts_.find(name);
363        if (it != CommandExecutor::getInstance().consoleCommandShortcuts_.end())
[947]364            return (*it).second;
365        else
366            return 0;
367    }
368
369    /**
370        @brief Returns the executor of a console command shortcut with given name in lowercase.
371        @brief name The name of the requested console command shortcut in lowercase
372        @return The executor of the requested console command shortcut
373    */
374    ExecutorStatic* CommandExecutor::getLowercaseConsoleCommandShortcut(const std::string& name)
375    {
[951]376        std::map<std::string, ExecutorStatic*>::const_iterator it = CommandExecutor::getInstance().consoleCommandShortcuts_LC_.find(name);
377        if (it != CommandExecutor::getInstance().consoleCommandShortcuts_LC_.end())
[947]378            return (*it).second;
379        else
380            return 0;
381    }
382
[1214]383    bool CommandExecutor::execute(const std::string& command, bool useTcl)
[947]384    {
[1214]385        if (useTcl)
[994]386        {
[1214]387            return TclBind::eval(command);
[994]388        }
[1214]389        else
[994]390        {
[1214]391            if ((CommandExecutor::getEvaluation().processedCommand_ != command) || (CommandExecutor::getEvaluation().state_ == CS_Uninitialized))
392                CommandExecutor::parse(command);
[994]393
[1214]394            return CommandExecutor::execute(CommandExecutor::getEvaluation());
[994]395        }
[952]396    }
[948]397
[952]398
399    bool CommandExecutor::execute(const CommandEvaluation& evaluation)
400    {
[994]401        SubString tokens(evaluation.processedCommand_, " ", SubString::WhiteSpaces, false, '\\', false, '"', false, '(', ')', false, '\0');
[952]402
[972]403        if (evaluation.bEvaluatedParams_ && evaluation.evaluatedExecutor_)
[967]404        {
[1214]405std::cout << "CE_execute (evaluation): " << evaluation.evaluatedExecutor_->getName() << " " << evaluation.param_[0] << " " << evaluation.param_[1] << " " << evaluation.param_[2] << " " << evaluation.param_[3] << " " << evaluation.param_[4] << std::endl;
[967]406            (*evaluation.evaluatedExecutor_)(evaluation.param_[0], evaluation.param_[1], evaluation.param_[2], evaluation.param_[3], evaluation.param_[4]);
[1214]407            return true;
[967]408        }
409
[1214]410std::cout << "CE_execute: " << evaluation.processedCommand_ << "\n";
[952]411        switch (evaluation.state_)
[947]412        {
[955]413            case CS_Uninitialized:
414                break;
[947]415            case CS_Empty:
416                break;
417            case CS_FunctionClass_Or_Shortcut_Or_Keyword:
418                break;
419            case CS_Shortcut_Params:
[957]420                // not enough parameters but lets hope there are some additional parameters and go on
[947]421            case CS_Shortcut_Finished:
422                // call the shortcut
[972]423                if (evaluation.shortcut_)
[957]424                {
425                    if (tokens.size() >= 2)
[1050]426                        return evaluation.shortcut_->parse(removeSlashes(tokens.subSet(1).join() + evaluation.getAdditionalParameter()));
[957]427                    else
[1050]428                        return evaluation.shortcut_->parse(removeSlashes(evaluation.additionalParameter_));
[957]429                }
[947]430                break;
431            case CS_Function:
432                break;
433            case CS_Function_Params:
[957]434                // not enough parameters but lets hope there are some additional parameters and go on
[947]435            case CS_Function_Finished:
436                // call the shortcut
[972]437                if (evaluation.function_)
[957]438                {
439                    if (tokens.size() >= 3)
[1050]440                        return evaluation.function_->parse(removeSlashes(tokens.subSet(2).join() + evaluation.getAdditionalParameter()));
[957]441                    else
[1050]442                        return evaluation.function_->parse(removeSlashes(evaluation.additionalParameter_));
[957]443                }
[947]444                break;
445            case CS_ConfigValueClass:
446                break;
447            case CS_ConfigValue:
448                break;
449            case CS_ConfigValueType:
[957]450                // not enough parameters but lets hope there are some additional parameters and go on
[947]451            case CS_ConfigValueFinished:
452                // set the config value
[972]453                if (evaluation.configvalue_)
[957]454                {
455                    if ((tokens.size() >= 1) && (tokens[0] == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE))
456                    {
457                        if (tokens.size() >= 4)
[1050]458                            return evaluation.configvalue_->set(removeSlashes(tokens.subSet(3).join() + evaluation.getAdditionalParameter()));
[957]459                        else
[1050]460                            return evaluation.configvalue_->set(removeSlashes(evaluation.additionalParameter_));
[957]461                    }
462                    else if ((tokens.size() >= 1) && (tokens[0] == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY))
463                    {
464                        if (tokens.size() >= 4)
[1050]465                            return evaluation.configvalue_->tset(removeSlashes(tokens.subSet(3).join() + evaluation.getAdditionalParameter()));
[957]466                        else
[1050]467                            return evaluation.configvalue_->tset(removeSlashes(evaluation.additionalParameter_));
[957]468                    }
469                }
[947]470                break;
471            case CS_KeybindKey:
472                break;
473            case CS_KeybindCommand:
[957]474                // not enough parameters but lets hope there are some additional parameters and go on
[947]475            case CS_KeybindFinished:
476                // set the keybind
477                // ...todo
478                break;
479            case CS_Error:
480                break;
481        }
482
483        return false;
484    }
485
486    std::string CommandExecutor::complete(const std::string& command)
487    {
[955]488        if ((CommandExecutor::getEvaluation().processedCommand_ != command) || (CommandExecutor::getEvaluation().state_ == CS_Uninitialized))
[947]489            CommandExecutor::parse(command);
490
[952]491        return CommandExecutor::complete(CommandExecutor::getEvaluation());
492    }
[948]493
[952]494    std::string CommandExecutor::complete(const CommandEvaluation& evaluation)
495    {
[994]496        SubString tokens(evaluation.processedCommand_, " ", SubString::WhiteSpaces, false, '\\', false, '"', false, '(', ')', false, '\0');
[952]497
[972]498        std::list<std::pair<const std::string*, const std::string*> > temp;
[952]499        if (evaluation.state_ == CS_Empty)
[948]500        {
[952]501            temp.insert(temp.end(), evaluation.listOfPossibleShortcuts_.begin(), evaluation.listOfPossibleShortcuts_.end());
502            temp.insert(temp.end(), evaluation.listOfPossibleFunctionClasses_.begin(), evaluation.listOfPossibleFunctionClasses_.end());
[948]503        }
504
[952]505        switch (evaluation.state_)
[947]506        {
[955]507            case CS_Uninitialized:
508                break;
[947]509            case CS_Empty:
[955]510                return (CommandExecutor::getCommonBegin(temp));
[947]511                break;
512            case CS_FunctionClass_Or_Shortcut_Or_Keyword:
513                break;
514            case CS_Shortcut_Params:
[972]515                if (evaluation.shortcut_)
516                    return (evaluation.shortcut_->getName() + " ");
[947]517                break;
518            case CS_Shortcut_Finished:
[972]519                if (evaluation.shortcut_)
520                {
521                    if (evaluation.shortcut_->getParamCount() == 0)
522                        return (evaluation.shortcut_->getName());
523                    else if (tokens.size() >= 2)
524                        return (evaluation.shortcut_->getName() + " " + tokens.subSet(1).join());
525                }
[947]526                break;
527            case CS_Function:
[972]528                if (evaluation.functionclass_)
529                    return (evaluation.functionclass_->getName() + " " + CommandExecutor::getCommonBegin(evaluation.listOfPossibleFunctions_));
[947]530                break;
531            case CS_Function_Params:
[972]532                if (evaluation.functionclass_ && evaluation.function_)
533                    return (evaluation.functionclass_->getName() + " " + evaluation.function_->getName() + " ");
[947]534                break;
535            case CS_Function_Finished:
[972]536                if (evaluation.functionclass_ && evaluation.function_)
537                {
538                    if (evaluation.function_->getParamCount() == 0)
539                        return (evaluation.functionclass_->getName() + " " + evaluation.function_->getName());
540                    else if (tokens.size() >= 3)
541                        return (evaluation.functionclass_->getName() + " " + evaluation.function_->getName() + " " + tokens.subSet(2).join());
542                }
[947]543                break;
544            case CS_ConfigValueClass:
[955]545                if (tokens.size() >= 1)
[972]546                    return (tokens[0] + " " + CommandExecutor::getCommonBegin(evaluation.listOfPossibleConfigValueClasses_));
[947]547                break;
548            case CS_ConfigValue:
[972]549                if ((tokens.size() >= 1) && evaluation.configvalueclass_)
550                    return (tokens[0] + " " + evaluation.configvalueclass_->getName() + " " + CommandExecutor::getCommonBegin(evaluation.listOfPossibleConfigValues_));
[947]551                break;
552            case CS_ConfigValueType:
[972]553                if ((tokens.size() >= 1) && evaluation.configvalueclass_ && evaluation.configvalue_)
554                    return (tokens[0] + " " + evaluation.configvalueclass_->getName() + " " + evaluation.configvalue_->getName() + " ");
[947]555                break;
556            case CS_ConfigValueFinished:
[972]557                if ((tokens.size() >= 1) && evaluation.configvalueclass_ && evaluation.configvalue_ && (tokens.size() >= 4))
558                    return (tokens[0] + " " + evaluation.configvalueclass_->getName() + " " + evaluation.configvalue_->getName() + " " + tokens.subSet(3).join());
[947]559                break;
560            case CS_KeybindKey:
[955]561                if (tokens.size() >= 1)
[972]562                    return (tokens[0] + " " + CommandExecutor::getCommonBegin(evaluation.listOfPossibleKeys_));
[947]563                break;
564            case CS_KeybindCommand:
[952]565                if ((evaluation.processedCommand_.size() >= 1) && (evaluation.processedCommand_[evaluation.processedCommand_.size() - 1] != ' '))
566                    return (evaluation.processedCommand_ + " ");
[947]567                break;
568            case CS_KeybindFinished:
569                break;
570            case CS_Error:
571                break;
572        }
573
[952]574        return evaluation.processedCommand_;
[947]575    }
576
577    std::string CommandExecutor::hint(const std::string& command)
578    {
[955]579        if ((CommandExecutor::getEvaluation().processedCommand_ != command) || (CommandExecutor::getEvaluation().state_ == CS_Uninitialized))
[947]580            CommandExecutor::parse(command);
581
[952]582        return CommandExecutor::hint(CommandExecutor::getEvaluation());
583    }
[948]584
[952]585    std::string CommandExecutor::hint(const CommandEvaluation& evaluation)
586    {
[994]587        SubString tokens(evaluation.processedCommand_, " ", SubString::WhiteSpaces, false, '\\', false, '"', false, '(', ')', false, '\0');
[952]588
589        switch (evaluation.state_)
[947]590        {
[955]591            case CS_Uninitialized:
592                break;
[947]593            case CS_Empty:
[952]594                return (CommandExecutor::dump(evaluation.listOfPossibleShortcuts_) + "\n" + CommandExecutor::dump(evaluation.listOfPossibleFunctionClasses_));
[947]595                break;
596            case CS_FunctionClass_Or_Shortcut_Or_Keyword:
597                break;
598            case CS_Shortcut_Params:
[972]599                if (evaluation.shortcut_)
[952]600                    return CommandExecutor::dump(evaluation.shortcut_);
[947]601                break;
602            case CS_Shortcut_Finished:
[972]603                if (evaluation.shortcut_)
[952]604                    return CommandExecutor::dump(evaluation.shortcut_);
[947]605                break;
606            case CS_Function:
[952]607                return CommandExecutor::dump(evaluation.listOfPossibleFunctions_);
[947]608                break;
609            case CS_Function_Params:
[972]610                if (evaluation.function_)
[952]611                    return CommandExecutor::dump(evaluation.function_);
[947]612                break;
613            case CS_Function_Finished:
[972]614                if (evaluation.function_)
[952]615                    return CommandExecutor::dump(evaluation.function_);
[947]616                break;
617            case CS_ConfigValueClass:
[952]618                return CommandExecutor::dump(evaluation.listOfPossibleConfigValueClasses_);
[947]619                break;
620            case CS_ConfigValue:
[952]621                return CommandExecutor::dump(evaluation.listOfPossibleConfigValues_);
[947]622                break;
623            case CS_ConfigValueType:
[972]624                if (evaluation.configvalue_)
[952]625                    return CommandExecutor::dump(evaluation.configvalue_);
[947]626                break;
627            case CS_ConfigValueFinished:
[972]628                if (evaluation.configvalue_)
[952]629                    return CommandExecutor::dump(evaluation.configvalue_);
[947]630                break;
631            case CS_KeybindKey:
[952]632                return CommandExecutor::dump(evaluation.listOfPossibleKeys_);
[947]633                break;
634            case CS_KeybindCommand:
[972]635                if (evaluation.key_)
[952]636                    return CommandExecutor::dump(evaluation.key_);
[947]637                break;
638            case CS_KeybindFinished:
[972]639                if (evaluation.key_)
[952]640                    return CommandExecutor::dump(evaluation.key_);
[947]641                break;
642            case CS_Error:
[955]643                return CommandExecutor::getEvaluation().errorMessage_;
[947]644                break;
645        }
646
647        return "";
648    }
649
[953]650    CommandEvaluation CommandExecutor::evaluate(const std::string& command)
[952]651    {
652        CommandExecutor::parse(command, true);
[1214]653
654        if (CommandExecutor::getEvaluation().tokens_.size() > 0)
655        {
656            std::string lastToken;
657            lastToken = CommandExecutor::getEvaluation().tokens_[CommandExecutor::getEvaluation().tokens_.size() - 1];
658            lastToken = lastToken.substr(0, lastToken.size() - 1);
659            CommandExecutor::getEvaluation().tokens_.pop_back();
660            CommandExecutor::getEvaluation().tokens_.append(SubString(lastToken, " ", "", true, '\0', false, '\0', false, '\0', '\0', false, '\0'));
661        }
662
[967]663        CommandExecutor::getEvaluation().evaluateParams();
[952]664        return CommandExecutor::getEvaluation();
665    }
666
[947]667    void CommandExecutor::parse(const std::string& command, bool bInitialize)
668    {
[994]669        CommandExecutor::getEvaluation().tokens_.split((command + COMMAND_EXECUTOR_CURSOR), " ", SubString::WhiteSpaces, false, '\\', false, '"', false, '(', ')', false, '\0');
[952]670        CommandExecutor::getEvaluation().processedCommand_ = command;
[947]671
672        if (bInitialize)
[953]673            CommandExecutor::initialize(command);
[947]674
[952]675        switch (CommandExecutor::getEvaluation().state_)
[947]676        {
[955]677            case CS_Uninitialized:
678                // Impossible
679                break;
[947]680            case CS_Empty:
681                if (CommandExecutor::argumentsGiven() == 0)
682                {
683                    // We want a hint for the first token
684                    // Check if there is already a perfect match
[952]685                    CommandExecutor::getEvaluation().functionclass_ = CommandExecutor::getIdentifierOfPossibleFunctionClass(CommandExecutor::getToken(0));
686                    CommandExecutor::getEvaluation().shortcut_ = CommandExecutor::getExecutorOfPossibleShortcut(CommandExecutor::getToken(0));
[947]687
[972]688                    if ((CommandExecutor::getEvaluation().functionclass_) || (CommandExecutor::getEvaluation().shortcut_))
[947]689                    {
690                        // Yes, there is a class or a shortcut with the searched name
691                        // Add a whitespace and continue parsing
[952]692                        CommandExecutor::getEvaluation().state_ = CS_FunctionClass_Or_Shortcut_Or_Keyword;
[947]693                        CommandExecutor::parse(command + " ", false);
694                        return;
695                    }
696
697                    // No perfect match: Create the lists of all possible classes and shortcuts and return
698                    CommandExecutor::createListOfPossibleFunctionClasses(CommandExecutor::getToken(0));
699                    CommandExecutor::createListOfPossibleShortcuts(CommandExecutor::getToken(0));
[965]700
701                    // Check if there's only one possiblility
702                    if ((CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.size() == 1) && (CommandExecutor::getEvaluation().listOfPossibleShortcuts_.size() == 0))
703                    {
704                        // There's only one possible class
705                        CommandExecutor::getEvaluation().state_ = CS_Function;
[972]706                        CommandExecutor::getEvaluation().functionclass_ = CommandExecutor::getIdentifierOfPossibleFunctionClass(*(*CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.begin()).first);
707                        CommandExecutor::parse(*(*CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.begin()).first + " ", false);
[965]708                        return;
709                    }
710                    else if ((CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.size() == 0) && (CommandExecutor::getEvaluation().listOfPossibleShortcuts_.size() == 1))
711                    {
[972]712                        if ((*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first != COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE)
713                         && (*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first != COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY)
714                         && (*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first != COMMAND_EXECUTOR_KEYWORD_SET_KEYBIND))
[965]715                        {
716                            // There's only one possible shortcut
717                            CommandExecutor::getEvaluation().state_ = CS_Shortcut_Params;
[972]718                            CommandExecutor::getEvaluation().shortcut_ = CommandExecutor::getExecutorOfPossibleShortcut(*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first);
[965]719                        }
[972]720                        else if ((*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE)
721                              || (*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY))
[965]722                        {
723                            // It's the 'set' or 'tset' keyword
724                            CommandExecutor::getEvaluation().state_ = CS_ConfigValueClass;
725                        }
[972]726                        else if (*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first != COMMAND_EXECUTOR_KEYWORD_SET_KEYBIND)
[965]727                        {
728                            // It's the 'bind' keyword
729                            CommandExecutor::getEvaluation().state_ = CS_KeybindKey;
730                        }
731
[972]732                        CommandExecutor::parse(*(*CommandExecutor::getEvaluation().listOfPossibleShortcuts_.begin()).first + " ", false);
[965]733                        return;
734                    }
735
736                    // It's ambiguous
[947]737                    return;
738                }
739                else
740                {
741                    // There is at least one argument: Check if it's a shortcut, a classname or a special keyword
[952]742                    CommandExecutor::getEvaluation().state_ = CS_FunctionClass_Or_Shortcut_Or_Keyword;
[947]743                    CommandExecutor::parse(command, false);
744                    return;
745                }
746                break;
747            case CS_FunctionClass_Or_Shortcut_Or_Keyword:
748                if (CommandExecutor::argumentsGiven() >= 1)
749                {
750                    if ((CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE) || (CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY))
751                    {
752                        // We want to set a config value
[952]753                        CommandExecutor::getEvaluation().state_ = CS_ConfigValueClass;
[947]754                        CommandExecutor::parse(command, false);
755                        return;
756                    }
757                    else if (CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_KEYBIND)
758                    {
759                        // We want to set a keybinding
[952]760                        CommandExecutor::getEvaluation().state_ = CS_KeybindKey;
[947]761                        CommandExecutor::parse(command, false);
762                        return;
763                    }
764
[972]765                    if (!CommandExecutor::getEvaluation().functionclass_)
[952]766                        CommandExecutor::getEvaluation().functionclass_ = CommandExecutor::getIdentifierOfPossibleFunctionClass(CommandExecutor::getToken(0));
[972]767                    if (!CommandExecutor::getEvaluation().shortcut_)
[952]768                        CommandExecutor::getEvaluation().shortcut_ = CommandExecutor::getExecutorOfPossibleShortcut(CommandExecutor::getToken(0));
[947]769
[972]770                    if ((!CommandExecutor::getEvaluation().functionclass_) && (!CommandExecutor::getEvaluation().shortcut_))
[947]771                    {
772                        // Argument 1 seems to be wrong
773                        AddLanguageEntry("CommandExecutor::NoSuchCommandOrClassName", "No such command or classname");
[952]774                        CommandExecutor::getEvaluation().errorMessage_ = (CommandExecutor::getToken(0) + ": " + GetLocalisation("CommandExecutor::NoSuchCommandOrClassName"));
775                        CommandExecutor::getEvaluation().state_ = CS_Error;
[947]776                        return;
777                    }
[972]778                    else if (CommandExecutor::getEvaluation().shortcut_)
[947]779                    {
780                        // Argument 1 is a shortcut: Return the needed parameter types
[952]781                        CommandExecutor::getEvaluation().state_ = CS_Shortcut_Params;
[947]782                        CommandExecutor::parse(command, false);
783                        return;
784                    }
785                    else
786                    {
787                        // Argument 1 is a classname: Return the possible functions
[952]788                        CommandExecutor::getEvaluation().state_ = CS_Function;
[947]789                        CommandExecutor::parse(command, false);
790                        return;
791                    }
792                }
793                else
794                {
[952]795                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]796                    return;
797                }
798                break;
799            case CS_Shortcut_Params:
[972]800                if (CommandExecutor::getEvaluation().shortcut_)
[947]801                {
802                    // Valid command
803                    // Check if there are enough parameters
[952]804                    if (CommandExecutor::enoughParametersGiven(1, CommandExecutor::getEvaluation().shortcut_))
[947]805                    {
[952]806                        CommandExecutor::getEvaluation().state_ = CS_Shortcut_Finished;
[947]807                        return;
808                    }
809                }
810                else
811                {
812                    // Something is wrong
[952]813                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]814                    return;
815                }
816                break;
817            case CS_Function:
[972]818                if (CommandExecutor::getEvaluation().functionclass_)
[947]819                {
820                    // We have a valid classname
821                    // Check if there is a second argument
822                    if (CommandExecutor::argumentsGiven() >= 2)
823                    {
824                        // There is a second argument: Check if it's a valid functionname
[952]825                        CommandExecutor::getEvaluation().function_ = CommandExecutor::getExecutorOfPossibleFunction(CommandExecutor::getToken(1), CommandExecutor::getEvaluation().functionclass_);
[972]826                        if (!CommandExecutor::getEvaluation().function_)
[947]827                        {
828                            // Argument 2 seems to be wrong
829                            AddLanguageEntry("CommandExecutor::NoSuchFunctionnameIn", "No such functionname in");
[952]830                            CommandExecutor::getEvaluation().errorMessage_ = (CommandExecutor::getToken(1) + ": " + GetLocalisation("CommandExecutor::NoSuchFunctionnameIn") + " " + CommandExecutor::getEvaluation().functionclass_->getName());
831                            CommandExecutor::getEvaluation().state_ = CS_Error;
[947]832                            return;
833                        }
834                        else
835                        {
836                            // Argument 2 seems to be a valid functionname: Get the parameters
[952]837                            CommandExecutor::getEvaluation().state_ = CS_Function_Params;
[947]838                            CommandExecutor::parse(command, false);
839                            return;
840                        }
841                    }
842                    else
843                    {
844                        // There is no finished second argument
845                        // Check if there's already a perfect match
[952]846                        if (CommandExecutor::getEvaluation().tokens_.size() >= 2)
[947]847                        {
[952]848                            CommandExecutor::getEvaluation().function_ = CommandExecutor::getExecutorOfPossibleFunction(CommandExecutor::getToken(1), CommandExecutor::getEvaluation().functionclass_);
[972]849                            if (CommandExecutor::getEvaluation().function_)
[947]850                            {
851                                // There is a perfect match: Add a whitespace and continue parsing
[952]852                                CommandExecutor::getEvaluation().state_ = CS_Function_Params;
[947]853                                CommandExecutor::parse(command + " ", false);
854                                return;
855                            }
856                        }
857
858                        // No perfect match: Create the list of all possible functions and return
[952]859                        CommandExecutor::createListOfPossibleFunctions(CommandExecutor::getToken(1), CommandExecutor::getEvaluation().functionclass_);
[965]860
861                        // Check if there's only one possiblility
862                        if (CommandExecutor::getEvaluation().listOfPossibleFunctions_.size() == 1)
863                        {
864                            // There's only one possible function
865                            CommandExecutor::getEvaluation().state_ = CS_Function_Params;
[972]866                            CommandExecutor::getEvaluation().function_ = CommandExecutor::getExecutorOfPossibleFunction(*(*CommandExecutor::getEvaluation().listOfPossibleFunctions_.begin()).first, CommandExecutor::getEvaluation().functionclass_);
867                            CommandExecutor::parse(CommandExecutor::getToken(0) + " " + *(*CommandExecutor::getEvaluation().listOfPossibleFunctions_.begin()).first + " ", false);
[965]868                            return;
869                        }
870
871                        // It's ambiguous
[947]872                        return;
873                    }
874                }
875                else
876                {
[952]877                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]878                    return;
879                }
880                break;
881            case CS_Function_Params:
[972]882                if (CommandExecutor::getEvaluation().functionclass_ && CommandExecutor::getEvaluation().function_)
[947]883                {
884                    // Valid command
885                    // Check if there are enough parameters
[952]886                    if (CommandExecutor::enoughParametersGiven(2, CommandExecutor::getEvaluation().function_))
[947]887                    {
[952]888                        CommandExecutor::getEvaluation().state_ = CS_Function_Finished;
[947]889                        return;
890                    }
891                }
892                else
893                {
894                    // Something is wrong
[952]895                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]896                    return;
897                }
898                break;
899            case CS_ConfigValueClass:
900                if (((CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE) || (CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY)))
901                {
902                    // We want to set a config value
903                    // Check if there is a second argument
904                    if (CommandExecutor::argumentsGiven() >= 2)
905                    {
906                        // There is a second argument: Check if it's a valid classname
[952]907                        CommandExecutor::getEvaluation().configvalueclass_ = CommandExecutor::getIdentifierOfPossibleConfigValueClass(CommandExecutor::getToken(1));
[972]908                        if (!CommandExecutor::getEvaluation().configvalueclass_)
[947]909                        {
910                            // Argument 2 seems to be wrong
911                            AddLanguageEntry("CommandExecutor::NoSuchClassWithConfigValues", "No such class with config values");
[952]912                            CommandExecutor::getEvaluation().errorMessage_ = (CommandExecutor::getToken(1) + ": " + GetLocalisation("CommandExecutor::NoSuchClassWithConfigValues"));
913                            CommandExecutor::getEvaluation().state_ = CS_Error;
[947]914                            return;
915                        }
916                        else
917                        {
918                            // Argument 2 seems to be a valid classname: Search for possible config values
[952]919                            CommandExecutor::getEvaluation().state_ = CS_ConfigValue;
[947]920                            CommandExecutor::parse(command, false);
921                            return;
922                        }
923                    }
924                    else
925                    {
926                        // There's no finished second argument
927                        // Check if there's already a perfect match
[952]928                        if (CommandExecutor::getEvaluation().tokens_.size() >= 2)
[947]929                        {
[952]930                            CommandExecutor::getEvaluation().configvalueclass_ = CommandExecutor::getIdentifierOfPossibleConfigValueClass(CommandExecutor::getToken(1));
[972]931                            if (CommandExecutor::getEvaluation().configvalueclass_)
[947]932                            {
933                                // There is a perfect match: Add a whitespace and continue parsing
[952]934                                CommandExecutor::getEvaluation().state_ = CS_ConfigValue;
[947]935                                CommandExecutor::parse(command + " ", false);
936                                return;
937                            }
938                        }
939
940                        // No perfect match: Create the list of all possible classnames and return
941                        CommandExecutor::createListOfPossibleConfigValueClasses(CommandExecutor::getToken(1));
[965]942
943                        // Check if there's only one possiblility
944                        if (CommandExecutor::getEvaluation().listOfPossibleConfigValueClasses_.size() == 1)
945                        {
946                            // There's only one possible classname
947                            CommandExecutor::getEvaluation().state_ = CS_ConfigValue;
[972]948                            CommandExecutor::getEvaluation().configvalueclass_ = CommandExecutor::getIdentifierOfPossibleConfigValueClass(*(*CommandExecutor::getEvaluation().listOfPossibleConfigValueClasses_.begin()).first);
949                            CommandExecutor::parse(CommandExecutor::getToken(0) + " " + *(*CommandExecutor::getEvaluation().listOfPossibleConfigValueClasses_.begin()).first + " ", false);
[965]950                            return;
951                        }
952
953                        // It's ambiguous
[947]954                        return;
955                    }
956                }
957                else
958                {
959                    // Something is wrong
[952]960                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]961                    return;
962                }
963                break;
964            case CS_ConfigValue:
[972]965                if (((CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE) || (CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY)) && (CommandExecutor::getEvaluation().configvalueclass_))
[947]966                {
967                    // Check if there is a third argument
968                    if (CommandExecutor::argumentsGiven() >= 3)
969                    {
970                        // There is a third argument: Check if it's a valid config value
[952]971                        CommandExecutor::getEvaluation().configvalue_ = CommandExecutor::getContainerOfPossibleConfigValue(CommandExecutor::getToken(2), CommandExecutor::getEvaluation().configvalueclass_);
[972]972                        if (!CommandExecutor::getEvaluation().configvalue_)
[947]973                        {
974                            // Argument 3 seems to be wrong
975                            AddLanguageEntry("CommandExecutor::NoSuchConfigValueIn", "No such config value in");
[952]976                            CommandExecutor::getEvaluation().errorMessage_ = (CommandExecutor::getToken(2) + ": " + GetLocalisation("CommandExecutor::NoSuchConfigValueIn") + " " + CommandExecutor::getEvaluation().configvalueclass_->getName());
977                            CommandExecutor::getEvaluation().state_ = CS_Error;
[947]978                            return;
979                        }
980                        else
981                        {
982                            // Argument 3 seems to be a valid config value: Get the type
[952]983                            CommandExecutor::getEvaluation().state_ = CS_ConfigValueType;
[947]984                            CommandExecutor::parse(command, false);
985                            return;
986                        }
987                    }
988                    else
989                    {
990                        // There is no finished third argument
991                        // Check if there's already a perfect match
[952]992                        if (CommandExecutor::getEvaluation().tokens_.size() >= 3)
[947]993                        {
[952]994                            CommandExecutor::getEvaluation().configvalue_ = CommandExecutor::getContainerOfPossibleConfigValue(CommandExecutor::getToken(2), CommandExecutor::getEvaluation().configvalueclass_);
[972]995                            if (CommandExecutor::getEvaluation().configvalue_)
[947]996                            {
997                                // There is a perfect match: Add a whitespace and continue parsing
[952]998                                CommandExecutor::getEvaluation().state_ = CS_ConfigValueType;
[947]999                                CommandExecutor::parse(command + " ", false);
1000                                return;
1001                            }
1002                        }
1003
1004                        // No perfect match: Create the list of all possible config values
[952]1005                        CommandExecutor::createListOfPossibleConfigValues(CommandExecutor::getToken(2), CommandExecutor::getEvaluation().configvalueclass_);
[965]1006
1007                        // Check if there's only one possiblility
1008                        if (CommandExecutor::getEvaluation().listOfPossibleConfigValues_.size() == 1)
1009                        {
1010                            // There's only one possible config value
1011                            CommandExecutor::getEvaluation().state_ = CS_ConfigValueType;
[972]1012                            CommandExecutor::getEvaluation().configvalue_ = CommandExecutor::getContainerOfPossibleConfigValue(*(*CommandExecutor::getEvaluation().listOfPossibleConfigValues_.begin()).first, CommandExecutor::getEvaluation().configvalueclass_);
1013                            CommandExecutor::parse(CommandExecutor::getToken(0) + " " + CommandExecutor::getToken(1) + " " + *(*CommandExecutor::getEvaluation().listOfPossibleConfigValues_.begin()).first + " ", false);
[965]1014                            return;
1015                        }
1016
1017                        // It's ambiguous
[947]1018                        return;
1019                    }
1020                }
1021                else
1022                {
1023                    // Something is wrong
[952]1024                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]1025                    return;
1026                }
1027                break;
1028            case CS_ConfigValueType:
[972]1029                if (((CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE) || (CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_CONFIG_VALUE_TEMPORARY)) && CommandExecutor::getEvaluation().configvalueclass_ && CommandExecutor::getEvaluation().configvalue_)
[947]1030                {
1031                    // Valid command
1032                    // Check if there are enough parameters
[972]1033                    if ((CommandExecutor::getEvaluation().tokens_.size() >= 4) && (CommandExecutor::getEvaluation().tokens_[3] != COMMAND_EXECUTOR_CURSOR))
[947]1034                    {
[952]1035                        CommandExecutor::getEvaluation().state_ = CS_ConfigValueFinished;
[947]1036                        return;
1037                    }
1038                }
1039                else
1040                {
1041                    // Something is wrong
[952]1042                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]1043                    return;
1044                }
1045                break;
1046            case CS_KeybindKey:
1047                if ((CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_KEYBIND))
1048                {
1049                    // todo
1050                }
1051                else
1052                {
1053                    // Something is wrong
[952]1054                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]1055                    return;
1056                }
1057                break;
1058            case CS_KeybindCommand:
1059                if ((CommandExecutor::getToken(0) == COMMAND_EXECUTOR_KEYWORD_SET_KEYBIND) && (false)) // todo
1060                {
1061                    // Valid command
1062                    // Check if there are enough parameters
[952]1063                    if (CommandExecutor::getEvaluation().tokens_.size() >= 3)
[947]1064                    {
[952]1065                        CommandExecutor::getEvaluation().state_ = CS_KeybindFinished;
[947]1066                        return;
1067                    }
1068
1069                }
1070                else
1071                {
1072                    // Something is wrong
[952]1073                    CommandExecutor::getEvaluation().state_ = CS_Error;
[947]1074                    return;
1075                }
1076                break;
1077            case CS_Shortcut_Finished:
1078                // Nothing to do
1079                break;
1080            case CS_Function_Finished:
1081                // Nothing to do
1082                break;
1083            case CS_ConfigValueFinished:
1084                // Nothing to do
1085                break;
1086            case CS_KeybindFinished:
1087                // Nothing to do
1088                break;
1089            case CS_Error:
1090                // This is bad
1091                break;
1092        }
1093    }
1094
[953]1095    void CommandExecutor::initialize(const std::string& command)
[947]1096    {
[953]1097        CommandExecutor::getEvaluation().processedCommand_ = command;
1098        CommandExecutor::getEvaluation().additionalParameter_ = "";
1099
[1214]1100        CommandExecutor::getEvaluation().bEvaluatedParams_ = false;
1101        CommandExecutor::getEvaluation().evaluatedExecutor_ = 0;
1102
[952]1103        CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.clear();
1104        CommandExecutor::getEvaluation().listOfPossibleShortcuts_.clear();
1105        CommandExecutor::getEvaluation().listOfPossibleFunctions_.clear();
1106        CommandExecutor::getEvaluation().listOfPossibleConfigValueClasses_.clear();
1107        CommandExecutor::getEvaluation().listOfPossibleConfigValues_.clear();
1108        CommandExecutor::getEvaluation().listOfPossibleKeys_.clear();
[947]1109
[952]1110        CommandExecutor::getEvaluation().functionclass_ = 0;
1111        CommandExecutor::getEvaluation().configvalueclass_ = 0;
1112        CommandExecutor::getEvaluation().shortcut_ = 0;
1113        CommandExecutor::getEvaluation().function_ = 0;
1114        CommandExecutor::getEvaluation().configvalue_ = 0;
1115        CommandExecutor::getEvaluation().key_ = 0;
[947]1116
[952]1117        CommandExecutor::getEvaluation().errorMessage_ = "";
1118        CommandExecutor::getEvaluation().state_ = CS_Empty;
[947]1119    }
1120
1121    bool CommandExecutor::argumentsGiven(unsigned int num)
1122    {
1123        // Because we added a cursor we have +1 arguments
1124        // There are num arguments given if there are at least num arguments + one cursor
[952]1125        return (CommandExecutor::getEvaluation().tokens_.size() >= (num + 1));
[947]1126    }
1127
1128    unsigned int CommandExecutor::argumentsGiven()
1129    {
1130        // Because we added a cursor we have +1 arguments
[952]1131        if (CommandExecutor::getEvaluation().tokens_.size() >= 1)
1132            return (CommandExecutor::getEvaluation().tokens_.size() - 1);
[947]1133        else
1134            return 0;
1135    }
1136
1137    std::string CommandExecutor::getToken(unsigned int index)
1138    {
[952]1139        if ((index >= 0) && (index < (CommandExecutor::getEvaluation().tokens_.size() - 1)))
1140            return CommandExecutor::getEvaluation().tokens_[index];
1141        else if (index == (CommandExecutor::getEvaluation().tokens_.size() - 1))
1142            return CommandExecutor::getEvaluation().tokens_[index].substr(0, CommandExecutor::getEvaluation().tokens_[index].size() - 1);
[947]1143        else
1144            return "";
1145    }
1146
1147    bool CommandExecutor::enoughParametersGiven(unsigned int head, Executor* executor)
1148    {
1149        unsigned int neededParams = head + executor->getParamCount();
[972]1150        /*
[947]1151        for (unsigned int i = executor->getParamCount() - 1; i >= 0; i--)
1152        {
1153            if (executor->defaultValueSet(i))
1154                neededParams--;
1155            else
1156                break;
1157        }
[972]1158        */
1159        return ((CommandExecutor::getEvaluation().tokens_.size() >= neededParams) && (CommandExecutor::getEvaluation().tokens_[neededParams - 1] != COMMAND_EXECUTOR_CURSOR));
[947]1160    }
1161
1162    void CommandExecutor::createListOfPossibleFunctionClasses(const std::string& fragment)
1163    {
1164        for (std::map<std::string, Identifier*>::const_iterator it = Identifier::getLowercaseIdentifierMapBegin(); it != Identifier::getLowercaseIdentifierMapEnd(); ++it)
1165        {
1166            if ((*it).second->hasConsoleCommands())
1167            {
[955]1168                if ((*it).first.find(getLowercase(fragment)) == 0 || fragment == "")
[947]1169                {
[972]1170                    CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.push_back(std::pair<const std::string*, const std::string*>(&(*it).first, &(*it).second->getName()));
[947]1171                }
1172            }
1173        }
1174
[952]1175        CommandExecutor::getEvaluation().listOfPossibleFunctionClasses_.sort(CommandExecutor::compareStringsInList);
[947]1176    }
1177
1178    void CommandExecutor::createListOfPossibleShortcuts(const std::string& fragment)
1179    {
1180        for (std::map<std::string, ExecutorStatic*>::const_iterator it = CommandExecutor::getLowercaseConsoleCommandShortcutMapBegin(); it != CommandExecutor::getLowercaseConsoleCommandShortcutMapEnd(); ++it)
1181        {
[955]1182            if ((*it).first.find(getLowercase(fragment)) == 0 || fragment == "")
[947]1183            {
[972]1184                CommandExecutor::getEvaluation().listOfPossibleShortcuts_.push_back(std::pair<const std::string*, const std::string*>(&(*it).first, &(*it).second->getName()));
[947]1185            }
1186        }
1187
[952]1188        CommandExecutor::getEvaluation().listOfPossibleShortcuts_.sort(CommandExecutor::compareStringsInList);
[947]1189    }
1190
1191    void CommandExecutor::createListOfPossibleFunctions(const std::string& fragment, Identifier* identifier)
1192    {
1193        for (std::map<std::string, ExecutorStatic*>::const_iterator it = identifier->getLowercaseConsoleCommandMapBegin(); it != identifier->getLowercaseConsoleCommandMapEnd(); ++it)
1194        {
[955]1195            if ((*it).first.find(getLowercase(fragment)) == 0 || fragment == "")
[947]1196            {
[972]1197                CommandExecutor::getEvaluation().listOfPossibleFunctions_.push_back(std::pair<const std::string*, const std::string*>(&(*it).first, &(*it).second->getName()));
[947]1198            }
1199        }
1200
[952]1201        CommandExecutor::getEvaluation().listOfPossibleFunctions_.sort(CommandExecutor::compareStringsInList);
[947]1202    }
1203
1204    void CommandExecutor::createListOfPossibleConfigValueClasses(const std::string& fragment)
1205    {
1206        for (std::map<std::string, Identifier*>::const_iterator it = Identifier::getLowercaseIdentifierMapBegin(); it != Identifier::getLowercaseIdentifierMapEnd(); ++it)
1207        {
1208            if ((*it).second->hasConfigValues())
1209            {
[955]1210                if ((*it).first.find(getLowercase(fragment)) == 0 || fragment == "")
[947]1211                {
[972]1212                    CommandExecutor::getEvaluation().listOfPossibleConfigValueClasses_.push_back(std::pair<const std::string*, const std::string*>(&(*it).first, &(*it).second->getName()));
[947]1213                }
1214            }
1215        }
1216
[952]1217        CommandExecutor::getEvaluation().listOfPossibleConfigValueClasses_.sort(CommandExecutor::compareStringsInList);
[947]1218    }
1219
1220    void CommandExecutor::createListOfPossibleConfigValues(const std::string& fragment, Identifier* identifier)
1221    {
1222        for (std::map<std::string, ConfigValueContainer*>::const_iterator it = identifier->getLowercaseConfigValueMapBegin(); it != identifier->getLowercaseConfigValueMapEnd(); ++it)
1223        {
[955]1224            if ((*it).first.find(getLowercase(fragment)) == 0 || fragment == "")
[947]1225            {
[972]1226                CommandExecutor::getEvaluation().listOfPossibleConfigValues_.push_back(std::pair<const std::string*, const std::string*>(&(*it).first, &(*it).second->getName()));
[947]1227            }
1228        }
1229
[952]1230        CommandExecutor::getEvaluation().listOfPossibleConfigValues_.sort(CommandExecutor::compareStringsInList);
[947]1231    }
1232
1233    void CommandExecutor::createListOfPossibleKeys(const std::string& fragment)
1234    {
1235        // todo
1236
[952]1237        CommandExecutor::getEvaluation().listOfPossibleKeys_.sort(CommandExecutor::compareStringsInList);
[947]1238    }
1239
[972]1240    bool CommandExecutor::compareStringsInList(const std::pair<const std::string*, const std::string*>& first, const std::pair<const std::string*, const std::string*>& second)
[947]1241    {
[972]1242        return ((*first.first) < (*second.first));
[947]1243    }
1244
1245    Identifier* CommandExecutor::getIdentifierOfPossibleFunctionClass(const std::string& name)
1246    {
1247        std::map<std::string, Identifier*>::const_iterator it = Identifier::getLowercaseIdentifierMap().find(getLowercase(name));
1248        if ((it != Identifier::getLowercaseIdentifierMapEnd()) && (*it).second->hasConsoleCommands())
1249            return (*it).second;
1250
1251        return 0;
1252    }
1253
1254    ExecutorStatic* CommandExecutor::getExecutorOfPossibleShortcut(const std::string& name)
1255    {
1256        std::map<std::string, ExecutorStatic*>::const_iterator it = CommandExecutor::getLowercaseConsoleCommandShortcutMap().find(getLowercase(name));
1257        if (it != CommandExecutor::getLowercaseConsoleCommandShortcutMapEnd())
1258            return (*it).second;
1259
1260        return 0;
1261    }
1262
1263    ExecutorStatic* CommandExecutor::getExecutorOfPossibleFunction(const std::string& name, Identifier* identifier)
1264    {
1265        std::map<std::string, ExecutorStatic*>::const_iterator it = identifier->getLowercaseConsoleCommandMap().find(getLowercase(name));
1266        if (it != identifier->getLowercaseConsoleCommandMapEnd())
1267            return (*it).second;
1268
1269        return 0;
1270    }
1271
1272    Identifier* CommandExecutor::getIdentifierOfPossibleConfigValueClass(const std::string& name)
1273    {
1274        std::map<std::string, Identifier*>::const_iterator it = Identifier::getLowercaseIdentifierMap().find(getLowercase(name));
1275        if ((it != Identifier::getLowercaseIdentifierMapEnd()) && (*it).second->hasConfigValues())
1276            return (*it).second;
1277
1278        return 0;
1279    }
1280
1281    ConfigValueContainer* CommandExecutor::getContainerOfPossibleConfigValue(const std::string& name, Identifier* identifier)
1282    {
1283        std::map<std::string, ConfigValueContainer*>::const_iterator it = identifier->getLowercaseConfigValueMap().find(getLowercase(name));
1284        if (it != identifier->getLowercaseConfigValueMapEnd())
[955]1285        {
[947]1286            return (*it).second;
[955]1287        }
[947]1288
1289        return 0;
1290    }
1291
1292    ConfigValueContainer* CommandExecutor::getContainerOfPossibleKey(const std::string& name)
1293    {
1294        // todo
1295
1296        return 0;
1297    }
1298
[972]1299    std::string CommandExecutor::dump(const std::list<std::pair<const std::string*, const std::string*> >& list)
[947]1300    {
1301        std::string output = "";
[972]1302        for (std::list<std::pair<const std::string*, const std::string*> >::const_iterator it = list.begin(); it != list.end(); ++it)
[947]1303        {
1304            if (it != list.begin())
1305                output += " ";
1306
[972]1307            output += *(*it).second;
[947]1308        }
1309        return output;
1310    }
1311
1312    std::string CommandExecutor::dump(const ExecutorStatic* executor)
1313    {
1314        std::string output = "";
1315        for (unsigned int i = 0; i < executor->getParamCount(); i++)
1316        {
1317            if (i != 0)
1318                output += " ";
1319
1320            if (executor->defaultValueSet(i))
1321                output += "[";
1322            else
1323                output += "{";
1324
1325            output += executor->getTypenameParam(i);
1326
1327            if (executor->defaultValueSet(i))
[957]1328                output += "=" + executor->getDefaultValue(i).toString() + "]";
[947]1329            else
1330                output += "}";
1331        }
1332        return output;
1333    }
1334
1335    std::string CommandExecutor::dump(const ConfigValueContainer* container)
1336    {
[957]1337        AddLanguageEntry("CommandExecutor::oldvalue", "old value");
[1030]1338        if (!container->isVector())
1339            return ("{" + container->getTypename() + "} (" + GetLocalisation("CommandExecutor::oldvalue") + ": " + container->toString() + ")");
1340        else
1341            return ("(vector<" + container->getTypename() + ">) (size: " + getConvertedValue<unsigned int, std::string>(container->getVectorSize()) + ")");
[947]1342    }
[948]1343
[972]1344    std::string CommandExecutor::getCommonBegin(const std::list<std::pair<const std::string*, const std::string*> >& list)
[948]1345    {
1346        if (list.size() == 0)
1347        {
1348            return "";
1349        }
1350        else if (list.size() == 1)
1351        {
[972]1352            return ((*(*list.begin()).first) + " ");
[948]1353        }
1354        else
1355        {
1356            std::string output = "";
1357            for (unsigned int i = 0; true; i++)
1358            {
1359                char temp = 0;
[972]1360                for (std::list<std::pair<const std::string*, const std::string*> >::const_iterator it = list.begin(); it != list.end(); ++it)
[948]1361                {
[972]1362                    if ((*(*it).first).size() > i)
[948]1363                    {
1364                        if (it == list.begin())
1365                        {
[972]1366                            temp = (*(*it).first)[i];
[948]1367                        }
1368                        else
1369                        {
[972]1370                            if (temp != (*(*it).first)[i])
[948]1371                                return output;
1372                        }
1373                    }
1374                    else
1375                    {
1376                        return output;
1377                    }
1378                }
1379                output += temp;
1380            }
1381            return output;
1382        }
1383    }
[947]1384}
Note: See TracBrowser for help on using the repository browser.