| [4597] | 1 | /* | 
|---|
| [1853] | 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. | 
|---|
| [1855] | 10 |  | 
|---|
 | 11 |    ### File Specific: | 
|---|
| [3655] | 12 |    main-programmer: Benjamin Grauer | 
|---|
| [3672] | 13 |    co-programmer: Patrick Boenzli | 
|---|
| [1853] | 14 | */ | 
|---|
 | 15 |  | 
|---|
| [3655] | 16 | #define DEBUG_SPECIAL_MODULE DEBUG_MODULE_LOAD | 
|---|
| [1853] | 17 |  | 
|---|
| [3655] | 18 | #include "resource_manager.h" | 
|---|
| [1853] | 19 |  | 
|---|
| [6648] | 20 | #include "substring.h" | 
|---|
| [4642] | 21 | #include "debug.h" | 
|---|
 | 22 |  | 
|---|
| [6222] | 23 | #include <algorithm> | 
|---|
| [6655] | 24 | #include <assert.h> | 
|---|
| [6222] | 25 |  | 
|---|
| [3655] | 26 | // different resource Types | 
|---|
| [4534] | 27 | #ifndef NO_MODEL | 
|---|
| [3655] | 28 | #include "objModel.h" | 
|---|
| [3657] | 29 | #include "primitive_model.h" | 
|---|
| [4462] | 30 | #include "md2Model.h" | 
|---|
| [4534] | 31 | #endif /* NO_MODEL */ | 
|---|
 | 32 | #ifndef NO_TEXTURES | 
|---|
| [3655] | 33 | #include "texture.h" | 
|---|
| [4534] | 34 | #endif /* NO_TEXTURES */ | 
|---|
 | 35 | #ifndef NO_TEXT | 
|---|
| [5344] | 36 | #include "font.h" | 
|---|
| [4534] | 37 | #endif /* NO_TEXT */ | 
|---|
 | 38 | #ifndef NO_AUDIO | 
|---|
| [5930] | 39 | #include "sound_buffer.h" | 
|---|
| [4961] | 40 | #include "ogg_player.h" | 
|---|
| [4534] | 41 | #endif /* NO_AUDIO */ | 
|---|
| [5323] | 42 | #ifndef NO_SHADERS | 
|---|
 | 43 | #include "shader.h" | 
|---|
 | 44 | #endif /* NO_SHADERS */ | 
|---|
| [1853] | 45 |  | 
|---|
| [3655] | 46 | // File Handling Includes | 
|---|
 | 47 | #include <sys/types.h> | 
|---|
 | 48 | #include <sys/stat.h> | 
|---|
 | 49 | #include <unistd.h> | 
|---|
 | 50 |  | 
|---|
| [1856] | 51 | using namespace std; | 
|---|
| [1853] | 52 |  | 
|---|
| [3245] | 53 | /** | 
|---|
| [6647] | 54 |  * @brief standard constructor | 
|---|
| [3245] | 55 | */ | 
|---|
| [4597] | 56 | ResourceManager::ResourceManager () | 
|---|
| [3365] | 57 | { | 
|---|
| [4597] | 58 |   this->setClassID(CL_RESOURCE_MANAGER, "ResourceManager"); | 
|---|
 | 59 |   this->setName("ResourceManager"); | 
|---|
 | 60 |  | 
|---|
| [5423] | 61 |   this->dataDir = new char[3]; | 
|---|
 | 62 |   strcpy(this->dataDir, "./"); | 
|---|
| [5480] | 63 |   this->tryDataDir("./data"); | 
|---|
| [3365] | 64 | } | 
|---|
| [1853] | 65 |  | 
|---|
| [3658] | 66 | //! Singleton Reference to the ResourceManager | 
|---|
| [3655] | 67 | ResourceManager* ResourceManager::singletonRef = NULL; | 
|---|
 | 68 |  | 
|---|
| [3245] | 69 | /** | 
|---|
| [6647] | 70 |  * @brief standard destructor | 
|---|
| [3655] | 71 | */ | 
|---|
| [4746] | 72 | ResourceManager::~ResourceManager () | 
|---|
| [3655] | 73 | { | 
|---|
| [3660] | 74 |   // deleting the Resources-List | 
|---|
| [3672] | 75 |   this->unloadAllByPriority(RP_GAME); | 
|---|
| [5303] | 76 |  | 
|---|
| [6222] | 77 |   if (!this->resourceList.empty()) | 
|---|
 | 78 |     PRINTF(1)("Not removed all Resources, since there are still %d resources registered\n", this->resourceList.size()); | 
|---|
| [5303] | 79 |  | 
|---|
| [3660] | 80 |   // deleting the Directorie Lists | 
|---|
| [6222] | 81 |   while (!this->imageDirs.empty()) | 
|---|
| [6640] | 82 |   { | 
|---|
| [6642] | 83 |     delete[] this->imageDirs.back(); | 
|---|
 | 84 |     this->imageDirs.pop_back(); | 
|---|
| [6640] | 85 |   } | 
|---|
| [3672] | 86 |  | 
|---|
| [5211] | 87 |   delete[] this->dataDir; | 
|---|
 | 88 |  | 
|---|
| [3655] | 89 |   ResourceManager::singletonRef = NULL; | 
|---|
 | 90 | } | 
|---|
| [1853] | 91 |  | 
|---|
| [3655] | 92 | /** | 
|---|
| [6647] | 93 |  * @brief sets the data main directory | 
|---|
| [4836] | 94 |  * @param dataDir the DataDirectory. | 
|---|
| [5480] | 95 |  */ | 
|---|
| [3883] | 96 | bool ResourceManager::setDataDir(const char* dataDir) | 
|---|
| [3543] | 97 | { | 
|---|
| [4341] | 98 |   char* realDir = ResourceManager::homeDirCheck(dataDir); | 
|---|
 | 99 |   if (isDir(realDir)) | 
|---|
| [5480] | 100 |   { | 
|---|
 | 101 |     delete[] this->dataDir; | 
|---|
 | 102 |     if (dataDir[strlen(dataDir)-1] == '/' || dataDir[strlen(dataDir)-1] == '\\') | 
|---|
| [3655] | 103 |     { | 
|---|
| [5480] | 104 |       this->dataDir = new char[strlen(realDir)+1]; | 
|---|
 | 105 |       strcpy(this->dataDir, realDir); | 
|---|
| [3655] | 106 |     } | 
|---|
| [5480] | 107 |     else | 
|---|
 | 108 |     { | 
|---|
 | 109 |       this->dataDir = new char[strlen(realDir)+2]; | 
|---|
 | 110 |       strcpy(this->dataDir, realDir); | 
|---|
 | 111 |       this->dataDir[strlen(realDir)] = '/'; | 
|---|
 | 112 |       this->dataDir[strlen(realDir)+1] = '\0'; | 
|---|
 | 113 |     } | 
|---|
 | 114 |     delete[] realDir; | 
|---|
 | 115 |     return true; | 
|---|
 | 116 |   } | 
|---|
| [3655] | 117 |   else | 
|---|
| [5480] | 118 |   { | 
|---|
 | 119 |     PRINTF(1)("%s is not a Directory, and can not be the Data Directory, leaving as %s \n", realDir, this->dataDir); | 
|---|
 | 120 |     delete[] realDir; | 
|---|
 | 121 |     return false; | 
|---|
 | 122 |   } | 
|---|
 | 123 | } | 
|---|
 | 124 |  | 
|---|
 | 125 | /** | 
|---|
| [6647] | 126 |  * @brief sets the data main directory | 
|---|
| [5480] | 127 |  * @param dataDir the DataDirectory. | 
|---|
 | 128 |  * | 
|---|
 | 129 |  * this is essentially the same as setDataDir, but it ommits the error-message | 
|---|
 | 130 |  */ | 
|---|
 | 131 | bool ResourceManager::tryDataDir(const char* dataDir) | 
|---|
 | 132 | { | 
|---|
 | 133 |   char* realDir = ResourceManager::homeDirCheck(dataDir); | 
|---|
 | 134 |   if (isDir(realDir)) | 
|---|
 | 135 |   { | 
|---|
 | 136 |     delete[] this->dataDir; | 
|---|
 | 137 |     if (dataDir[strlen(dataDir)-1] == '/' || dataDir[strlen(dataDir)-1] == '\\') | 
|---|
| [3655] | 138 |     { | 
|---|
| [5480] | 139 |       this->dataDir = new char[strlen(realDir)+1]; | 
|---|
 | 140 |       strcpy(this->dataDir, realDir); | 
|---|
| [3655] | 141 |     } | 
|---|
| [5480] | 142 |     else | 
|---|
 | 143 |     { | 
|---|
 | 144 |       this->dataDir = new char[strlen(realDir)+2]; | 
|---|
 | 145 |       strcpy(this->dataDir, realDir); | 
|---|
 | 146 |       this->dataDir[strlen(realDir)] = '/'; | 
|---|
 | 147 |       this->dataDir[strlen(realDir)+1] = '\0'; | 
|---|
 | 148 |     } | 
|---|
 | 149 |     delete[] realDir; | 
|---|
 | 150 |     return true; | 
|---|
 | 151 |   } | 
|---|
| [5483] | 152 |   delete[] realDir; | 
|---|
| [5482] | 153 |   return false; | 
|---|
| [3543] | 154 | } | 
|---|
| [1853] | 155 |  | 
|---|
| [5480] | 156 |  | 
|---|
| [3660] | 157 | /** | 
|---|
| [6647] | 158 |  * @brief checks for the DataDirectory, by looking if | 
|---|
| [5480] | 159 |  * @param fileInside is iniside of the given directory. | 
|---|
| [4091] | 160 | */ | 
|---|
| [5480] | 161 | bool ResourceManager::verifyDataDir(const char* fileInside) | 
|---|
| [4091] | 162 | { | 
|---|
 | 163 |   bool retVal; | 
|---|
 | 164 |   if (!isDir(this->dataDir)) | 
|---|
| [6640] | 165 |   { | 
|---|
 | 166 |     PRINTF(1)("%s is not a directory\n", this->dataDir); | 
|---|
 | 167 |     return false; | 
|---|
 | 168 |   } | 
|---|
| [4597] | 169 |  | 
|---|
| [4091] | 170 |   char* testFile = new char[strlen(this->dataDir)+strlen(fileInside)+1]; | 
|---|
 | 171 |   sprintf(testFile, "%s%s", this->dataDir, fileInside); | 
|---|
 | 172 |   retVal = isFile(testFile); | 
|---|
| [5208] | 173 |   delete[] testFile; | 
|---|
| [4091] | 174 |   return retVal; | 
|---|
 | 175 | } | 
