Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: orxonox.OLD/trunk/src/util/shell_command.cc @ 5144

Last change on this file since 5144 was 5144, checked in by bensch, 20 years ago

orxonox/trunk: now it is also possible, to execute Functions on NON-singleton classes

File size: 5.7 KB
Line 
1/*
2   orxonox - the future of 3D-vertical-scrollers
3
4   Copyright (C) 2004 orx
5
6   This program is free software; you can redistribute it and/or modify
7   it under the terms of the GNU General Public License as published by
8   the Free Software Foundation; either version 2, or (at your option)
9   any later version.
10
11   ### File Specific:
12   main-programmer: Benjamin Grauer
13   co-programmer: ...
14*/
15
16//#define DEBUG_SPECIAL_MODULE DEBUG_MODULE_
17
18#include "shell_command.h"
19
20#include "list.h"
21#include "debug.h"
22#include "class_list.h"
23
24#include "key_names.h"
25#include <stdarg.h>
26#include <stdio.h>
27
28using namespace std;
29
30ShellCommandBase::ShellCommandBase(const char* commandName, ClassID classID, unsigned int paramCount, ...)
31{
32  this->setClassID(CL_SHELL_COMMAND, "ShellCommand");
33  this->setName(commandName);
34
35  va_list parameters;
36  va_start(parameters, paramCount);
37
38  this->classID = classID;
39  this->className = ClassList::IDToString(classID);
40
41  if (classID & CL_MASK_SINGLETON == CL_MASK_SINGLETON)
42    this->isSingleton = true;
43  else
44    this->isSingleton = false;
45
46  // handling parameters, and storing them:
47  if (paramCount > FUNCTOR_MAX_ARGUMENTS)
48    paramCount = FUNCTOR_MAX_ARGUMENTS;
49  this->paramCount = paramCount;
50  this->parameters = new long[paramCount];
51
52  for (unsigned int i = 0; i < paramCount; i++)
53  {
54    parameters[i] = va_arg(parameters, long);
55
56    switch (parameters[i])
57    {
58      case ParameterBool:
59        this->defaultBools[i] = va_arg(parameters, int);
60        break;
61      case ParameterChar:
62        this->defaultStrings[i] = new char[2];
63        sprintf(this->defaultStrings[0], "%c",  va_arg(parameters, int));
64        break;
65      case ParameterString:
66        this->defaultStrings[i] = va_arg(parameters, char*);
67        break;
68      case ParameterInt:
69        this->defaultInts[i] = va_arg(parameters, int);
70        break;
71      case ParameterUInt:
72        this->defaultInts[i] = va_arg(parameters, unsigned int);
73        break;
74      case ParameterFloat:
75        this->defaultFloats[i] = va_arg(parameters, double);
76        break;
77      case ParameterLong:
78        this->defaultInts[i] = va_arg(parameters, long);
79        break;
80      default:
81        break;
82    }
83  }
84
85  // adding this ShellCommand to the list of known Commands
86  ShellCommandBase::commandList->add(this);
87}
88
89ShellCommandBase::~ShellCommandBase()
90{
91  delete[] this->parameters;
92}
93
94
95tList<ShellCommandBase>* ShellCommandBase::commandList = NULL;
96
97bool ShellCommandBase::isRegistered(const char* commandName, ClassID classID, unsigned int paramCount, ...)
98{
99  va_list parameters;
100  va_start(parameters, paramCount);
101
102  if (ShellCommandBase::commandList == NULL)
103  {
104    ShellCommandBase::commandList = new tList<ShellCommandBase>;
105    return false;
106  }
107
108  tIterator<ShellCommandBase>* iterator = ShellCommandBase::commandList->getIterator();
109  ShellCommandBase* elem = iterator->firstElement();
110  while(elem != NULL)
111  {
112    if (classID == elem->classID && !strcmp(commandName, elem->getName()))
113    {
114      PRINTF(2)("Command already registered\n");
115      delete iterator;
116      return true;
117    }
118    elem = iterator->nextElement();
119  }
120  delete iterator;
121  return false;
122}
123
124
125bool ShellCommandBase::execute(const char* executionString)
126{
127  if (ShellCommandBase::commandList == NULL)
128    return false;
129
130  tIterator<ShellCommandBase>* iterator = ShellCommandBase::commandList->getIterator();
131  ShellCommandBase* elem = iterator->firstElement();
132  while(elem != NULL)
133  {
134    printf("%s\n", elem->getName());
135    if (!strncasecmp (executionString, elem->className, strlen(elem->className)) &&
136         (*(executionString+strlen(elem->className)) == ' ' ||
137          *(executionString+strlen(elem->className)) == ':' ))
138    {
139      const char* commandBegin = executionString + strlen(elem->className);
140
141      PRINTF(4)("Class %s matches\n", elem->className);
142      BaseObject* objectPointer = NULL;
143      if (elem->isSingleton)
144      {
145        while(*commandBegin == ' ')
146          commandBegin++;
147        if (strncmp (commandBegin, elem->getName(), strlen(elem->getName())))
148        {
149          elem = iterator->nextElement();
150          continue;
151        }
152        // getting singleton-reference
153        objectPointer = ClassList::getList(elem->classID)->firstElement();
154      }
155      else
156      {
157        // checking for the Object
158        while(*commandBegin == ' ')
159          commandBegin++;
160        tIterator<BaseObject>* iterBO = ClassList::getList(elem->classID)->getIterator();
161        BaseObject* enumBO = iterBO->firstElement();
162        while(enumBO != NULL)
163        {
164          if(!strncmp(commandBegin, enumBO->getName(), strlen(enumBO->getName())))
165          {
166            PRINTF(4)("Object %s matches\n", enumBO->getName());
167            objectPointer = enumBO;
168            break;
169          }
170          enumBO = iterBO->nextElement();
171        }
172        delete iterBO;
173
174        // break on no object Found. We cannot operate on Classes, but on Objects
175        if (objectPointer == NULL)
176          break;
177        commandBegin = commandBegin + strlen(objectPointer->getName());
178        while(*commandBegin == ' ')
179          commandBegin++;
180        // checking for the requested function.
181        if (strncmp (commandBegin, elem->getName(), strlen(elem->getName())))
182        {
183          elem = iterator->nextElement();
184          continue;
185        }
186        PRINTF(3)("Function '%s' found\n", commandBegin);
187      }
188      const char* paramBegin = strchr(commandBegin, ' ');
189      if (paramBegin == NULL)
190        paramBegin = commandBegin + strlen(elem->getName());
191
192      if (objectPointer != NULL)
193      {
194        elem->executeCommand(objectPointer, paramBegin);
195        delete iterator;
196        return true;
197      }
198    }
199    elem = iterator->nextElement();
200  }
201  delete iterator;
202  return true;
203}
Note: See TracBrowser for help on using the repository browser.