2007-05-30 21:56:52 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
2003-10-08 21:59:23 +00:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2003-10-08 21:59:23 +00:00
|
|
|
*
|
2006-02-11 09:53:53 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2003-10-08 21:59:23 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2007-03-17 15:45:05 +00:00
|
|
|
#if defined(WIN32)
|
|
|
|
#include <windows.h>
|
|
|
|
// winnt.h defines ARRAYSIZE, but we want our own one...
|
|
|
|
#undef ARRAYSIZE
|
|
|
|
#endif
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
#include "common/config-manager.h"
|
2005-04-22 18:10:02 +00:00
|
|
|
#include "common/file.h"
|
2005-01-10 22:35:43 +00:00
|
|
|
#include "common/util.h"
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2007-05-27 11:40:03 +00:00
|
|
|
DECLARE_SINGLETON(Common::ConfigManager);
|
|
|
|
|
2006-06-08 06:53:00 +00:00
|
|
|
#ifdef __PLAYSTATION2__
|
2006-08-04 17:47:57 +00:00
|
|
|
#include "backends/platform/ps2/systemps2.h"
|
2006-06-08 06:53:00 +00:00
|
|
|
#endif
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
#if defined(UNIX)
|
|
|
|
#ifdef MACOSX
|
|
|
|
#define DEFAULT_CONFIG_FILE "Library/Preferences/ScummVM Preferences"
|
|
|
|
#else
|
|
|
|
#define DEFAULT_CONFIG_FILE ".scummvmrc"
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define DEFAULT_CONFIG_FILE "scummvm.ini"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define MAXLINELEN 256
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
static bool isValidDomainName(const Common::String &domName) {
|
|
|
|
const char *p = domName.c_str();
|
2005-02-22 18:11:36 +00:00
|
|
|
while (*p && (isalnum(*p) || *p == '-' || *p == '_'))
|
|
|
|
p++;
|
|
|
|
return *p == 0;
|
|
|
|
}
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
namespace Common {
|
|
|
|
|
2005-11-27 02:35:57 +00:00
|
|
|
#if !(defined(PALMOS_ARM) || defined(PALMOS_DEBUG) || defined(__GP32__))
|
2005-09-03 16:08:27 +00:00
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
const String ConfigManager::kApplicationDomain("scummvm");
|
2003-11-10 23:17:11 +00:00
|
|
|
const String ConfigManager::kTransientDomain("__TRANSIENT");
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2005-09-03 16:08:27 +00:00
|
|
|
#else
|
|
|
|
|
|
|
|
const char *ConfigManager::kApplicationDomain = "scummvm";
|
|
|
|
const char *ConfigManager::kTransientDomain = "__TRANSIENT";
|
|
|
|
|
|
|
|
#endif
|
2003-10-08 21:59:23 +00:00
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
2006-04-15 13:34:02 +00:00
|
|
|
ConfigManager::ConfigManager()
|
|
|
|
: _activeDomain(0) {
|
2004-12-25 19:03:13 +00:00
|
|
|
}
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2004-12-25 19:03:13 +00:00
|
|
|
void ConfigManager::loadDefaultConfigFile() {
|
2003-10-08 21:59:23 +00:00
|
|
|
char configFile[MAXPATHLEN];
|
2007-11-13 08:06:15 +00:00
|
|
|
// GP2X is Linux based but Home dir can be read only so do not use it and put the config in the executable dir.
|
|
|
|
// On the iPhone, the home dir of the user when you launch the app from the Springboard, is /. Which we don't want.
|
|
|
|
#if defined(UNIX) && !defined(GP2X) && !defined(IPHONE)
|
2005-10-31 01:50:51 +00:00
|
|
|
const char *home = getenv("HOME");
|
|
|
|
if (home != NULL && strlen(home) < MAXPATHLEN)
|
|
|
|
snprintf(configFile, MAXPATHLEN, "%s/%s", home, DEFAULT_CONFIG_FILE);
|
2004-12-05 23:24:43 +00:00
|
|
|
else
|
|
|
|
strcpy(configFile, DEFAULT_CONFIG_FILE);
|
2003-10-08 21:59:23 +00:00
|
|
|
#else
|
2005-06-21 22:08:21 +00:00
|
|
|
#if defined (WIN32) && !defined(_WIN32_WCE) && !defined(__SYMBIAN32__)
|
2006-07-10 01:25:52 +00:00
|
|
|
OSVERSIONINFO win32OsVersion;
|
|
|
|
ZeroMemory(&win32OsVersion, sizeof(OSVERSIONINFO));
|
|
|
|
win32OsVersion.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
|
|
|
GetVersionEx(&win32OsVersion);
|
|
|
|
// Check for non-9X version of Windows.
|
|
|
|
if (win32OsVersion.dwPlatformId != VER_PLATFORM_WIN32_WINDOWS) {
|
|
|
|
// Use the Application Data directory of the user profile.
|
|
|
|
if (win32OsVersion.dwMajorVersion >= 5) {
|
|
|
|
if (!GetEnvironmentVariable("APPDATA", configFile, sizeof(configFile)))
|
|
|
|
error("Unable to access application data directory");
|
|
|
|
} else {
|
|
|
|
if (!GetEnvironmentVariable("USERPROFILE", configFile, sizeof(configFile)))
|
|
|
|
error("Unable to access user profile directory");
|
|
|
|
|
|
|
|
strcat(configFile, "\\Application Data");
|
|
|
|
CreateDirectory(configFile, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
strcat(configFile, "\\ScummVM");
|
|
|
|
CreateDirectory(configFile, NULL);
|
|
|
|
strcat(configFile, "\\" DEFAULT_CONFIG_FILE);
|
|
|
|
|
|
|
|
if (fopen(configFile, "r") == NULL) {
|
|
|
|
// Check windows directory
|
|
|
|
char oldConfigFile[MAXPATHLEN];
|
|
|
|
GetWindowsDirectory(oldConfigFile, MAXPATHLEN);
|
|
|
|
strcat(oldConfigFile, "\\" DEFAULT_CONFIG_FILE);
|
2006-07-10 11:12:11 +00:00
|
|
|
if (fopen(oldConfigFile, "r")) {
|
|
|
|
printf("The default location of the config file (scummvm.ini) in ScummVM has changed,\n");
|
|
|
|
printf("under Windows NT4/2000/XP/Vista. You may want to consider moving your config\n");
|
|
|
|
printf("file from the old default location:\n");
|
|
|
|
printf("%s\n", oldConfigFile);
|
|
|
|
printf("to the new default location:\n");
|
|
|
|
printf("%s\n\n", configFile);
|
2006-07-10 01:25:52 +00:00
|
|
|
strcpy(configFile, oldConfigFile);
|
2006-07-10 11:12:11 +00:00
|
|
|
}
|
2006-07-10 01:25:52 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Check windows directory
|
|
|
|
GetWindowsDirectory(configFile, MAXPATHLEN);
|
|
|
|
strcat(configFile, "\\" DEFAULT_CONFIG_FILE);
|
|
|
|
}
|
|
|
|
|
2005-09-03 16:08:27 +00:00
|
|
|
#elif defined(PALMOS_MODE)
|
2004-12-05 23:24:43 +00:00
|
|
|
strcpy(configFile,"/PALM/Programs/ScummVM/" DEFAULT_CONFIG_FILE);
|
2007-11-13 08:06:15 +00:00
|
|
|
#elif defined(IPHONE)
|
|
|
|
strcpy(configFile,"/var/root/" DEFAULT_CONFIG_FILE);
|
2005-03-31 05:35:04 +00:00
|
|
|
#elif defined(__PLAYSTATION2__)
|
2006-06-08 06:53:00 +00:00
|
|
|
((OSystem_PS2*)g_system)->makeConfigPath(configFile);
|
2005-08-16 17:15:37 +00:00
|
|
|
#elif defined(__PSP__)
|
|
|
|
strcpy(configFile, "ms0:/" DEFAULT_CONFIG_FILE);
|
2005-06-21 22:08:21 +00:00
|
|
|
#elif defined (__SYMBIAN32__)
|
2006-02-12 00:33:04 +00:00
|
|
|
strcpy(configFile, Symbian::GetExecutablePath());
|
|
|
|
strcat(configFile, DEFAULT_CONFIG_FILE);
|
2003-10-08 21:59:23 +00:00
|
|
|
#else
|
|
|
|
strcpy(configFile, DEFAULT_CONFIG_FILE);
|
|
|
|
#endif
|
|
|
|
#endif
|
2005-10-08 19:10:05 +00:00
|
|
|
|
2004-12-25 19:03:13 +00:00
|
|
|
loadConfigFile(configFile);
|
2006-01-15 00:22:29 +00:00
|
|
|
flushToDisk();
|
2004-02-07 04:53:59 +00:00
|
|
|
}
|
|
|
|
|
2004-12-25 19:03:13 +00:00
|
|
|
void ConfigManager::loadConfigFile(const String &filename) {
|
2006-04-15 13:12:03 +00:00
|
|
|
_appDomain.clear();
|
2004-02-07 04:53:59 +00:00
|
|
|
_gameDomains.clear();
|
|
|
|
_transientDomain.clear();
|
|
|
|
|
|
|
|
_filename = filename;
|
2004-09-28 12:10:49 +00:00
|
|
|
_domainSaveOrder.clear();
|
2003-10-08 21:59:23 +00:00
|
|
|
loadFile(_filename);
|
2006-07-10 01:25:52 +00:00
|
|
|
printf("Using configuration file: %s\n", _filename.c_str());
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::loadFile(const String &filename) {
|
2005-04-22 18:10:02 +00:00
|
|
|
File cfg_file;
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2006-04-14 01:48:51 +00:00
|
|
|
if (!cfg_file.open(filename)) {
|
2006-01-15 00:22:29 +00:00
|
|
|
printf("Creating configuration file: %s\n", filename.c_str());
|
2003-10-08 21:59:23 +00:00
|
|
|
} else {
|
2004-03-28 18:29:27 +00:00
|
|
|
char buf[MAXLINELEN];
|
|
|
|
String domain;
|
2004-03-28 20:28:45 +00:00
|
|
|
String comment;
|
2004-03-28 18:29:27 +00:00
|
|
|
int lineno = 0;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
// TODO: Detect if a domain occurs multiple times (or likewise, if
|
|
|
|
// a key occurs multiple times inside one domain).
|
2004-03-28 18:29:27 +00:00
|
|
|
|
2005-04-22 18:10:02 +00:00
|
|
|
while (!cfg_file.eof()) {
|
2004-03-28 18:29:27 +00:00
|
|
|
lineno++;
|
2005-04-22 18:10:02 +00:00
|
|
|
if (!cfg_file.readLine(buf, MAXLINELEN))
|
|
|
|
break;
|
2004-03-28 20:28:45 +00:00
|
|
|
|
|
|
|
if (buf[0] == '#') {
|
|
|
|
// Accumulate comments here. Once we encounter either the start
|
|
|
|
// of a new domain, or a key-value-pair, we associate the value
|
|
|
|
// of the 'comment' variable with that entity.
|
|
|
|
comment += buf;
|
2005-04-22 18:10:02 +00:00
|
|
|
comment += '\n';
|
2004-03-28 18:29:27 +00:00
|
|
|
} else if (buf[0] == '[') {
|
|
|
|
// It's a new domain which begins here.
|
|
|
|
char *p = buf + 1;
|
|
|
|
// Get the domain name, and check whether it's valid (that
|
|
|
|
// is, verify that it only consists of alphanumerics,
|
|
|
|
// dashes and underscores).
|
|
|
|
while (*p && (isalnum(*p) || *p == '-' || *p == '_'))
|
|
|
|
p++;
|
|
|
|
|
|
|
|
switch (*p) {
|
|
|
|
case '\0':
|
|
|
|
error("Config file buggy: missing ] in line %d", lineno);
|
|
|
|
break;
|
|
|
|
case ']':
|
|
|
|
*p = 0;
|
|
|
|
domain = buf + 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
error("Config file buggy: Invalid character '%c' occured in domain name in line %d", *p, lineno);
|
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
// Store domain comment
|
2006-04-15 13:12:03 +00:00
|
|
|
if (domain == kApplicationDomain) {
|
|
|
|
_appDomain.setDomainComment(comment);
|
2004-03-28 20:28:45 +00:00
|
|
|
} else {
|
|
|
|
_gameDomains[domain].setDomainComment(comment);
|
|
|
|
}
|
|
|
|
comment.clear();
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-09-28 12:10:49 +00:00
|
|
|
_domainSaveOrder.push_back(domain);
|
2004-03-28 18:29:27 +00:00
|
|
|
} else {
|
2004-03-28 20:28:45 +00:00
|
|
|
// Skip leading & trailing whitespaces
|
|
|
|
char *t = rtrim(ltrim(buf));
|
2004-03-28 18:29:27 +00:00
|
|
|
|
|
|
|
// Skip empty lines
|
|
|
|
if (*t == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// If no domain has been set, this config file is invalid!
|
2006-03-28 09:42:54 +00:00
|
|
|
if (domain.empty()) {
|
2004-03-28 18:29:27 +00:00
|
|
|
error("Config file buggy: Key/value pair found outside a domain in line %d", lineno);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Split string at '=' into 'key' and 'value'.
|
2004-03-28 20:28:45 +00:00
|
|
|
char *p = strchr(t, '=');
|
|
|
|
if (!p)
|
2004-03-28 18:29:27 +00:00
|
|
|
error("Config file buggy: Junk found in line line %d: '%s'", lineno, t);
|
2004-03-28 20:28:45 +00:00
|
|
|
*p = 0;
|
|
|
|
String key = rtrim(t);
|
|
|
|
String value = ltrim(p + 1);
|
|
|
|
set(key, value, domain);
|
2006-01-21 11:57:19 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
// Store comment
|
2006-04-15 13:12:03 +00:00
|
|
|
if (domain == kApplicationDomain) {
|
|
|
|
_appDomain.setKVComment(key, comment);
|
2003-10-08 21:59:23 +00:00
|
|
|
} else {
|
2004-03-28 20:28:45 +00:00
|
|
|
_gameDomains[domain].setKVComment(key, comment);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
2004-03-28 20:28:45 +00:00
|
|
|
comment.clear();
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::flushToDisk() {
|
2006-06-03 15:34:48 +00:00
|
|
|
#ifndef __DC__
|
2007-03-08 16:43:33 +00:00
|
|
|
File cfg_file;
|
2003-10-08 21:59:23 +00:00
|
|
|
|
|
|
|
// TODO
|
|
|
|
// if (!willwrite)
|
|
|
|
// return;
|
|
|
|
|
2007-03-08 16:43:33 +00:00
|
|
|
if (!cfg_file.open(_filename, File::kFileWriteMode)) {
|
2006-01-15 10:22:44 +00:00
|
|
|
warning("Unable to write configuration file: %s", _filename.c_str());
|
2003-10-08 21:59:23 +00:00
|
|
|
} else {
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-09-28 12:10:49 +00:00
|
|
|
// First write the domains in _domainSaveOrder, in that order.
|
|
|
|
// Note: It's possible for _domainSaveOrder to list domains which
|
|
|
|
// are not present anymore.
|
|
|
|
StringList::const_iterator i;
|
|
|
|
for (i = _domainSaveOrder.begin(); i != _domainSaveOrder.end(); ++i) {
|
2006-04-15 13:12:03 +00:00
|
|
|
if (kApplicationDomain == *i) {
|
|
|
|
writeDomain(cfg_file, *i, _appDomain);
|
2004-09-28 12:10:49 +00:00
|
|
|
} else if (_gameDomains.contains(*i)) {
|
|
|
|
writeDomain(cfg_file, *i, _gameDomains[*i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-02-05 00:19:57 +00:00
|
|
|
DomainMap::const_iterator d;
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
// Now write the domains which haven't been written yet
|
|
|
|
if (!_domainSaveOrder.contains(kApplicationDomain))
|
|
|
|
writeDomain(cfg_file, kApplicationDomain, _appDomain);
|
2003-10-08 21:59:23 +00:00
|
|
|
for (d = _gameDomains.begin(); d != _gameDomains.end(); ++d) {
|
2004-09-28 12:10:49 +00:00
|
|
|
if (!_domainSaveOrder.contains(d->_key))
|
|
|
|
writeDomain(cfg_file, d->_key, d->_value);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
}
|
2006-06-03 15:34:48 +00:00
|
|
|
#endif // !__DC__
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2007-03-08 16:43:33 +00:00
|
|
|
void ConfigManager::writeDomain(WriteStream &stream, const String &name, const Domain &domain) {
|
2006-03-28 09:42:54 +00:00
|
|
|
if (domain.empty())
|
2003-10-08 21:59:23 +00:00
|
|
|
return; // Don't bother writing empty domains.
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
String comment;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
// Write domain comment (if any)
|
|
|
|
comment = domain.getDomainComment();
|
2006-03-28 09:42:54 +00:00
|
|
|
if (!comment.empty())
|
2007-03-08 16:43:33 +00:00
|
|
|
stream.writeString(comment);
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
// Write domain start
|
2007-03-08 16:43:33 +00:00
|
|
|
stream.writeByte('[');
|
|
|
|
stream.writeString(name);
|
|
|
|
stream.writeByte(']');
|
|
|
|
stream.writeByte('\n');
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2004-03-28 20:28:45 +00:00
|
|
|
// Write all key/value pairs in this domain, including comments
|
2004-02-05 00:19:57 +00:00
|
|
|
Domain::const_iterator x;
|
2003-10-08 21:59:23 +00:00
|
|
|
for (x = domain.begin(); x != domain.end(); ++x) {
|
2007-03-08 16:43:33 +00:00
|
|
|
if (!x->_value.empty()) {
|
2004-03-28 20:28:45 +00:00
|
|
|
// Write comment (if any)
|
2004-03-29 19:15:23 +00:00
|
|
|
if (domain.hasKVComment(x->_key)) {
|
|
|
|
comment = domain.getKVComment(x->_key);
|
2007-03-08 16:43:33 +00:00
|
|
|
stream.writeString(comment);
|
2004-03-29 19:15:23 +00:00
|
|
|
}
|
2004-03-28 20:28:45 +00:00
|
|
|
// Write the key/value pair
|
2007-03-08 16:43:33 +00:00
|
|
|
stream.writeString(x->_key);
|
|
|
|
stream.writeByte('=');
|
|
|
|
stream.writeString(x->_value);
|
|
|
|
stream.writeByte('\n');
|
2004-03-28 20:28:45 +00:00
|
|
|
}
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
2007-03-08 16:43:33 +00:00
|
|
|
stream.writeByte('\n');
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
const ConfigManager::Domain *ConfigManager::getDomain(const String &domName) const {
|
|
|
|
assert(!domName.empty());
|
|
|
|
assert(isValidDomainName(domName));
|
2006-10-06 19:01:39 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (domName == kTransientDomain)
|
|
|
|
return &_transientDomain;
|
|
|
|
if (domName == kApplicationDomain)
|
|
|
|
return &_appDomain;
|
|
|
|
if (_gameDomains.contains(domName))
|
|
|
|
return &_gameDomains[domName];
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConfigManager::Domain *ConfigManager::getDomain(const String &domName) {
|
|
|
|
assert(!domName.empty());
|
|
|
|
assert(isValidDomainName(domName));
|
2006-10-06 19:01:39 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (domName == kTransientDomain)
|
|
|
|
return &_transientDomain;
|
|
|
|
if (domName == kApplicationDomain)
|
|
|
|
return &_appDomain;
|
|
|
|
if (_gameDomains.contains(domName))
|
|
|
|
return &_gameDomains[domName];
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
bool ConfigManager::hasKey(const String &key) const {
|
|
|
|
// Search the domains in the following order:
|
2006-04-15 13:12:03 +00:00
|
|
|
// 1) the transient domain,
|
|
|
|
// 2) the active game domain (if any),
|
|
|
|
// 3) the application domain.
|
2003-10-08 21:59:23 +00:00
|
|
|
// The defaults domain is explicitly *not* checked.
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2003-11-10 23:17:11 +00:00
|
|
|
if (_transientDomain.contains(key))
|
|
|
|
return true;
|
2003-11-07 16:54:58 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (_activeDomain && _activeDomain->contains(key))
|
2003-10-08 21:59:23 +00:00
|
|
|
return true;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (_appDomain.contains(key))
|
|
|
|
return true;
|
2003-11-07 16:54:58 +00:00
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
bool ConfigManager::hasKey(const String &key, const String &domName) const {
|
|
|
|
// FIXME: For now we continue to allow empty domName to indicate
|
|
|
|
// "use 'default' domain". This is mainly needed for the SCUMM ConfigDialog
|
|
|
|
// and should be removed ASAP.
|
|
|
|
if (domName.empty())
|
|
|
|
return hasKey(key);
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
const Domain *domain = getDomain(domName);
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (!domain)
|
|
|
|
return false;
|
|
|
|
return domain->contains(key);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
void ConfigManager::removeKey(const String &key, const String &domName) {
|
|
|
|
Domain *domain = getDomain(domName);
|
2003-11-05 00:57:00 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (!domain)
|
|
|
|
error("ConfigManager::removeKey(%s, %s) called on non-existent domain",
|
|
|
|
key.c_str(), domName.c_str());
|
|
|
|
|
|
|
|
domain->erase(key);
|
2003-11-05 00:57:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
const String & ConfigManager::get(const String &key) const {
|
|
|
|
if (_transientDomain.contains(key))
|
|
|
|
return _transientDomain[key];
|
|
|
|
else if (_activeDomain && _activeDomain->contains(key))
|
|
|
|
return (*_activeDomain)[key];
|
|
|
|
else if (_appDomain.contains(key))
|
|
|
|
return _appDomain[key];
|
|
|
|
else if (_defaultsDomain.contains(key))
|
|
|
|
return _defaultsDomain[key];
|
2005-02-22 18:11:36 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
#if !(defined(PALMOS_ARM) || defined(PALMOS_DEBUG) || defined(__GP32__))
|
|
|
|
return String::emptyString;
|
|
|
|
#else
|
|
|
|
return ConfMan._emptyString;
|
|
|
|
#endif
|
|
|
|
}
|
2003-10-08 21:59:23 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
const String & ConfigManager::get(const String &key, const String &domName) const {
|
|
|
|
// FIXME: For now we continue to allow empty domName to indicate
|
|
|
|
// "use 'default' domain". This is mainly needed for the SCUMM ConfigDialog
|
|
|
|
// and should be removed ASAP.
|
|
|
|
if (domName.empty())
|
|
|
|
return get(key);
|
2003-11-10 23:17:11 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
const Domain *domain = getDomain(domName);
|
2003-11-10 23:17:11 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (!domain)
|
|
|
|
error("ConfigManager::get(%s,%s) called on non-existent domain",
|
|
|
|
key.c_str(), domName.c_str());
|
2003-11-07 15:57:36 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if (domain->contains(key))
|
|
|
|
return (*domain)[key];
|
2003-11-08 22:43:46 +00:00
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
return _defaultsDomain.get(key);
|
2006-04-15 13:12:03 +00:00
|
|
|
|
|
|
|
if (!domain->contains(key)) {
|
|
|
|
#if 1
|
|
|
|
#if !(defined(PALMOS_ARM) || defined(PALMOS_DEBUG) || defined(__GP32__))
|
|
|
|
return String::emptyString;
|
|
|
|
#else
|
|
|
|
return ConfMan._emptyString;
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
error("ConfigManager::get(%s,%s) called on non-existent key",
|
|
|
|
key.c_str(), domName.c_str());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return (*domain)[key];
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
int ConfigManager::getInt(const String &key, const String &domName) const {
|
|
|
|
String value(get(key, domName));
|
2004-03-28 18:29:27 +00:00
|
|
|
char *errpos;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-03-31 18:00:46 +00:00
|
|
|
// For now, be tolerant against missing config keys. Strictly spoken, it is
|
|
|
|
// a bug in the calling code to retrieve an int for a key which isn't even
|
|
|
|
// present... and a default value of 0 seems rather arbitrary.
|
2006-03-28 09:42:54 +00:00
|
|
|
if (value.empty())
|
2004-03-31 18:00:46 +00:00
|
|
|
return 0;
|
|
|
|
|
2006-07-08 20:20:07 +00:00
|
|
|
// We use the special value '0' for the base passed to strtol. Doing that
|
2006-04-15 13:58:01 +00:00
|
|
|
// makes it possible to enter hex values as "0x1234", but also decimal
|
|
|
|
// values ("123") are still valid.
|
|
|
|
int ivalue = (int)strtol(value.c_str(), &errpos, 0);
|
2004-03-28 18:29:27 +00:00
|
|
|
if (value.c_str() == errpos)
|
2006-04-15 13:12:03 +00:00
|
|
|
error("ConfigManager::getInt(%s,%s): '%s' is not a valid integer",
|
|
|
|
key.c_str(), domName.c_str(), errpos);
|
2004-03-28 18:29:27 +00:00
|
|
|
|
|
|
|
return ivalue;
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
bool ConfigManager::getBool(const String &key, const String &domName) const {
|
|
|
|
String value(get(key, domName));
|
2004-03-28 18:29:27 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
if ((value == "true") || (value == "yes") || (value == "1"))
|
2004-03-28 18:29:27 +00:00
|
|
|
return true;
|
2006-04-15 13:12:03 +00:00
|
|
|
if ((value == "false") || (value == "no") || (value == "0"))
|
2004-03-28 18:29:27 +00:00
|
|
|
return false;
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
error("ConfigManager::getBool(%s,%s): '%s' is not a valid bool",
|
|
|
|
key.c_str(), domName.c_str(), value.c_str());
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
void ConfigManager::set(const String &key, const String &value) {
|
|
|
|
// Remove the transient domain value, if any.
|
|
|
|
_transientDomain.erase(key);
|
2003-10-09 11:42:59 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
// Write the new key/value pair into the active domain, resp. into
|
|
|
|
// the application domain if no game domain is active.
|
|
|
|
if (_activeDomain)
|
|
|
|
(*_activeDomain)[key] = value;
|
|
|
|
else
|
|
|
|
_appDomain[key] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::set(const String &key, const String &value, const String &domName) {
|
|
|
|
// FIXME: For now we continue to allow empty domName to indicate
|
|
|
|
// "use 'default' domain". This is mainly needed for the SCUMM ConfigDialog
|
|
|
|
// and should be removed ASAP.
|
2006-04-21 20:59:05 +00:00
|
|
|
if (domName.empty()) {
|
|
|
|
set(key, value);
|
|
|
|
return;
|
|
|
|
}
|
2006-04-15 13:12:03 +00:00
|
|
|
|
|
|
|
Domain *domain = getDomain(domName);
|
|
|
|
|
|
|
|
if (!domain)
|
|
|
|
error("ConfigManager::set(%s,%s,%s) called on non-existent domain",
|
|
|
|
key.c_str(), value.c_str(), domName.c_str());
|
|
|
|
|
|
|
|
(*domain)[key] = value;
|
2006-10-06 19:01:39 +00:00
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
// TODO/FIXME: We used to erase the given key from the transient domain
|
|
|
|
// here. Do we still want to do that?
|
|
|
|
// It was probably there to simplify the options dialogs code:
|
|
|
|
// Imagine you are editing the current options (via the SCUMM ConfigDialog,
|
|
|
|
// for example). If you edit the game domain for that, but a matching
|
|
|
|
// entry in the transient domain is present, than your changes may not take
|
|
|
|
// effect. So you want to remove the key from the transient domain before
|
|
|
|
// adding it to the active domain.
|
|
|
|
// But doing this here seems rather evil... need to comb the options dialog
|
|
|
|
// code to find out if it's still necessary, and if that's the case, how
|
|
|
|
// to replace it in a clean fashion...
|
|
|
|
/*
|
|
|
|
if (domName == kTransientDomain)
|
|
|
|
_transientDomain[key] = value;
|
|
|
|
else {
|
|
|
|
if (domName == kApplicationDomain) {
|
|
|
|
_appDomain[key] = value;
|
|
|
|
if (_activeDomainName.empty() || !_gameDomains[_activeDomainName].contains(key))
|
|
|
|
_transientDomain.erase(key);
|
|
|
|
} else {
|
|
|
|
_gameDomains[domName][key] = value;
|
|
|
|
if (domName == _activeDomainName)
|
|
|
|
_transientDomain.erase(key);
|
2003-11-10 23:17:11 +00:00
|
|
|
}
|
|
|
|
}
|
2006-04-15 13:12:03 +00:00
|
|
|
*/
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-16 19:23:14 +00:00
|
|
|
void ConfigManager::setInt(const String &key, int value, const String &domName) {
|
2003-10-08 21:59:23 +00:00
|
|
|
char tmp[128];
|
|
|
|
snprintf(tmp, sizeof(tmp), "%i", value);
|
2006-04-15 13:12:03 +00:00
|
|
|
set(key, String(tmp), domName);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-16 19:23:14 +00:00
|
|
|
void ConfigManager::setBool(const String &key, bool value, const String &domName) {
|
2006-04-15 13:12:03 +00:00
|
|
|
set(key, String(value ? "true" : "false"), domName);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
void ConfigManager::registerDefault(const String &key, const String &value) {
|
|
|
|
_defaultsDomain[key] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::registerDefault(const String &key, const char *value) {
|
|
|
|
registerDefault(key, String(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::registerDefault(const String &key, int value) {
|
|
|
|
char tmp[128];
|
|
|
|
snprintf(tmp, sizeof(tmp), "%i", value);
|
|
|
|
registerDefault(key, tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::registerDefault(const String &key, bool value) {
|
2006-04-15 13:12:03 +00:00
|
|
|
registerDefault(key, value ? "true" : "false");
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
void ConfigManager::setActiveDomain(const String &domName) {
|
|
|
|
if (domName.empty()) {
|
|
|
|
_activeDomain = 0;
|
|
|
|
} else {
|
|
|
|
assert(isValidDomainName(domName));
|
|
|
|
_activeDomain = & _gameDomains[domName];
|
|
|
|
}
|
|
|
|
_activeDomainName = domName;
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-16 13:58:11 +00:00
|
|
|
void ConfigManager::addGameDomain(const String &domName) {
|
|
|
|
assert(!domName.empty());
|
|
|
|
assert(isValidDomainName(domName));
|
|
|
|
|
|
|
|
// TODO: Do we want to generate an error/warning if a domain with
|
|
|
|
// the given name already exists?
|
|
|
|
|
|
|
|
_gameDomains[domName];
|
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
void ConfigManager::removeGameDomain(const String &domName) {
|
|
|
|
assert(!domName.empty());
|
|
|
|
assert(isValidDomainName(domName));
|
|
|
|
_gameDomains.erase(domName);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::renameGameDomain(const String &oldName, const String &newName) {
|
|
|
|
if (oldName == newName)
|
|
|
|
return;
|
|
|
|
|
2006-03-28 09:42:54 +00:00
|
|
|
assert(!oldName.empty());
|
|
|
|
assert(!newName.empty());
|
2005-02-22 18:11:36 +00:00
|
|
|
assert(isValidDomainName(oldName));
|
|
|
|
assert(isValidDomainName(newName));
|
2003-10-09 11:42:59 +00:00
|
|
|
|
2006-03-31 07:30:43 +00:00
|
|
|
// _gameDomains[newName].merge(_gameDomains[oldName]);
|
2006-04-04 21:32:19 +00:00
|
|
|
Domain &oldDom = _gameDomains[oldName];
|
|
|
|
Domain &newDom = _gameDomains[newName];
|
2006-03-31 07:30:43 +00:00
|
|
|
Domain::const_iterator iter;
|
|
|
|
for (iter = oldDom.begin(); iter != oldDom.end(); ++iter)
|
|
|
|
newDom[iter->_key] = iter->_value;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2006-03-28 12:35:50 +00:00
|
|
|
_gameDomains.erase(oldName);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2006-04-15 13:12:03 +00:00
|
|
|
bool ConfigManager::hasGameDomain(const String &domName) const {
|
|
|
|
assert(!domName.empty());
|
|
|
|
return isValidDomainName(domName) && _gameDomains.contains(domName);
|
2003-10-08 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
2004-03-29 19:15:23 +00:00
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
const String &ConfigManager::Domain::get(const String &key) const {
|
2006-03-28 12:35:50 +00:00
|
|
|
const_iterator iter(find(key));
|
|
|
|
if (iter != end())
|
|
|
|
return iter->_value;
|
|
|
|
|
2005-11-27 02:35:57 +00:00
|
|
|
#if !(defined(PALMOS_ARM) || defined(PALMOS_DEBUG) || defined(__GP32__))
|
2006-03-28 12:35:50 +00:00
|
|
|
return String::emptyString;
|
2005-10-08 19:10:05 +00:00
|
|
|
#else
|
2006-03-28 12:35:50 +00:00
|
|
|
return ConfMan._emptyString;
|
2005-10-08 19:10:05 +00:00
|
|
|
#endif
|
2004-03-29 19:15:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::Domain::setDomainComment(const String &comment) {
|
|
|
|
_domainComment = comment;
|
|
|
|
}
|
|
|
|
const String &ConfigManager::Domain::getDomainComment() const {
|
|
|
|
return _domainComment;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigManager::Domain::setKVComment(const String &key, const String &comment) {
|
|
|
|
_keyValueComments[key] = comment;
|
|
|
|
}
|
|
|
|
const String &ConfigManager::Domain::getKVComment(const String &key) const {
|
|
|
|
return _keyValueComments[key];
|
|
|
|
}
|
|
|
|
bool ConfigManager::Domain::hasKVComment(const String &key) const {
|
|
|
|
return _keyValueComments.contains(key);
|
|
|
|
}
|
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
} // End of namespace Common
|