|---|
 | 176 |  | 
|---|
| [4653] | 177 | #ifndef NO_TEXTURES | 
|---|
| [4091] | 178 | /** | 
|---|
| [6647] | 179 |  * @brief adds a new Path for Images | 
|---|
| [4836] | 180 |  * @param imageDir The path to insert | 
|---|
 | 181 |  * @returns true, if the Path was well and injected (or already existent within the list) | 
|---|
| [3660] | 182 |    false otherwise | 
|---|
 | 183 | */ | 
|---|
| [4370] | 184 | bool ResourceManager::addImageDir(const char* imageDir) | 
|---|
| [3658] | 185 | { | 
|---|
| [5335] | 186 |   if (imageDir == NULL) | 
|---|
 | 187 |     return false; | 
|---|
 | 188 |  | 
|---|
 | 189 |   char* newDir; | 
|---|
 | 190 |   if (imageDir[strlen(imageDir)-1] == '/' || imageDir[strlen(imageDir)-1] == '\\') | 
|---|
 | 191 |   { | 
|---|
 | 192 |     newDir = new char[strlen(imageDir)+1]; | 
|---|
 | 193 |     strcpy(newDir, imageDir); | 
|---|
 | 194 |   } | 
|---|
 | 195 |   else | 
|---|
 | 196 |   { | 
|---|
 | 197 |     newDir = new char[strlen(imageDir)+2]; | 
|---|
 | 198 |     strcpy(newDir, imageDir); | 
|---|
 | 199 |     newDir[strlen(imageDir)] = '/'; | 
|---|
 | 200 |     newDir[strlen(imageDir)+1] = '\0'; | 
|---|
 | 201 |   } | 
|---|
| [3660] | 202 |   // check if the param is a Directory | 
|---|
| [5335] | 203 |   if (isDir(newDir)) | 
|---|
| [6640] | 204 |   { | 
|---|
 | 205 |     // check if the Directory has been added before | 
|---|
| [6642] | 206 |     std::vector<char*>::const_iterator imageDir; | 
|---|
| [6640] | 207 |     for (imageDir = this->imageDirs.begin(); imageDir != this->imageDirs.end(); imageDir++) | 
|---|
| [3658] | 208 |     { | 
|---|
| [6640] | 209 |       if (!strcmp(*imageDir, newDir)) | 
|---|
| [6222] | 210 |       { | 
|---|
| [6640] | 211 |         PRINTF(3)("Path %s already loaded\n", newDir); | 
|---|
 | 212 |         delete[] newDir; | 
|---|
 | 213 |         return true; | 
|---|
| [6222] | 214 |       } | 
|---|
| [3658] | 215 |     } | 
|---|
| [6640] | 216 |     // adding the directory to the List | 
|---|
 | 217 |     this->imageDirs.push_back(newDir); | 
|---|
 | 218 |     return true; | 
|---|
 | 219 |   } | 
|---|
| [3658] | 220 |   else | 
|---|
| [6640] | 221 |   { | 
|---|
 | 222 |     PRINTF(1)("%s is not a Directory, and can not be added to the Paths of Images\n", newDir); | 
|---|
 | 223 |     delete[] newDir; | 
|---|
 | 224 |     return false; | 
|---|
 | 225 |   } | 
|---|
| [3658] | 226 | } | 
|---|
| [4534] | 227 | #endif /* NO_TEXTURES */ | 
|---|
| [3658] | 228 |  | 
|---|
| [3245] | 229 | /** | 
|---|
| [6647] | 230 |  * @brief loads resources | 
|---|
| [4836] | 231 |  * @param fileName: The fileName of the resource to load | 
|---|
 | 232 |  * @param prio: The ResourcePriority of this resource (will only be increased) | 
|---|
| [6645] | 233 |  * @param param0: an additional option to parse (see the constuctors for more help) | 
|---|
| [4836] | 234 |  * @param param1: an additional option to parse (see the constuctors for more help) | 
|---|
 | 235 |  * @param param2: an additional option to parse (see the constuctors for more help) | 
|---|
 | 236 |  * @returns a pointer to a desired Resource. | 
|---|
| [3655] | 237 | */ | 
|---|
| [6645] | 238 | BaseObject* ResourceManager::load(const char* fileName, ResourcePriority prio, | 
|---|
| [6648] | 239 |                                   const MultiType& param0, const MultiType& param1, const MultiType& param2) | 
|---|
| [3655] | 240 | { | 
|---|
| [5366] | 241 |   if (fileName == NULL) | 
|---|
 | 242 |     return NULL; | 
|---|
| [3655] | 243 |   ResourceType tmpType; | 
|---|
| [4534] | 244 | #ifndef NO_MODEL | 
|---|
| [4637] | 245 | #define __IF_OK | 
|---|
| [5323] | 246 |   if (!strncasecmp(fileName+(strlen(fileName)-4), ".obj", 4)) | 
|---|
| [3655] | 247 |     tmpType = OBJ; | 
|---|
| [4534] | 248 |   else if (!strncmp(fileName+(strlen(fileName)-4), ".md2", 4)) | 
|---|
| [4462] | 249 |     tmpType = MD2; | 
|---|
| [5323] | 250 |   else if (!strcasecmp(fileName, "cube") || | 
|---|
 | 251 |            !strcasecmp(fileName, "sphere") || | 
|---|
 | 252 |            !strcasecmp(fileName, "plane") || | 
|---|
 | 253 |            !strcasecmp(fileName, "cylinder") || | 
|---|
 | 254 |            !strcasecmp(fileName, "cone")) | 
|---|
| [4534] | 255 |     tmpType = PRIM; | 
|---|
 | 256 | #endif /* NO_MODEL */ | 
|---|
 | 257 | #ifndef NO_AUDIO | 
|---|
| [4637] | 258 | #ifdef __IF_OK | 
|---|
 | 259 |   else | 
|---|
 | 260 | #endif | 
|---|
 | 261 | #define __IF_OK | 
|---|
| [6640] | 262 |     if (!strncasecmp(fileName+(strlen(fileName)-4), ".wav", 4)) | 
|---|
 | 263 |       tmpType = WAV; | 
|---|
 | 264 |     else if (!strncasecmp(fileName+(strlen(fileName)-4), ".mp3", 4)) | 
|---|
 | 265 |       tmpType = MP3; | 
|---|
 | 266 |     else if (!strncasecmp(fileName+(strlen(fileName)-4), ".ogg", 4)) | 
|---|
 | 267 |       tmpType = OGG; | 
|---|
| [4534] | 268 | #endif /* NO_AUDIO */ | 
|---|
 | 269 | #ifndef NO_TEXT | 
|---|
| [4637] | 270 | #ifdef __IF_OK | 
|---|
| [6640] | 271 |     else | 
|---|
| [4637] | 272 | #endif | 
|---|
 | 273 | #define __IF_OK | 
|---|
| [6640] | 274 |       if (!strncasecmp(fileName+(strlen(fileName)-4), ".ttf", 4)) | 
|---|
 | 275 |         tmpType = TTF; | 
|---|
| [4534] | 276 | #endif /* NO_TEXT */ | 
|---|
| [5323] | 277 | #ifndef NO_SHADERS | 
|---|
 | 278 | #ifdef __IF_OK | 
|---|
| [6640] | 279 |       else | 
|---|
| [5323] | 280 | #endif | 
|---|
 | 281 | #define __IF_OK | 
|---|
| [6640] | 282 |         if (!strncasecmp(fileName+(strlen(fileName)-5), ".vert", 5)) | 
|---|
 | 283 |           tmpType = SHADER; | 
|---|
| [5323] | 284 | #endif /* NO_SHADERS */ | 
|---|
| [4534] | 285 | #ifndef NO_TEXTURES | 
|---|
| [4637] | 286 | #ifdef __IF_OK | 
|---|
| [6640] | 287 |         else | 
|---|
| [4637] | 288 | #else | 
|---|
 | 289 |   if | 
|---|
 | 290 | #endif | 
|---|
| [6640] | 291 |           tmpType = IMAGE; | 
|---|
| [4534] | 292 | #endif /* NO_TEXTURES */ | 
|---|
| [4653] | 293 | #undef __IF_OK | 
|---|
| [6645] | 294 |   return this->load(fileName, tmpType, prio, param0, param1, param2); | 
|---|
| [3655] | 295 | } | 
|---|
 | 296 |  | 
