/* * The Mana Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * * This file is part of The Mana Client. * * 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 * 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, see . */ #include "configuration.h" #include "configlistener.h" #include "log.h" #include "utils/stringutils.h" #include "utils/xml.h" void ConfigurationObject::setValue(const std::string &key, const std::string &value) { mOptions[key] = value; } void Configuration::setValue(const std::string &key, const std::string &value) { ConfigurationObject::setValue(key, value); // Notify listeners ListenerMapIterator list = mListenerMap.find(key); if (list != mListenerMap.end()) { Listeners listeners = list->second; for (ListenerIterator i = listeners.begin(); i != listeners.end(); i++) (*i)->optionChanged(key); } } std::string ConfigurationObject::getValue(const std::string &key, const std::string &deflt) const { Options::const_iterator iter = mOptions.find(key); return ((iter != mOptions.end()) ? iter->second : deflt); } int ConfigurationObject::getValue(const std::string &key, int deflt) const { Options::const_iterator iter = mOptions.find(key); return (iter != mOptions.end()) ? atoi(iter->second.c_str()) : deflt; } unsigned ConfigurationObject::getValue(const std::string &key, unsigned deflt) const { Options::const_iterator iter = mOptions.find(key); return (iter != mOptions.end()) ? atol(iter->second.c_str()) : deflt; } double ConfigurationObject::getValue(const std::string &key, double deflt) const { Options::const_iterator iter = mOptions.find(key); return (iter != mOptions.end()) ? atof(iter->second.c_str()) : deflt; } void ConfigurationObject::deleteList(const std::string &name) { for (ConfigurationList::const_iterator it = mContainerOptions[name].begin(); it != mContainerOptions[name].end(); it++) { delete *it; } mContainerOptions[name].clear(); } void ConfigurationObject::clear() { for (std::map::const_iterator it = mContainerOptions.begin(); it != mContainerOptions.end(); it++) { deleteList(it->first); } mOptions.clear(); } ConfigurationObject::~ConfigurationObject() { clear(); } void Configuration::cleanDefaults() { if (mDefaultsData) { for (DefaultsData::const_iterator iter = mDefaultsData->begin(); iter != mDefaultsData->end(); iter++) { if (iter->second) delete(iter->second); } mDefaultsData->clear(); delete mDefaultsData; mDefaultsData = 0; } } Configuration::~Configuration() { cleanDefaults(); } void Configuration::setDefaultValues(DefaultsData *defaultsData) { cleanDefaults(); mDefaultsData = defaultsData; } Mana::VariableData* Configuration::getDefault(const std::string &key, Mana::VariableData::DataType type ) const { if (mDefaultsData) { DefaultsData::const_iterator itdef = mDefaultsData->find(key); if (itdef != mDefaultsData->end() && itdef->second && itdef->second->getType() == type) { return itdef->second; } else { logger->log("%s: No value in registry for key %s", mConfigPath.c_str(), key.c_str()); } } return NULL; } int Configuration::getIntValue(const std::string &key) const { int defaultValue = 0; Options::const_iterator iter = mOptions.find(key); if (iter == mOptions.end()) { Mana::VariableData* vd = getDefault(key, Mana::VariableData::DATA_INT); if (vd) defaultValue = ((Mana::IntData*)vd)->getData(); } else { defaultValue = atoi(iter->second.c_str()); } return defaultValue; } std::string Configuration::getStringValue(const std::string &key) const { std::string defaultValue = ""; Options::const_iterator iter = mOptions.find(key); if (iter == mOptions.end()) { Mana::VariableData* vd = getDefault(key, Mana::VariableData::DATA_STRING); if (vd) defaultValue = ((Mana::StringData*)vd)->getData(); } else { defaultValue = iter->second; } return defaultValue; } float Configuration::getFloatValue(const std::string &key) const { float defaultValue = 0.0f; Options::const_iterator iter = mOptions.find(key); if (iter == mOptions.end()) { Mana::VariableData* vd = getDefault(key, Mana::VariableData::DATA_FLOAT); if (vd) defaultValue = ((Mana::FloatData*)vd)->getData(); } else { defaultValue = atof(iter->second.c_str()); } return defaultValue; } bool Configuration::getBoolValue(const std::string &key) const { bool defaultValue = false; Options::const_iterator iter = mOptions.find(key); if (iter == mOptions.end()) { Mana::VariableData* vd = getDefault(key, Mana::VariableData::DATA_BOOL); if (vd) defaultValue = ((Mana::BoolData*)vd)->getData(); } else { defaultValue = getBoolFromString(iter->second); } return defaultValue; } void ConfigurationObject::initFromXML(xmlNodePtr parent_node) { clear(); for_each_xml_child_node(node, parent_node) { if (xmlStrEqual(node->name, BAD_CAST "list")) { // List option handling. std::string name = XML::getProperty(node, "name", std::string()); for_each_xml_child_node(subnode, node) { if (xmlStrEqual(subnode->name, BAD_CAST name.c_str()) && subnode->type == XML_ELEMENT_NODE) { ConfigurationObject *cobj = new ConfigurationObject; cobj->initFromXML(subnode); // Recurse mContainerOptions[name].push_back(cobj); } } } else if (xmlStrEqual(node->name, BAD_CAST "option")) { // Single option handling. std::string name = XML::getProperty(node, "name", std::string()); std::string value = XML::getProperty(node, "value", std::string()); if (!name.empty()) mOptions[name] = value; } // Otherwise ignore } } void Configuration::init(const std::string &filename, bool useResManager) { XML::Document doc(filename, useResManager); if (useResManager) mConfigPath = "PhysFS://" + filename; else mConfigPath = filename; if (!doc.rootNode()) { logger->log("Couldn't open configuration file: %s", filename.c_str()); return; } xmlNodePtr rootNode = doc.rootNode(); if (!rootNode || !xmlStrEqual(rootNode->name, BAD_CAST "configuration")) { logger->log("Warning: No configuration file (%s)", filename.c_str()); return; } initFromXML(rootNode); } void ConfigurationObject::writeToXML(xmlTextWriterPtr writer) { for (Options::const_iterator i = mOptions.begin(), i_end = mOptions.end(); i != i_end; ++i) { xmlTextWriterStartElement(writer, BAD_CAST "option"); xmlTextWriterWriteAttribute(writer, BAD_CAST "name", BAD_CAST i->first.c_str()); xmlTextWriterWriteAttribute(writer, BAD_CAST "value", BAD_CAST i->second.c_str()); xmlTextWriterEndElement(writer); } for (std::map::const_iterator it = mContainerOptions.begin(); it != mContainerOptions.end(); it++) { const char *name = it->first.c_str(); xmlTextWriterStartElement(writer, BAD_CAST "list"); xmlTextWriterWriteAttribute(writer, BAD_CAST "name", BAD_CAST name); // Recurse on all elements for (ConfigurationList::const_iterator elt_it = it->second.begin(); elt_it != it->second.end(); elt_it++) { xmlTextWriterStartElement(writer, BAD_CAST name); (*elt_it)->writeToXML(writer); xmlTextWriterEndElement(writer); } xmlTextWriterEndElement(writer); } } void Configuration::write() { // Do not attempt to write to file that cannot be opened for writing FILE *testFile = fopen(mConfigPath.c_str(), "w"); if (!testFile) { logger->log("Configuration::write() couldn't open %s for writing", mConfigPath.c_str()); return; } else { fclose(testFile); } xmlTextWriterPtr writer = xmlNewTextWriterFilename(mConfigPath.c_str(), 0); if (!writer) { logger->log("Configuration::write() error while creating writer"); return; } logger->log("Configuration::write() writing configuration..."); xmlTextWriterSetIndent(writer, 1); xmlTextWriterStartDocument(writer, NULL, NULL, NULL); xmlTextWriterStartElement(writer, BAD_CAST "configuration"); writeToXML(writer); xmlTextWriterEndDocument(writer); xmlFreeTextWriter(writer); } void Configuration::addListener(const std::string &key, ConfigListener *listener) { mListenerMap[key].push_front(listener); } void Configuration::removeListener(const std::string &key, ConfigListener *listener) { mListenerMap[key].remove(listener); }