2006-04-02 14:20:45 +00:00
|
|
|
/* Residual - Virtual machine to run LucasArts' 3D adventure games
|
2008-06-13 14:57:47 +00:00
|
|
|
*
|
|
|
|
* Residual is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the AUTHORS
|
|
|
|
* file distributed with this source distribution.
|
2006-04-02 14:20:45 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
*/
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2008-01-26 11:47:23 +00:00
|
|
|
#include "engine/resource.h"
|
|
|
|
#include "engine/colormap.h"
|
|
|
|
#include "engine/costume.h"
|
|
|
|
#include "engine/keyframe.h"
|
|
|
|
#include "engine/material.h"
|
|
|
|
#include "engine/engine.h"
|
2009-04-21 18:04:24 +00:00
|
|
|
#include "engine/lipsync.h"
|
2005-01-01 12:27:57 +00:00
|
|
|
|
2004-12-31 21:35:04 +00:00
|
|
|
ResourceLoader *g_resourceloader = NULL;
|
2003-08-15 18:00:22 +00:00
|
|
|
|
|
|
|
ResourceLoader::ResourceLoader() {
|
2003-08-24 09:34:19 +00:00
|
|
|
int lab_counter = 0;
|
2009-05-12 14:31:48 +00:00
|
|
|
_cacheDirty = false;
|
|
|
|
|
2009-04-05 14:48:54 +00:00
|
|
|
Lab *l;
|
|
|
|
Common::ArchiveMemberList files;
|
2008-08-01 18:27:11 +00:00
|
|
|
|
2009-04-05 14:48:54 +00:00
|
|
|
SearchMan.listMatchingMembers(files, "*.lab");
|
2008-08-01 18:27:11 +00:00
|
|
|
|
2009-04-05 14:48:54 +00:00
|
|
|
if (files.empty())
|
|
|
|
error("Cannot find game data - check configuration file");
|
2008-08-01 18:27:11 +00:00
|
|
|
|
2009-04-05 14:48:54 +00:00
|
|
|
for (Common::ArchiveMemberList::const_iterator x = files.begin(); x != files.end(); ++x) {
|
|
|
|
const Common::String filename = (*x)->getName();
|
|
|
|
l = new Lab(filename.c_str());
|
2008-08-01 18:27:11 +00:00
|
|
|
if (l->isOpen()) {
|
|
|
|
if (filename == "005.lab")
|
|
|
|
_labs.push_front(l);
|
|
|
|
else {
|
|
|
|
if (filename == "gfdemo01.lab")
|
|
|
|
g_flags |= GF_DEMO;
|
|
|
|
_labs.push_back(l);
|
|
|
|
}
|
2003-08-24 09:34:19 +00:00
|
|
|
lab_counter++;
|
2008-08-01 18:27:11 +00:00
|
|
|
} else {
|
|
|
|
delete l;
|
2003-08-24 09:34:19 +00:00
|
|
|
}
|
2008-08-01 18:27:11 +00:00
|
|
|
}
|
|
|
|
|
2009-04-05 14:48:54 +00:00
|
|
|
files.clear();
|
|
|
|
|
|
|
|
SearchMan.listMatchingMembers(files, "*.mus");
|
|
|
|
|
|
|
|
for (Common::ArchiveMemberList::const_iterator x = files.begin(); x != files.end(); ++x) {
|
|
|
|
const Common::String filename = (*x)->getName();
|
|
|
|
l = new Lab(filename.c_str());
|
2008-08-01 18:27:11 +00:00
|
|
|
if (l->isOpen()) {
|
|
|
|
_labs.push_back(l);
|
2003-08-24 09:34:19 +00:00
|
|
|
lab_counter++;
|
2008-08-01 18:27:11 +00:00
|
|
|
} else {
|
|
|
|
delete l;
|
2003-08-24 09:34:19 +00:00
|
|
|
}
|
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2007-02-05 13:49:32 +00:00
|
|
|
ResourceLoader::~ResourceLoader() {
|
|
|
|
for (LabList::const_iterator i = _labs.begin(); i != _labs.end(); i++)
|
2006-02-26 18:38:35 +00:00
|
|
|
delete (*i);
|
|
|
|
}
|
|
|
|
|
2009-05-12 14:31:48 +00:00
|
|
|
const Lab *ResourceLoader::getLab(const char *filename) const {
|
2004-12-09 23:55:43 +00:00
|
|
|
for (LabList::const_iterator i = _labs.begin(); i != _labs.end(); i++)
|
2004-02-24 22:43:32 +00:00
|
|
|
if ((*i)->fileExists(filename))
|
|
|
|
return *i;
|
2007-02-05 13:49:32 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
return NULL;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2009-05-12 14:31:48 +00:00
|
|
|
static int sortCallback(const void *entry1, const void *entry2) {
|
|
|
|
return strcasecmp(((ResourceLoader::ResourceCache *)entry1)->fname, ((ResourceLoader::ResourceCache *)entry2)->fname);
|
|
|
|
}
|
|
|
|
|
|
|
|
Resource *ResourceLoader::getFileFromCache(const char *filename) {
|
|
|
|
ResourceLoader::ResourceCache *entry = getEntryFromCache(filename);
|
|
|
|
if (entry)
|
|
|
|
return entry->resPtr;
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ResourceLoader::ResourceCache *ResourceLoader::getEntryFromCache(const char *filename) {
|
|
|
|
if (_cacheDirty) {
|
|
|
|
qsort(_cache.begin(), _cache.size(), sizeof(ResourceCache), sortCallback);
|
|
|
|
_cacheDirty = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ResourceCache key;
|
|
|
|
key.fname = (char *)filename;
|
|
|
|
|
|
|
|
return (ResourceLoader::ResourceCache *)bsearch(&key, _cache.begin(), _cache.size(), sizeof(ResourceCache), sortCallback);
|
|
|
|
}
|
|
|
|
|
2003-08-15 18:00:22 +00:00
|
|
|
bool ResourceLoader::fileExists(const char *filename) const {
|
2009-05-12 14:31:48 +00:00
|
|
|
return getLab(filename) != NULL;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Block *ResourceLoader::getFileBlock(const char *filename) const {
|
2009-05-12 14:31:48 +00:00
|
|
|
const Lab *l = getLab(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!l)
|
2004-02-24 22:43:32 +00:00
|
|
|
return NULL;
|
|
|
|
else
|
|
|
|
return l->getFileBlock(filename);
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2009-04-05 14:48:54 +00:00
|
|
|
LuaFile *ResourceLoader::openNewStreamLua(const char *filename) const {
|
2009-05-12 14:31:48 +00:00
|
|
|
const Lab *l = getLab(filename);
|
2009-04-05 14:48:54 +00:00
|
|
|
|
|
|
|
if (!l)
|
|
|
|
return NULL;
|
|
|
|
else
|
|
|
|
return l->openNewStreamLua(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::File *ResourceLoader::openNewStreamFile(const char *filename) const {
|
2009-05-12 14:31:48 +00:00
|
|
|
const Lab *l = getLab(filename);
|
2003-12-13 11:03:42 +00:00
|
|
|
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!l)
|
2004-02-24 22:43:32 +00:00
|
|
|
return NULL;
|
|
|
|
else
|
2009-04-05 14:48:54 +00:00
|
|
|
return l->openNewStreamFile(filename);
|
2003-12-13 11:03:42 +00:00
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
|
|
|
|
int ResourceLoader::fileLength(const char *filename) const {
|
2009-05-12 14:31:48 +00:00
|
|
|
const Lab *l = getLab(filename);
|
2008-07-26 09:18:46 +00:00
|
|
|
if (l)
|
2004-02-24 22:43:32 +00:00
|
|
|
return l->fileLength(filename);
|
2008-07-26 09:18:46 +00:00
|
|
|
else
|
|
|
|
return 0;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Bitmap *ResourceLoader::loadBitmap(const char *filename) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<Bitmap *>(ptr);
|
2004-02-24 22:43:32 +00:00
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b) { // Grim sometimes asks for non-existant bitmaps (eg, ha_overhead)
|
2005-07-10 18:57:27 +00:00
|
|
|
if (debugLevel == DEBUG_WARN || debugLevel == DEBUG_ALL)
|
2008-09-28 15:23:15 +00:00
|
|
|
warning("Could not find bitmap %s", filename);
|
2005-01-03 20:49:49 +00:00
|
|
|
return NULL;
|
2004-09-11 14:09:43 +00:00
|
|
|
}
|
2003-08-16 11:10:40 +00:00
|
|
|
|
2004-09-11 14:09:43 +00:00
|
|
|
Bitmap *result = new Bitmap(filename, b->data(), b->len());
|
2004-02-24 22:43:32 +00:00
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
|
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
return result;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2003-10-05 17:45:46 +00:00
|
|
|
CMap *ResourceLoader::loadColormap(const char *filename) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
2004-12-09 23:55:43 +00:00
|
|
|
|
2009-05-12 14:31:48 +00:00
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<CMap *>(ptr);
|
2004-02-24 22:43:32 +00:00
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b)
|
2008-09-28 15:23:15 +00:00
|
|
|
error("Could not find colormap %s", filename);
|
2004-02-24 22:43:32 +00:00
|
|
|
CMap *result = new CMap(filename, b->data(), b->len());
|
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
|
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
return result;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2003-08-20 14:29:59 +00:00
|
|
|
Costume *ResourceLoader::loadCostume(const char *filename, Costume *prevCost) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2004-02-24 22:43:32 +00:00
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b)
|
2008-09-28 15:23:15 +00:00
|
|
|
error("Could not find costume %s", filename);
|
2004-02-24 22:43:32 +00:00
|
|
|
Costume *result = new Costume(filename, b->data(), b->len(), prevCost);
|
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
return result;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2005-03-18 19:54:40 +00:00
|
|
|
Font *ResourceLoader::loadFont(const char *filename) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<Font *>(ptr);
|
2005-03-18 19:54:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b)
|
2008-09-28 15:23:15 +00:00
|
|
|
error("Could not find font file %s", filename);
|
2005-03-18 19:54:40 +00:00
|
|
|
Font *result = new Font(filename, b->data(), b->len());
|
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
|
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
|
|
|
|
2005-03-18 19:54:40 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2003-08-15 18:00:22 +00:00
|
|
|
KeyframeAnim *ResourceLoader::loadKeyframe(const char *filename) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<KeyframeAnim *>(ptr);
|
2004-02-24 22:43:32 +00:00
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b)
|
2008-09-28 15:23:15 +00:00
|
|
|
error("Could not find keyframe file %s", filename);
|
2004-02-24 22:43:32 +00:00
|
|
|
KeyframeAnim *result = new KeyframeAnim(filename, b->data(), b->len());
|
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
|
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
return result;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2009-04-21 18:04:24 +00:00
|
|
|
LipSync *ResourceLoader::loadLipSync(const char *filename) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-04-21 18:04:24 +00:00
|
|
|
LipSync *result;
|
2004-09-11 14:09:43 +00:00
|
|
|
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<LipSync *>(ptr);
|
2004-09-11 14:09:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b) {
|
2005-07-10 18:57:27 +00:00
|
|
|
if (debugLevel == DEBUG_WARN || debugLevel == DEBUG_ALL)
|
2009-04-21 18:04:24 +00:00
|
|
|
warning("Could not find lipsync file %s", filename);
|
2004-09-11 14:09:43 +00:00
|
|
|
result = NULL;
|
|
|
|
} else {
|
2009-04-21 18:04:24 +00:00
|
|
|
result = new LipSync(filename, b->data(), b->len());
|
2009-05-09 17:47:28 +00:00
|
|
|
|
2009-04-21 18:04:24 +00:00
|
|
|
// Some lipsync files have no data
|
2008-07-18 22:44:11 +00:00
|
|
|
if (result->isValid()) {
|
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
2008-07-18 22:44:11 +00:00
|
|
|
} else {
|
|
|
|
delete result;
|
|
|
|
result = NULL;
|
|
|
|
}
|
2009-05-09 17:47:28 +00:00
|
|
|
}
|
2004-09-11 14:09:43 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
Material *ResourceLoader::loadMaterial(const char *filename, const CMap &c) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = Common::String(filename) + "@" + c.filename();
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<Material *>(ptr);
|
2004-02-24 22:43:32 +00:00
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b)
|
2008-09-28 15:23:15 +00:00
|
|
|
error("Could not find material %s", filename);
|
2006-02-25 23:00:09 +00:00
|
|
|
Material *result = new Material(fname.c_str(), b->data(), b->len(), c);
|
2004-02-24 22:43:32 +00:00
|
|
|
delete b;
|
2009-05-12 14:31:48 +00:00
|
|
|
|
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
return result;
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|
|
|
|
|
2003-10-05 17:45:46 +00:00
|
|
|
Model *ResourceLoader::loadModel(const char *filename, const CMap &c) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
Resource *ptr = getFileFromCache(fname.c_str());
|
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
return dynamic_cast<Model *>(ptr);
|
2004-02-24 22:43:32 +00:00
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2004-02-24 22:43:32 +00:00
|
|
|
Block *b = getFileBlock(filename);
|
2008-07-30 07:04:32 +00:00
|
|
|
if (!b)
|
2008-09-28 15:23:15 +00:00
|
|
|
error("Could not find model %s", filename);
|
2004-02-24 22:43:32 +00:00
|
|
|
Model *result = new Model(filename, b->data(), b->len(), c);
|
|
|
|
delete b;
|
2003-08-15 18:00:22 +00:00
|
|
|
|
2009-05-12 14:31:48 +00:00
|
|
|
ResourceCache entry;
|
|
|
|
entry.resPtr = result;
|
|
|
|
entry.fname = new char[fname.size() + 1];
|
|
|
|
strcpy(entry.fname, fname.c_str());
|
|
|
|
_cache.push_back(entry);
|
|
|
|
_cacheDirty = true;
|
2009-05-09 17:47:28 +00:00
|
|
|
|
2009-05-12 14:31:48 +00:00
|
|
|
return result;
|
2005-07-17 23:40:22 +00:00
|
|
|
}
|
|
|
|
|
2003-08-15 18:00:22 +00:00
|
|
|
void ResourceLoader::uncache(const char *filename) {
|
2009-05-10 16:43:41 +00:00
|
|
|
Common::String fname = filename;
|
|
|
|
fname.toLowercase();
|
2009-05-12 14:31:48 +00:00
|
|
|
|
|
|
|
if (_cacheDirty)
|
|
|
|
qsort(_cache.begin(), _cache.size(), sizeof(ResourceCache), sortCallback);
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < _cache.size(); i++) {
|
|
|
|
if (fname.compareTo(_cache[i].fname) == 0) {
|
|
|
|
delete[] _cache[i].fname;
|
|
|
|
_cache.remove_at(i);
|
|
|
|
_cacheDirty = true;
|
|
|
|
}
|
|
|
|
}
|
2003-08-15 18:00:22 +00:00
|
|
|
}
|