|---|
 | 297 | /** | 
|---|
| [6647] | 298 |  * @brief caches a Resource | 
|---|
| [6641] | 299 |  * | 
|---|
 | 300 |  * @see load; | 
|---|
 | 301 |  * | 
|---|
| [6650] | 302 |  * @brief returns true if ok, false otherwise. | 
|---|
| [6641] | 303 |  * This function loads a Resource without applying it to an Object. | 
|---|
 | 304 |  * This is for loading purposes, e.g, when the user is loading a Resource | 
|---|
 | 305 |  * during the initialisation instead of at Runtime. | 
|---|
 | 306 |  */ | 
|---|
| [6650] | 307 | bool ResourceManager::cache(const char* fileName, ResourceType type, ResourcePriority prio, | 
|---|
| [6645] | 308 |                             const MultiType& param0, const MultiType& param1, const MultiType& param2) | 
|---|
| [6641] | 309 | { | 
|---|
 | 310 |   assert(fileName != NULL); | 
|---|
 | 311 |  | 
|---|
 | 312 |   // searching if the resource was loaded before. | 
|---|
 | 313 |   Resource* tmpResource; | 
|---|
 | 314 |   // check if we already loaded this Resource | 
|---|
| [6645] | 315 |   tmpResource = this->locateResourceByInfo(fileName, type, param0, param1, param2); | 
|---|
| [6641] | 316 |   // otherwise load it | 
|---|
 | 317 |   if (tmpResource == NULL) | 
|---|
| [6645] | 318 |     tmpResource = this->loadResource(fileName, type, prio, param0, param1, param2); | 
|---|
| [6641] | 319 |   // return cached pointer. | 
|---|
 | 320 |   if (tmpResource != NULL) // if the resource was loaded before. | 
|---|
| [6650] | 321 |   { | 
|---|
| [6641] | 322 |     if(tmpResource->prio < prio) | 
|---|
 | 323 |       tmpResource->prio = prio; | 
|---|
| [6650] | 324 |     return true; | 
|---|
 | 325 |   } | 
|---|
 | 326 |   else | 
|---|
 | 327 |     return false; | 
|---|
| [6641] | 328 | } | 
|---|
 | 329 |  | 
|---|
| [6651] | 330 | /** | 
|---|
 | 331 |  * tells the ResourceManager to generate a Copy of the Resource. | 
|---|
 | 332 |  * @brief resourcePointer: The Pointer to the resource to copy | 
|---|
 | 333 |  * @returns the Resource pointed to resourcePointer. | 
|---|
 | 334 |  */ | 
|---|
 | 335 | BaseObject* ResourceManager::copy(BaseObject* resourcePointer) | 
|---|
 | 336 | { | 
|---|
 | 337 |   Resource* tmp = locateResourceByPointer(resourcePointer); | 
|---|
 | 338 |   if (tmp!=NULL) | 
|---|
 | 339 |   { | 
|---|
 | 340 |     tmp->count++; | 
|---|
 | 341 |     return tmp->pointer; | 
|---|
 | 342 |   } | 
|---|
 | 343 |   else | 
|---|
 | 344 |     return NULL; | 
|---|
 | 345 | } | 
|---|
| [6641] | 346 |  | 
|---|
| [6651] | 347 |  | 
|---|
| [6641] | 348 | /** | 
|---|
| [6647] | 349 |  * @brief loads resources | 
|---|
| [4836] | 350 |  * @param fileName: The fileName of the resource to load | 
|---|
| [5306] | 351 |  * @param type: The Type of Resource to load. | 
|---|
| [4836] | 352 |  * @param prio: The ResourcePriority of this resource (will only be increased) | 
|---|
| [6645] | 353 |  * @param param0: an additional option to parse (see the constuctors for more help) | 
|---|
| [4836] | 354 |  * @param param1: an additional option to parse (see the constuctors for more help) | 
|---|
 | 355 |  * @param param2: an additional option to parse (see the constuctors for more help) | 
|---|
 | 356 |  * @returns a pointer to a desired Resource. | 
|---|
| [3245] | 357 | */ | 
|---|
| [5304] | 358 | BaseObject* ResourceManager::load(const char* fileName, ResourceType type, ResourcePriority prio, | 
|---|
| [6645] | 359 |                                   const MultiType& param0, const MultiType& param1, const MultiType& param2) | 
|---|
| [3655] | 360 | { | 
|---|
| [6640] | 361 |   assert(fileName != NULL); | 
|---|
| [5366] | 362 |  | 
|---|
| [3658] | 363 |   // searching if the resource was loaded before. | 
|---|
| [6640] | 364 |   Resource* tmpResource; | 
|---|
 | 365 |   // check if we already loaded this Resource | 
|---|
| [6645] | 366 |   tmpResource = this->locateResourceByInfo(fileName, type, param0, param1, param2); | 
|---|
| [6640] | 367 |   // otherwise load it | 
|---|
 | 368 |   if (tmpResource == NULL) | 
|---|
| [6648] | 369 |   { | 
|---|
| [6645] | 370 |     tmpResource = this->loadResource(fileName, type, prio, param0, param1, param2); | 
|---|
| [6648] | 371 |   } | 
|---|
| [6640] | 372 |   // return cached pointer. | 
|---|
| [5306] | 373 |   if (tmpResource != NULL) // if the resource was loaded before. | 
|---|
| [6640] | 374 |   { | 
|---|
 | 375 |     tmpResource->count++; | 
|---|
 | 376 |     if(tmpResource->prio < prio) | 
|---|
 | 377 |       tmpResource->prio = prio; | 
|---|
| [6648] | 378 |  | 
|---|
| [6640] | 379 |     return tmpResource->pointer; | 
|---|
 | 380 |   } | 
|---|
| [3677] | 381 |   else | 
|---|
| [6640] | 382 |     return NULL; | 
|---|
 | 383 | } | 
|---|
| [3658] | 384 |  | 
|---|
| [6640] | 385 |  | 
|---|
 | 386 | /** | 
|---|
| [6647] | 387 |  * @brief loads resources for internal purposes | 
|---|
| [6640] | 388 |  * @param fileName: The fileName of the resource to load | 
|---|
 | 389 |  * @param type: The Type of Resource to load. | 
|---|
 | 390 |  * @param prio: The ResourcePriority of this resource (will only be increased) | 
|---|
| [6645] | 391 |  * @param param0: an additional option to parse (see the constuctors for more help) | 
|---|
| [6640] | 392 |  * @param param1: an additional option to parse (see the constuctors for more help) | 
|---|
 | 393 |  * @param param2: an additional option to parse (see the constuctors for more help) | 
|---|
 | 394 |  * @returns a pointer to a desired Resource. | 
|---|
 | 395 |  */ | 
|---|
 | 396 | Resource* ResourceManager::loadResource(const char* fileName, ResourceType type, ResourcePriority prio, | 
|---|
| [6645] | 397 |                                         const MultiType& param0, const MultiType& param1, const MultiType& param2) | 
|---|
| [6640] | 398 | { | 
|---|
 | 399 |   // Setting up the new Resource | 
|---|
 | 400 |   Resource* tmpResource = new Resource; | 
|---|
 | 401 |   tmpResource->count = 0; | 
|---|
 | 402 |   tmpResource->type = type; | 
|---|
 | 403 |   tmpResource->prio = prio; | 
|---|
 | 404 |   tmpResource->pointer = NULL; | 
|---|
 | 405 |   tmpResource->name = new char[strlen(fileName)+1]; | 
|---|
 | 406 |   strcpy(tmpResource->name, fileName); | 
|---|
 | 407 |  | 
|---|
 | 408 |   // creating the full name. (directoryName + FileName) | 
|---|
 | 409 |   char* fullName = ResourceManager::getFullName(fileName); | 
|---|
 | 410 |   // Checking for the type of resource \see ResourceType | 
|---|
 | 411 |   switch(type) | 
|---|
 | 412 |   { | 
|---|
| [4534] | 413 | #ifndef NO_MODEL | 
|---|
| [6648] | 414 |     case OBJ: | 
|---|
 | 415 |       if (param0.getType() != MT_NULL) | 
|---|
 | 416 |         tmpResource->param[0] = param0; | 
|---|
 | 417 |       else | 
|---|
 | 418 |         tmpResource->param[0] = 1.0f; | 
|---|
| [3790] | 419 |  | 
|---|
| [6648] | 420 |       if(ResourceManager::isFile(fullName)) | 
|---|
 | 421 |         tmpResource->pointer = new OBJModel(fullName, tmpResource->param[0].getFloat()); | 
|---|
 | 422 |       else | 
|---|
 | 423 |       { | 
|---|
 | 424 |         PRINTF(2)("File %s in %s does not exist. Loading a cube-Model instead\n", fileName, dataDir); | 
|---|
 | 425 |         tmpResource->pointer = ResourceManager::load("cube", PRIM, prio, tmpResource->param[0].getFloat()); | 
|---|
 | 426 |       } | 
|---|
 | 427 |       break; | 
|---|
 | 428 |     case PRIM: | 
|---|
 | 429 |       if (param0 != MT_NULL) | 
|---|
 | 430 |         tmpResource->param[0] = param0; | 
|---|
 | 431 |       else | 
|---|
 | 432 |         tmpResource->param[0] = 1.0f; | 
|---|
| [3790] | 433 |  | 
|---|
| [6648] | 434 |       if (!strcmp(tmpResource->name, "cube")) | 
|---|
 | 435 |         tmpResource->pointer = new PrimitiveModel(PRIM_CUBE, tmpResource->param[0].getFloat()); | 
|---|
 | 436 |       else if (!strcmp(tmpResource->name, "sphere")) | 
|---|
 | 437 |         tmpResource->pointer = new PrimitiveModel(PRIM_SPHERE, tmpResource->param[0].getFloat()); | 
|---|
 | 438 |       else if (!strcmp(tmpResource->name, "plane")) | 
|---|
 | 439 |         tmpResource->pointer = new PrimitiveModel(PRIM_PLANE, tmpResource->param[0].getFloat()); | 
|---|
 | 440 |       else if (!strcmp(tmpResource->name, "cylinder")) | 
|---|
 | 441 |         tmpResource->pointer = new PrimitiveModel(PRIM_CYLINDER, tmpResource->param[0].getFloat()); | 
|---|
 | 442 |       else if (!strcmp(tmpResource->name, "cone")) | 
|---|
 | 443 |         tmpResource->pointer = new PrimitiveModel(PRIM_CONE, tmpResource->param[0].getFloat()); | 
|---|
 | 444 |       break; | 
|---|
 | 445 |     case MD2: | 
|---|
 | 446 |       if(ResourceManager::isFile(fullName)) | 
|---|
 | 447 |       { | 
|---|
 | 448 |         tmpResource->param[0] = param0; | 
|---|
 | 449 |         tmpResource->pointer = new MD2Data(fullName, tmpResource->param[0].getString()); | 
|---|
 | 450 |         //               tmpResource->pointer = new MD2Model(fullName, tmpResource->secFileName); | 
|---|
| [6222] | 451 |  | 
|---|
| [6648] | 452 |       } | 
|---|
 | 453 |       break; | 
|---|
| [4534] | 454 | #endif /* NO_MODEL */ | 
|---|
 | 455 | #ifndef NO_TEXT | 
|---|
| [6648] | 456 |     case TTF: | 
|---|
 | 457 |       if (param0 != MT_NULL) | 
|---|
 | 458 |       { | 
|---|
 | 459 |         assert(param0.getInt() >= 0); | 
|---|
 | 460 |         tmpResource->param[0] = param0; | 
|---|
 | 461 |       } | 
|---|
 | 462 |       else | 
|---|
 | 463 |         tmpResource->param[0] = FONT_DEFAULT_RENDER_SIZE; | 
|---|
| [4597] | 464 |  | 
|---|
| [6648] | 465 |       if(isFile(fullName)) | 
|---|
 | 466 |         tmpResource->pointer = new Font(fullName, (unsigned int) tmpResource->param[0].getInt()); | 
|---|
 | 467 |       else | 
|---|
 | 468 |         PRINTF(2)("%s does not exist in %s. Not loading Font\n", fileName, this->dataDir); | 
|---|
 | 469 |       break; | 
|---|
| [4534] | 470 | #endif /* NO_TEXT */ | 
|---|
 | 471 | #ifndef NO_AUDIO | 
|---|
| [6648] | 472 |     case WAV: | 
|---|
 | 473 |       if(isFile(fullName)) | 
|---|
 | 474 |         tmpResource->pointer = new SoundBuffer(fullName); | 
|---|
 | 475 |       break; | 
|---|
 | 476 |     case OGG: | 
|---|
 | 477 |       if (isFile(fullName)) | 
|---|
 | 478 |         tmpResource->pointer = new OggPlayer(fullName); | 
|---|
 | 479 |       break; | 
|---|
| [4534] | 480 | #endif /* NO_AUDIO */ | 
|---|
 | 481 | #ifndef NO_TEXTURES | 
|---|
| [6648] | 482 |     case IMAGE: | 
|---|
 | 483 |       if (param0 != MT_NULL) | 
|---|
 | 484 |         tmpResource->param[0] = param0; | 
|---|
 | 485 |       else | 
|---|
 | 486 |         tmpResource->param[0] = GL_TEXTURE_2D; | 
|---|
 | 487 |       if(isFile(fullName)) | 
|---|
| [6640] | 488 |       { | 
|---|
| [6648] | 489 |         PRINTF(4)("Image %s resides to %s\n", fileName, fullName); | 
|---|
 | 490 |         tmpResource->pointer = new Texture(fullName); | 
|---|
 | 491 |       } | 
|---|
 | 492 |       else | 
|---|
 | 493 |       { | 
|---|
 | 494 |         std::vector<char*>::iterator imageDir; | 
|---|
 | 495 |         for (imageDir = this->imageDirs.begin(); imageDir != this->imageDirs.end(); imageDir++) | 
|---|
| [6640] | 496 |         { | 
|---|
| [6648] | 497 |           char* imgName = new char[strlen(*imageDir)+strlen(fileName)+1]; | 
|---|
 | 498 |           sprintf(imgName, "%s%s", *imageDir, fileName); | 
|---|
 | 499 |           if(isFile(imgName)) | 
|---|
 | 500 |           { | 
|---|
 | 501 |             PRINTF(4)("Image %s resides to %s\n", fileName, imgName); | 
|---|
 | 502 |             tmpResource->pointer = new Texture(imgName, tmpResource->param[0].getInt()); | 
|---|
 | 503 |             delete[] imgName; | 
|---|
 | 504 |             break; | 
|---|
 | 505 |           } | 
|---|
| [6640] | 506 |           delete[] imgName; | 
|---|
 | 507 |         } | 
|---|
 | 508 |       } | 
|---|
| [6648] | 509 |       if(!tmpResource) | 
|---|
 | 510 |         PRINTF(2)("!!Image %s not Found!!\n", fileName); | 
|---|
 | 511 |       break; | 
|---|
| [4534] | 512 | #endif /* NO_TEXTURES */ | 
|---|
| [5323] | 513 | #ifndef NO_SHADERS | 
|---|
| [6648] | 514 |     case SHADER: | 
|---|
 | 515 |       if(ResourceManager::isFile(fullName)) | 
|---|
| [6640] | 516 |       { | 
|---|
| [6648] | 517 |         if (param0 != MT_NULL) | 
|---|
| [6640] | 518 |         { | 
|---|
| [6648] | 519 |           MultiType param = param0; /// HACK | 
|---|
 | 520 |           char* secFullName = ResourceManager::getFullName(param.getString()); | 
|---|
 | 521 |           if (ResourceManager::isFile(secFullName)) | 
|---|
 | 522 |           { | 
|---|
 | 523 |             tmpResource->param[0] = secFullName; | 
|---|
 | 524 |             tmpResource->pointer = new Shader(fullName, secFullName); | 
|---|
 | 525 |           } | 
|---|
 | 526 |           delete[] secFullName; | 
|---|
| [4597] | 527 |         } | 
|---|
| [6648] | 528 |         else | 
|---|
 | 529 |         { | 
|---|
 | 530 |           tmpResource->param[0] = param0; | 
|---|
 | 531 |           tmpResource->pointer = new Shader(fullName, NULL); | 
|---|
 | 532 |         } | 
|---|
| [6640] | 533 |       } | 
|---|
| [6648] | 534 |       break; | 
|---|
| [6640] | 535 | #endif /* NO_SHADERS */ | 
|---|
| [6648] | 536 |     default: | 
|---|
 | 537 |       tmpResource->pointer = NULL; | 
|---|
 | 538 |       PRINTF(1)("No type found for %s.\n   !!This should not happen unless the Type is not supported yet. JUST DO IT!!\n", tmpResource->name); | 
|---|
 | 539 |       break; | 
|---|
| [6640] | 540 |   } | 
|---|
| [5216] | 541 |   if (tmpResource->pointer != NULL) | 
|---|
| [6640] | 542 |     this->resourceList.push_back(tmpResource); | 
|---|
 | 543 |   delete[] fullName; | 
|---|
 | 544 |  | 
|---|
 | 545 |  | 
|---|
 | 546 |   if (tmpResource->pointer != NULL) | 
|---|
 | 547 |     return tmpResource; | 
|---|
| [4597] | 548 |   else | 
|---|
| [6640] | 549 |   { | 
|---|
 | 550 |     PRINTF(2)("Resource %s could not be loaded\n", fileName); | 
|---|
 | 551 |     delete[] tmpResource->name; | 
|---|
 | 552 |     delete tmpResource; | 
|---|
 | 553 |     return NULL; | 
|---|
 | 554 |   } | 
|---|
| [3658] | 555 | } | 
|---|
 | 556 |  | 
|---|
 | 557 | /** | 
|---|
| [6647] | 558 |  * @brief unloads a Resource | 
|---|
| [4836] | 559 |  * @param pointer: The pointer to free | 
|---|
 | 560 |  * @param prio: the PriorityLevel to unload this resource | 
|---|
 | 561 |  * @returns true if successful (pointer found, and deleted), false otherwise | 
|---|
| [3658] | 562 | */ | 
|---|
| [6651] | 563 | bool ResourceManager::unload(BaseObject* pointer, ResourcePriority prio) | 
|---|
| [3658] | 564 | { | 
|---|
| [5366] | 565 |   if (pointer == NULL) | 
|---|
 | 566 |     return false; | 
|---|
| [3658] | 567 |   // if pointer is existent. and only one resource of this type exists. | 
|---|
| [3672] | 568 |   Resource* tmpResource = this->locateResourceByPointer(pointer); | 
|---|
| [5366] | 569 |   if (tmpResource != NULL) | 
|---|
 | 570 |     return unload(tmpResource, prio); | 
|---|
| [3660] | 571 |   else | 
|---|
| [5366] | 572 |   { | 
|---|
 | 573 |     PRINTF(2)("Resource not Found %p\n", pointer); | 
|---|
 | 574 |     return false; | 
|---|
 | 575 |   } | 
|---|
| [3660] | 576 | } | 
|---|
 | 577 |  | 
|---|
| [4465] | 578 | /** | 
|---|
| [6647] | 579 |  * @brief unloads a Resource | 
|---|
| [4836] | 580 |  * @param resource: The resource to unloade | 
|---|
 | 581 |  * @param prio the PriorityLevel to unload this resource | 
|---|
| [5308] | 582 |  * @returns true on success, false otherwise. | 
|---|
| [4465] | 583 | */ | 
|---|
| [3660] | 584 | bool ResourceManager::unload(Resource* resource, ResourcePriority prio) | 
|---|
 | 585 | { | 
|---|
| [5306] | 586 |   if (resource == NULL) | 
|---|
 | 587 |     return false; | 
|---|
| [3665] | 588 |   if (resource->count > 0) | 
|---|
 | 589 |     resource->count--; | 
|---|
| [5306] | 590 |  | 
|---|
| [3660] | 591 |   if (resource->prio <= prio) | 
|---|
| [6640] | 592 |   { | 
|---|
 | 593 |     if (resource->count == 0) | 
|---|
| [3658] | 594 |     { | 
|---|
| [6640] | 595 |       // deleting the Resource | 
|---|
 | 596 |       switch(resource->type) | 
|---|
 | 597 |       { | 
|---|
| [4534] | 598 | #ifndef NO_MODEL | 
|---|
| [6648] | 599 |         case OBJ: | 
|---|
 | 600 |         case PRIM: | 
|---|
 | 601 |           delete (Model*)resource->pointer; | 
|---|
 | 602 |           break; | 
|---|
 | 603 |         case MD2: | 
|---|
 | 604 |           delete (MD2Data*)resource->pointer; | 
|---|
 | 605 |           break; | 
|---|
| [4534] | 606 | #endif /* NO_MODEL */ | 
|---|
 | 607 | #ifndef NO_AUDIO | 
|---|
| [6648] | 608 |         case WAV: | 
|---|
 | 609 |           delete (SoundBuffer*)resource->pointer; | 
|---|
 | 610 |           break; | 
|---|
 | 611 |         case OGG: | 
|---|
 | 612 |           delete (OggPlayer*)resource->pointer; | 
|---|
 | 613 |           break; | 
|---|
| [4534] | 614 | #endif /* NO_AUDIO */ | 
|---|
 | 615 | #ifndef NO_TEXT | 
|---|
| [6648] | 616 |         case TTF: | 
|---|
 | 617 |           delete (Font*)resource->pointer; | 
|---|
 | 618 |           break; | 
|---|
| [4534] | 619 | #endif /* NO_TEXT */ | 
|---|
 | 620 | #ifndef NO_TEXTURES | 
|---|
| [6648] | 621 |         case IMAGE: | 
|---|
 | 622 |           delete (Texture*)resource->pointer; | 
|---|
 | 623 |           break; | 
|---|
| [4534] | 624 | #endif /* NO_TEXTURES */ | 
|---|
| [5323] | 625 | #ifndef NO_SHADERS | 
|---|
| [6648] | 626 |         case SHADER: | 
|---|
 | 627 |           delete (Shader*)resource->pointer; | 
|---|
 | 628 |           break; | 
|---|
| [5323] | 629 | #endif /* NO_SHADERS */ | 
|---|
| [6648] | 630 |         default: | 
|---|
 | 631 |           PRINTF(2)("NOT YET IMPLEMENTED !!FIX FIX!!\n"); | 
|---|
 | 632 |           return false; | 
|---|
 | 633 |           break; | 
|---|
| [6640] | 634 |       } | 
|---|
 | 635 |       // deleting the List Entry: | 
|---|
 | 636 |       PRINTF(4)("Resource %s safely removed.\n", resource->name); | 
|---|
 | 637 |       delete[] resource->name; | 
|---|
| [6642] | 638 |       std::vector<Resource*>::iterator resourceIT = std::find(this->resourceList.begin(), this->resourceList.end(), resource); | 
|---|
| [6640] | 639 |       this->resourceList.erase(resourceIT); | 
|---|
 | 640 |       delete resource; | 
|---|
| [3658] | 641 |     } | 
|---|
| [6640] | 642 |     else | 
|---|
 | 643 |       PRINTF(4)("Resource %s not removed, because there are still %d References to it.\n", resource->name, resource->count); | 
|---|
 | 644 |   } | 
|---|
| [3658] | 645 |   else | 
|---|
| [3660] | 646 |     PRINTF(4)("not deleting resource %s because DeleteLevel to high\n", resource->name); | 
|---|
| [3658] | 647 |   return true; | 
|---|
| [3655] | 648 | } | 
|---|
 | 649 |  | 
|---|
| [3660] | 650 |  | 
|---|
| [3655] | 651 | /** | 
|---|
| [6647] | 652 |  * @brief unloads all alocated Memory of Resources with a pririty lower than prio | 
|---|
| [4836] | 653 |  * @param prio The priority to delete | 
|---|
| [3660] | 654 | */ | 
|---|
 | 655 | bool ResourceManager::unloadAllByPriority(ResourcePriority prio) | 
|---|
 | 656 | { | 
|---|
| [6222] | 657 |   unsigned int removeCount; | 
|---|
 | 658 |   for (unsigned int round = 0; round < 3; round++) | 
|---|
 | 659 |   { | 
|---|
| [6642] | 660 |     int index = this->resourceList.size() - 1; | 
|---|
| [6222] | 661 |     removeCount = 0; | 
|---|
| [6642] | 662 |     while (index >= 0) | 
|---|
| [3660] | 663 |     { | 
|---|
| [6642] | 664 |       if (this->resourceList[index]->prio <= prio) | 
|---|
| [6640] | 665 |       { | 
|---|
| [6642] | 666 |         if (this->resourceList[index]->count == 0) | 
|---|
 | 667 |           unload(this->resourceList[index], prio); | 
|---|
| [6640] | 668 |         else | 
|---|
| [6222] | 669 |         { | 
|---|
| [6664] | 670 |           if (round == 3) | 
|---|
 | 671 |             PRINTF(2)("unable to unload %s because there are still %d references to it\n", | 
|---|
 | 672 |                       this->resourceList[index]->name, this->resourceList[index]->count); | 
|---|
| [6640] | 673 |           removeCount++; | 
|---|
| [6222] | 674 |         } | 
|---|
| [6640] | 675 |       } | 
|---|
| [6642] | 676 |       index--; | 
|---|
| [6640] | 677 |     } | 
|---|
 | 678 |     if (removeCount == 0) break; | 
|---|
| [6222] | 679 |   } | 
|---|
| [3660] | 680 | } | 
|---|
 | 681 |  | 
|---|
| [5994] | 682 |  | 
|---|
| [3660] | 683 | /** | 
|---|
| [6647] | 684 |  * @brief Searches for a Resource by some information | 
|---|
| [4836] | 685 |  * @param fileName: The name to look for | 
|---|
 | 686 |  * @param type the Type of resource to locate. | 
|---|
| [6645] | 687 |  * @param param0: an additional option to parse (see the constuctors for more help) | 
|---|
| [4836] | 688 |  * @param param1: an additional option to parse (see the constuctors for more help) | 
|---|
 | 689 |  * @param param2: an additional option to parse (see the constuctors for more help) | 
|---|
 | 690 |  * @returns a Pointer to the Resource if found, NULL otherwise. | 
|---|
| [3658] | 691 | */ | 
|---|
| [4462] | 692 | Resource* ResourceManager::locateResourceByInfo(const char* fileName, ResourceType type, | 
|---|
| [6648] | 693 |     const MultiType& param0, const MultiType& param1, const MultiType& param2) const | 
|---|
| [3658] | 694 | { | 
|---|
| [6642] | 695 |   std::vector<Resource*>::const_iterator resource; | 
|---|
| [6222] | 696 |   for (resource = this->resourceList.begin(); resource != this->resourceList.end(); resource++) | 
|---|
 | 697 |   { | 
|---|
 | 698 |     if ((*resource)->type == type && !strcmp(fileName, (*resource)->name)) | 
|---|
| [6640] | 699 |     { | 
|---|
 | 700 |       bool match = false; | 
|---|
 | 701 |       switch (type) | 
|---|
 | 702 |       { | 
|---|
| [4534] | 703 | #ifndef NO_MODEL | 
|---|
| [6648] | 704 |         case PRIM: | 
|---|
 | 705 |         case OBJ: | 
|---|
 | 706 |           if (param0 == MT_NULL) | 
|---|
 | 707 |           { | 
|---|
 | 708 |             if ((*resource)->param[0] == 1.0f) | 
|---|
 | 709 |               match = true; | 
|---|
 | 710 |           } | 
|---|
 | 711 |           else if ((*resource)->param[0] == param0.getFloat()) | 
|---|
| [6640] | 712 |             match = true; | 
|---|
| [6648] | 713 |           break; | 
|---|
 | 714 |         case MD2: | 
|---|
 | 715 |           if (param0 == MT_NULL) | 
|---|
 | 716 |           { | 
|---|
 | 717 |             if ((*resource)->param[0] == "") | 
|---|
 | 718 |               match = true; | 
|---|
 | 719 |           } | 
|---|
 | 720 |           else if ((*resource)->param[0] == ((MultiType)param0).getString()) | 
|---|
| [6640] | 721 |             match = true; | 
|---|
| [6648] | 722 |           break; | 
|---|
| [4534] | 723 | #endif /* NO_MODEL */ | 
|---|
 | 724 | #ifndef NO_TEXT | 
|---|
| [6648] | 725 |         case TTF: | 
|---|
 | 726 |           if (param0 == MT_NULL) | 
|---|
 | 727 |           { | 
|---|
 | 728 |             if ((*resource)->param[0] == FONT_DEFAULT_RENDER_SIZE) | 
|---|
 | 729 |               match = true; | 
|---|
 | 730 |           } | 
|---|
 | 731 |           else if ((*resource)->param[0] == param0.getInt()) | 
|---|
| [6640] | 732 |             match = true; | 
|---|
| [6648] | 733 |           break; | 
|---|
| [4534] | 734 | #endif /* NO_TEXT */ | 
|---|
| [5323] | 735 | #ifndef NO_SHADERS | 
|---|
| [6648] | 736 |         case SHADER: | 
|---|
 | 737 |           if (param0 == MT_NULL) | 
|---|
 | 738 |           { | 
|---|
 | 739 |             if ((*resource)->param[0] == "") | 
|---|
 | 740 |               match = true; | 
|---|
 | 741 |           } | 
|---|
 | 742 |           else if ((*resource)->param[0] == ((MultiType)param0).getString()) | 
|---|
| [6640] | 743 |             match = true; | 
|---|
| [5323] | 744 | #endif /* NO_SHADERS */ | 
|---|
| [6467] | 745 | #ifndef NO_TEXTURES | 
|---|
| [6648] | 746 |         case IMAGE: | 
|---|
 | 747 |           if (param0 == MT_NULL) | 
|---|
 | 748 |           { | 
|---|
 | 749 |             if ((*resource)->param[0] == GL_TEXTURE_2D) | 
|---|
 | 750 |               match = true; | 
|---|
 | 751 |           } | 
|---|
 | 752 |           else if ((*resource)->param[0] ==  param0.getInt()) | 
|---|
| [6640] | 753 |             match = true; | 
|---|
| [6648] | 754 | #endif /* NO_TEXTURES */ | 
|---|
 | 755 |         default: | 
|---|
| [6640] | 756 |           match = true; | 
|---|
| [6648] | 757 |           break; | 
|---|
| [6640] | 758 |       } | 
|---|
 | 759 |       if (match) | 
|---|
 | 760 |       { | 
|---|
 | 761 |         return (*resource); | 
|---|
 | 762 |       } | 
|---|
| [3658] | 763 |     } | 
|---|
| [6640] | 764 |   } | 
|---|
| [3658] | 765 |   return NULL; | 
|---|
 | 766 | } | 
|---|
 | 767 |  | 
|---|
 | 768 | /** | 
|---|
| [6647] | 769 |  * @brief Searches for a Resource by Pointer | 
|---|
| [4836] | 770 |  * @param pointer the Pointer to search for | 
|---|
 | 771 |  * @returns a Pointer to the Resource if found, NULL otherwise. | 
|---|
| [6647] | 772 |  */ | 
|---|
| [5994] | 773 | Resource* ResourceManager::locateResourceByPointer(const void* pointer) const | 
|---|
| [3658] | 774 | { | 
|---|
| [3667] | 775 |   //  Resource* enumRes = resourceList->enumerate(); | 
|---|
| [6642] | 776 |   std::vector<Resource*>::const_iterator resource; | 
|---|
| [6222] | 777 |   for (resource = this->resourceList.begin(); resource != this->resourceList.end(); resource++) | 
|---|
 | 778 |     if (pointer == (*resource)->pointer) | 
|---|
| [6640] | 779 |       return (*resource); | 
|---|
| [3658] | 780 |   return NULL; | 
|---|
 | 781 | } | 
|---|
 | 782 |  | 
|---|
| [6648] | 783 | char* ResourceManager::toResourcableString(unsigned int i) | 
|---|
 | 784 | { | 
|---|
 | 785 |   int len = strlen(ResourceManager::ResourceTypeToChar(this->resourceList[i]->type)); | 
|---|
 | 786 |   len += strlen(this->resourceList[i]->name); | 
|---|
 | 787 |   if (this->resourceList[i]->param[0].getString()) len += strlen(this->resourceList[i]->param[0].getString()) +1; | 
|---|
 | 788 |   if (this->resourceList[i]->param[1].getString()) len += strlen(this->resourceList[i]->param[1].getString()) +1; | 
|---|
 | 789 |   if (this->resourceList[i]->param[2].getString()) len += strlen(this->resourceList[i]->param[2].getString()) +1; | 
|---|
 | 790 |   len += 10; | 
|---|
 | 791 |   char* tmp = new char[len]; | 
|---|
 | 792 |   tmp[0] = '\0'; | 
|---|
 | 793 |   strcat( tmp, ResourceManager::ResourceTypeToChar(this->resourceList[i]->type)); | 
|---|
 | 794 |   strcat(tmp,","); | 
|---|
 | 795 |   strcat (tmp, this->resourceList[i]->name); | 
|---|
 | 796 |   if (this->resourceList[i]->param[0].getString() && this->resourceList[i]->param[0].getString() != '\0') | 
|---|
 | 797 |   { | 
|---|
 | 798 |     strcat(tmp,","); | 
|---|
 | 799 |     strcat( tmp, this->resourceList[i]->param[0].getString()); | 
|---|
 | 800 |   } | 
|---|
 | 801 |   if (this->resourceList[i]->param[1].getString() && this->resourceList[i]->param[1].getString() != '\0') | 
|---|
 | 802 |   { | 
|---|
 | 803 |     strcat(tmp,","); | 
|---|
 | 804 |     strcat( tmp, this->resourceList[i]->param[1].getString()); | 
|---|
 | 805 |   } | 
|---|
 | 806 |   if (this->resourceList[i]->param[2].getString() && this->resourceList[i]->param[2].getString() != '\0') | 
|---|
 | 807 |   { | 
|---|
 | 808 |     strcat(tmp,","); | 
|---|
 | 809 |     strcat( tmp, this->resourceList[i]->param[2].getString()); | 
|---|
 | 810 |   } | 
|---|
 | 811 |   return tmp; | 
|---|
 | 812 | } | 
|---|
 | 813 |  | 
|---|
| [3658] | 814 | /** | 
|---|
| [6648] | 815 |  * @brief caches a Resource from a ResourceableString created with the toResourcableString-function | 
|---|
 | 816 |  * @param resourceableString the String to cache the resource from. | 
|---|
 | 817 |  */ | 
|---|
 | 818 | bool ResourceManager::fromResourceableString(const char* resourceableString) | 
|---|
 | 819 | { | 
|---|
 | 820 |   SubString splits(resourceableString, ','); | 
|---|
 | 821 |   splits.debug(); | 
|---|
 | 822 |   if (splits.getCount() == 2) | 
|---|
 | 823 |     this->cache(splits[1], ResourceManager::stringToResourceType(splits[0]), | 
|---|
 | 824 |                 RP_LEVEL); | 
|---|
 | 825 |   else if (splits.getCount() == 3) | 
|---|
| [6650] | 826 |     return this->cache(splits[1], ResourceManager::stringToResourceType(splits[0]), | 
|---|
| [6648] | 827 |                 RP_LEVEL, splits[2]); | 
|---|
 | 828 |   else if (splits.getCount() == 4) | 
|---|
| [6650] | 829 |     return this->cache(splits[1], ResourceManager::stringToResourceType(splits[0]), | 
|---|
| [6648] | 830 |                 RP_LEVEL, splits[2], splits[3]); | 
|---|
 | 831 |   else if (splits.getCount() == 5) | 
|---|
| [6650] | 832 |     return this->cache(splits[1], ResourceManager::stringToResourceType(splits[0]), | 
|---|
| [6648] | 833 |                 RP_LEVEL, splits[2], splits[3], splits[4]); | 
|---|
 | 834 | } | 
|---|
 | 835 |  | 
|---|
 | 836 |  | 
|---|
 | 837 | /** | 
|---|
| [6647] | 838 |  * @brief Checks if it is a Directory | 
|---|
| [4836] | 839 |  * @param directoryName the Directory to check for | 
|---|
 | 840 |  * @returns true if it is a directory/symlink false otherwise | 
|---|
| [3655] | 841 | */ | 
|---|
 | 842 | bool ResourceManager::isDir(const char* directoryName) | 
|---|
 | 843 | { | 
|---|
| [4462] | 844 |   if (directoryName == NULL) | 
|---|
 | 845 |     return false; | 
|---|
 | 846 |  | 
|---|
| [3883] | 847 |   char* tmpDirName = NULL; | 
|---|
| [3655] | 848 |   struct stat status; | 
|---|
| [3883] | 849 |  | 
|---|
 | 850 |   // checking for the termination of the string given. If there is a "/" at the end cut it away | 
|---|
| [5113] | 851 |   if (directoryName[strlen(directoryName)-1] == '/' || | 
|---|
 | 852 |       directoryName[strlen(directoryName)-1] == '\\') | 
|---|
| [6640] | 853 |   { | 
|---|
 | 854 |     tmpDirName = new char[strlen(directoryName)]; | 
|---|
 | 855 |     strncpy(tmpDirName, directoryName, strlen(directoryName)-1); | 
|---|
 | 856 |     tmpDirName[strlen(directoryName)-1] = '\0'; | 
|---|
 | 857 |   } | 
|---|
| [3883] | 858 |   else | 
|---|
| [6640] | 859 |   { | 
|---|
 | 860 |     tmpDirName = new char[strlen(directoryName)+1]; | 
|---|
 | 861 |     strcpy(tmpDirName, directoryName); | 
|---|
 | 862 |   } | 
|---|
| [3883] | 863 |  | 
|---|
| [4032] | 864 |   if(!stat(tmpDirName, &status)) | 
|---|
| [6640] | 865 |   { | 
|---|
 | 866 |     if (status.st_mode & (S_IFDIR | 
|---|
| [3790] | 867 | #ifndef __WIN32__ | 
|---|
| [6640] | 868 |                           | S_IFLNK | 
|---|
| [3790] | 869 | #endif | 
|---|
| [6640] | 870 |                          )) | 
|---|
 | 871 |     { | 
|---|
 | 872 |       delete[] tmpDirName; | 
|---|
 | 873 |       return true; | 
|---|
| [3883] | 874 |     } | 
|---|
| [6640] | 875 |     else | 
|---|
 | 876 |     { | 
|---|
 | 877 |       delete[] tmpDirName; | 
|---|
 | 878 |       return false; | 
|---|
 | 879 |     } | 
|---|
 | 880 |   } | 
|---|
| [3658] | 881 |   else | 
|---|
| [5211] | 882 |   { | 
|---|
 | 883 |     delete[] tmpDirName; | 
|---|
| [4032] | 884 |     return false; | 
|---|
| [5211] | 885 |   } | 
|---|
| [3655] | 886 | } | 
|---|
 | 887 |  | 
|---|
 | 888 | /** | 
|---|
| [6647] | 889 |  * @brief Checks if the file is either a Regular file or a Symlink | 
|---|
| [4836] | 890 |  * @param fileName the File to check for | 
|---|
 | 891 |  * @returns true if it is a regular file/symlink, false otherwise | 
|---|
| [3655] | 892 | */ | 
|---|
 | 893 | bool ResourceManager::isFile(const char* fileName) | 
|---|
 | 894 | { | 
|---|
| [4462] | 895 |   if (fileName == NULL) | 
|---|
 | 896 |     return false; | 
|---|
| [4032] | 897 |   char* tmpFileName = ResourceManager::homeDirCheck(fileName); | 
|---|
 | 898 |   // actually checks the File | 
|---|
| [3655] | 899 |   struct stat status; | 
|---|
| [4032] | 900 |   if (!stat(tmpFileName, &status)) | 
|---|
| [6640] | 901 |   { | 
|---|
 | 902 |     if (status.st_mode & (S_IFREG | 
|---|
| [3790] | 903 | #ifndef __WIN32__ | 
|---|
| [6640] | 904 |                           | S_IFLNK | 
|---|
| [3790] | 905 | #endif | 
|---|
| [6640] | 906 |                          )) | 
|---|
 | 907 |     { | 
|---|
 | 908 |       delete[] tmpFileName; | 
|---|
 | 909 |       return true; | 
|---|
| [4032] | 910 |     } | 
|---|
| [6640] | 911 |     else | 
|---|
| [4032] | 912 |     { | 
|---|
| [5208] | 913 |       delete[] tmpFileName; | 
|---|
| [4032] | 914 |       return false; | 
|---|
 | 915 |     } | 
|---|
| [6640] | 916 |   } | 
|---|
 | 917 |   else | 
|---|
 | 918 |   { | 
|---|
 | 919 |     delete[] tmpFileName; | 
|---|
 | 920 |     return false; | 
|---|
 | 921 |   } | 
|---|
| [4032] | 922 | } | 
|---|
 | 923 |  | 
|---|
| [4166] | 924 | /** | 
|---|
| [6647] | 925 |  * @brief touches a File on the disk (thereby creating it) | 
|---|
| [4836] | 926 |  * @param fileName The file to touch | 
|---|
| [4166] | 927 | */ | 
|---|
| [4032] | 928 | bool ResourceManager::touchFile(const char* fileName) | 
|---|
 | 929 | { | 
|---|
 | 930 |   char* tmpName = ResourceManager::homeDirCheck(fileName); | 
|---|
| [4462] | 931 |   if (tmpName == NULL) | 
|---|
 | 932 |     return false; | 
|---|
| [4032] | 933 |   FILE* stream; | 
|---|
 | 934 |   if( (stream = fopen (tmpName, "w")) == NULL) | 
|---|
| [6640] | 935 |   { | 
|---|
 | 936 |     PRINTF(1)("could not open %s fro writing\n", fileName); | 
|---|
 | 937 |     delete[] tmpName; | 
|---|
 | 938 |     return false; | 
|---|
 | 939 |   } | 
|---|
| [4033] | 940 |   fclose(stream); | 
|---|
| [4597] | 941 |  | 
|---|
| [5208] | 942 |   delete[] tmpName; | 
|---|
| [4032] | 943 | } | 
|---|
 | 944 |  | 
|---|
| [4166] | 945 | /** | 
|---|
| [6647] | 946 |  * @brief deletes a File from disk | 
|---|
| [4836] | 947 |  * @param fileName the File to delete | 
|---|
| [4166] | 948 | */ | 
|---|
| [4032] | 949 | bool ResourceManager::deleteFile(const char* fileName) | 
|---|
 | 950 | { | 
|---|
| [4462] | 951 |   if (fileName == NULL) | 
|---|
 | 952 |     return false; | 
|---|
| [4032] | 953 |   char* tmpName = ResourceManager::homeDirCheck(fileName); | 
|---|
 | 954 |   unlink(tmpName); | 
|---|
| [5208] | 955 |   delete[] tmpName; | 
|---|
| [4032] | 956 | } | 
|---|
 | 957 |  | 
|---|
| [4597] | 958 | /** | 
|---|
| [4961] | 959 |  * @param name the Name of the file to check | 
|---|
 | 960 |  * @returns The name of the file, including the HomeDir | 
|---|
 | 961 |  * IMPORTANT: this has to be deleted from the outside | 
|---|
 | 962 |  */ | 
|---|
| [4032] | 963 | char* ResourceManager::homeDirCheck(const char* name) | 
|---|
 | 964 | { | 
|---|
| [4462] | 965 |   if (name == NULL) | 
|---|
 | 966 |     return NULL; | 
|---|
| [4032] | 967 |   char* retName; | 
|---|
 | 968 |   if (!strncmp(name, "~/", 2)) | 
|---|
| [6640] | 969 |   { | 
|---|
 | 970 |     char tmpFileName[500]; | 
|---|
| [4032] | 971 | #ifdef __WIN32__ | 
|---|
| [6640] | 972 |     strcpy(tmpFileName, getenv("USERPROFILE")); | 
|---|
| [4032] | 973 | #else | 
|---|
| [6640] | 974 |     strcpy(tmpFileName, getenv("HOME")); | 
|---|
| [4032] | 975 | #endif | 
|---|
| [6640] | 976 |     retName = new char[strlen(tmpFileName)+strlen(name)]; | 
|---|
 | 977 |     sprintf(retName, "%s%s", tmpFileName, name+1); | 
|---|
 | 978 |   } | 
|---|
| [3655] | 979 |   else | 
|---|
| [6640] | 980 |   { | 
|---|
 | 981 |     retName = new char[strlen(name)+1]; | 
|---|
 | 982 |     strcpy(retName, name); | 
|---|
 | 983 |   } | 
|---|
| [4032] | 984 |   return retName; | 
|---|
| [3655] | 985 | } | 
|---|
| [3676] | 986 |  | 
|---|
| [4597] | 987 | /** | 
|---|
| [4961] | 988 |  * @param fileName the Name of the File to check | 
|---|
 | 989 |  * @returns The full name of the file, including the DataDir, and NULL if the file does not exist | 
|---|
| [5219] | 990 |  * !!IMPORTANT: this has to be deleted from the outside!! | 
|---|
| [4166] | 991 | */ | 
|---|
 | 992 | char* ResourceManager::getFullName(const char* fileName) | 
|---|
 | 993 | { | 
|---|
| [5335] | 994 |   if (fileName == NULL || ResourceManager::getInstance()->getDataDir() == NULL) | 
|---|
| [4462] | 995 |     return NULL; | 
|---|
 | 996 |  | 
|---|
| [4216] | 997 |   char* retName = new char[strlen(ResourceManager::getInstance()->getDataDir()) | 
|---|
| [4597] | 998 |                            + strlen(fileName) + 1]; | 
|---|
| [4166] | 999 |   sprintf(retName, "%s%s", ResourceManager::getInstance()->getDataDir(), fileName); | 
|---|
| [4462] | 1000 |   if (ResourceManager::isFile(retName) || ResourceManager::isDir(retName)) | 
|---|
| [4167] | 1001 |     return retName; | 
|---|
 | 1002 |   else | 
|---|
| [6640] | 1003 |   { | 
|---|
 | 1004 |     delete[] retName; | 
|---|
 | 1005 |     return NULL; | 
|---|
 | 1006 |   } | 
|---|
| [4166] | 1007 | } | 
|---|
| [4032] | 1008 |  | 
|---|
 | 1009 |  | 
|---|
| [3676] | 1010 | /** | 
|---|
| [6647] | 1011 |  * @brief checks wether a file is in the DataDir. | 
|---|
| [5335] | 1012 |  * @param fileName the File to check if it is in the Data-Dir structure. | 
|---|
 | 1013 |  * @returns true if the file exists, false otherwise | 
|---|
 | 1014 |  */ | 
|---|
 | 1015 | bool ResourceManager::isInDataDir(const char* fileName) | 
|---|
 | 1016 | { | 
|---|
 | 1017 |   if (fileName == NULL || ResourceManager::getInstance()->getDataDir() == NULL) | 
|---|
 | 1018 |     return false; | 
|---|
 | 1019 |  | 
|---|
 | 1020 |   bool retVal = false; | 
|---|
 | 1021 |   char* checkFile = new char[strlen(ResourceManager::getInstance()->getDataDir()) | 
|---|
| [6640] | 1022 |                              + strlen(fileName) + 1]; | 
|---|
| [5335] | 1023 |   sprintf(checkFile, "%s%s", ResourceManager::getInstance()->getDataDir(), fileName); | 
|---|
 | 1024 |  | 
|---|
 | 1025 |   if (ResourceManager::isFile(checkFile) || ResourceManager::isDir(checkFile)) | 
|---|
 | 1026 |     retVal = true; | 
|---|
 | 1027 |   else | 
|---|
 | 1028 |     retVal = false; | 
|---|
 | 1029 |   delete[] checkFile; | 
|---|
 | 1030 |   return retVal; | 
|---|
 | 1031 | } | 
|---|
 | 1032 |  | 
|---|
 | 1033 |  | 
|---|
 | 1034 | /** | 
|---|
| [6647] | 1035 |  * @brief outputs debug information about the ResourceManager | 
|---|
 | 1036 |  */ | 
|---|
| [4746] | 1037 | void ResourceManager::debug() const | 
|---|
| [3676] | 1038 | { | 
|---|
 | 1039 |   PRINT(0)("=RM===================================\n"); | 
|---|
 | 1040 |   PRINT(0)("= RESOURCE-MANAGER DEBUG INFORMATION =\n"); | 
|---|
 | 1041 |   PRINT(0)("======================================\n"); | 
|---|
 | 1042 |   // if it is not initialized | 
|---|
 | 1043 |   PRINT(0)(" Reference is: %p\n", ResourceManager::singletonRef); | 
|---|
 | 1044 |   PRINT(0)(" Data-Directory is: %s\n", this->dataDir); | 
|---|
 | 1045 |   PRINT(0)(" List of Image-Directories: "); | 
|---|
| [6642] | 1046 |   std::vector<char*>::const_iterator imageDir; | 
|---|
| [6222] | 1047 |   for (imageDir = this->imageDirs.begin(); imageDir != this->imageDirs.end(); imageDir++) | 
|---|
| [6640] | 1048 |     PRINT(0)("%s ", (*imageDir)); | 
|---|
| [3676] | 1049 |   PRINT(0)("\n"); | 
|---|
 | 1050 |  | 
|---|
 | 1051 |   PRINT(0)("List of all stored Resources:\n"); | 
|---|
| [6642] | 1052 |   std::vector<Resource*>::const_iterator resource; | 
|---|
| [6222] | 1053 |   for (resource = this->resourceList.begin(); resource != this->resourceList.end(); resource++) | 
|---|
 | 1054 |  | 
|---|
| [6640] | 1055 |   { | 
|---|
 | 1056 |     PRINT(0)("-----------------------------------------\n"); | 
|---|
 | 1057 |     PRINT(0)("Name: %s; References: %d; Type: %s ", (*resource)->name, (*resource)->count, ResourceManager::ResourceTypeToChar((*resource)->type)); | 
|---|
 | 1058 |  | 
|---|
 | 1059 |     PRINT(0)("gets deleted at "); | 
|---|
 | 1060 |     switch((*resource)->prio) | 
|---|
| [3676] | 1061 |     { | 
|---|
| [6648] | 1062 |       default: | 
|---|
 | 1063 |       case RP_NO: | 
|---|
 | 1064 |         PRINT(0)("first posibility (0)\n"); | 
|---|
 | 1065 |         break; | 
|---|
 | 1066 |       case RP_LEVEL: | 
|---|
 | 1067 |         PRINT(0)("the end of the Level (1)\n"); | 
|---|
 | 1068 |         break; | 
|---|
 | 1069 |       case RP_CAMPAIGN: | 
|---|
 | 1070 |         PRINT(0)("the end of the campaign (2)\n"); | 
|---|
 | 1071 |         break; | 
|---|
 | 1072 |       case RP_GAME: | 
|---|
 | 1073 |         PRINT(0)("when leaving the game (3)\n"); | 
|---|
 | 1074 |         break; | 
|---|
| [3676] | 1075 |     } | 
|---|
| [6640] | 1076 |   } | 
|---|
| [3676] | 1077 |  | 
|---|
 | 1078 |  | 
|---|
 | 1079 |  | 
|---|
 | 1080 |   PRINT(0)("==================================RM==\n"); | 
|---|
 | 1081 | } | 
|---|
| [5306] | 1082 |  | 
|---|
 | 1083 |  | 
|---|
 | 1084 | /** | 
|---|
| [6647] | 1085 |  * @brief converts a ResourceType into the corresponding String | 
|---|
| [5306] | 1086 |  * @param type the ResourceType to translate | 
|---|
 | 1087 |  * @returns the converted String. | 
|---|
 | 1088 |  */ | 
|---|
 | 1089 | const char* ResourceManager::ResourceTypeToChar(ResourceType type) | 
|---|
 | 1090 | { | 
|---|
| [6646] | 1091 |   return ResourceManager::resourceNames[type]; | 
|---|
 | 1092 | } | 
|---|
 | 1093 |  | 
|---|
 | 1094 | /** | 
|---|
 | 1095 |  * @brief converts a String into a ResourceType (good for loading) | 
|---|
 | 1096 |  * @param resourceType the name of the Type | 
|---|
 | 1097 |  * @returns the Number of the Type, or 0 (defautl) if not found. | 
|---|
 | 1098 |  */ | 
|---|
 | 1099 | ResourceType ResourceManager::stringToResourceType(const char* resourceType) | 
|---|
 | 1100 | { | 
|---|
 | 1101 |   assert(resourceType != NULL); | 
|---|
 | 1102 |   for (unsigned int i = 0; i < RESOURCE_TYPE_SIZE; i++) | 
|---|
 | 1103 |     if (!strcmp(resourceType, ResourceManager::resourceNames[i])) | 
|---|
 | 1104 |       return (ResourceType)i; | 
|---|
 | 1105 |   return (ResourceType)0; | 
|---|
 | 1106 | } | 
|---|
 | 1107 |  | 
|---|
 | 1108 | /** | 
|---|
 | 1109 |  * The Names of the ResourceTypes | 
|---|
 | 1110 |  */ | 
|---|
 | 1111 | const char* ResourceManager::resourceNames[] = | 
|---|
| [6648] | 1112 |   { | 
|---|
| [5306] | 1113 | #ifndef NO_MODEL | 
|---|
| [6648] | 1114 |     "ObjectModel", | 
|---|
 | 1115 |     "PrimitiveModel", | 
|---|
 | 1116 |     "MD2-Data", | 
|---|
| [5306] | 1117 | #endif | 
|---|
| [6648] | 1118 | #ifndef NO_TEXT | 
|---|
 | 1119 |     "Font", | 
|---|
| [5306] | 1120 | #endif | 
|---|
 | 1121 | #ifndef NO_AUDIO | 
|---|
| [6648] | 1122 |     "Wav", | 
|---|
 | 1123 |     "mp3", | 
|---|
 | 1124 |     "ogg", | 
|---|
| [5306] | 1125 | #endif | 
|---|
| [6648] | 1126 | #ifndef NO_TEXTURES | 
|---|
 | 1127 |     "Texture", | 
|---|
| [5306] | 1128 | #endif | 
|---|
| [5323] | 1129 | #ifndef NO_SHADERS | 
|---|
| [6648] | 1130 |     "Shader", | 
|---|
| [5323] | 1131 | #endif | 
|---|
| [6648] | 1132 |  | 
|---|
 | 1133 |   }; | 
|---|