Code Search for Developers
 
 
  

XMLNode.cpp from Scorched 3D at Krugle


Show XMLNode.cpp syntax highlighted

////////////////////////////////////////////////////////////////////////////////
//    Scorched3D (c) 2000-2003
//
//    This file is part of Scorched3D.
//
//    Scorched3D 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.
//
//    Scorched3D 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 Scorched3D; if not, write to the Free Software
//    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
////////////////////////////////////////////////////////////////////////////////

#include <XML/XMLNode.h>
#include <common/Defines.h>
#include <limits.h>
#include <float.h>
#include <stdlib.h>
#include <stdio.h>
#include <common/NumberParser.h>

void XMLNode::removeSpecialChars(std::string &content, std::string &result)
{
	result = "";
	for (char *c=(char *) content.c_str(); *c; c++)
	{
		char newchar = *c;
		if (*c == '\n') result += "&#10;";
		else if (*c < 32 || *c > 126) result += " ";
		else if (*c == '>') result += "&gt;";
		else if (*c == '<') result += "&lt;";
		else if (*c == '\'') result += "&apos;";
		else if (*c == '"') result += "&quot;";
		else if (*c == '&') result += "&amp;";
		else if (*c == '%') result += "&#37;";
		else result += *c;
	}
}

void XMLNode::addSpecialChars(std::string &content, std::string &result)
{
	result = "";
	for (char *c=(char *) content.c_str(); *c; c++)
	{
		if (*c == '&')
		{
			if (strstr(c, "&gt;") == c) { result += '>'; c+=3; }
			else if (strstr(c, "&lt;") == c) { result += '<'; c+=3; }
			else if (strstr(c, "&apos;") == c) { result += '\''; c+=5; }
			else if (strstr(c, "&quot;") == c) { result += '"'; c+=5; }
			else if (strstr(c, "&amp;") == c) { result += '&'; c+=4; }
			else if (strstr(c, "&#") == c)
			{
				char *pos = strchr(c, ';');
				if (pos <= c + 3)
				{
					c+=2;
					result += formatString("%c", atoi(c));
					*pos = '\0';
					c = pos + 1;
				}
			}
			else result += *c;
		}
		else result += *c;
	}	
}

const char *XMLNode::getSpacer(int space)
{
	static std::string spacestr;
	spacestr = "";

	for (int i=0; i<space; i++) spacestr+="\t";
	return spacestr.c_str();
}

XMLNode::XMLNode(const char *name, const char *content, NodeType type) : 
	name_(name), parent_(0), type_(type),
	useContentNodes_(false)
{
	addContent(content, strlen(content));
}

XMLNode::XMLNode(const char *name, float content, NodeType type) :
	name_(name), parent_(0), type_(type), useContentNodes_(false)
{
	char buffer[20];
	snprintf(buffer, 20, "%.2f", content);
	addContent(buffer, strlen(buffer));
}

XMLNode::XMLNode(const char *name, int content, NodeType type) :
	name_(name), parent_(0), type_(type), useContentNodes_(false)
{
	char buffer[20];
	snprintf(buffer, 20, "%i", content);
	addContent(buffer, strlen(buffer));
}

XMLNode::XMLNode(const char *name, unsigned int content, NodeType type) :
	name_(name), parent_(0), type_(type), useContentNodes_(false)
{
	char buffer[20];
	snprintf(buffer, 20, "%u", content);
	addContent(buffer, strlen(buffer));
}

XMLNode::XMLNode(const char *name, bool content, NodeType type) :
	name_(name), parent_(0), type_(type), useContentNodes_(false)
{
	const char *buffer = content?"true":"false";
	addContent(buffer, strlen(buffer));
}

XMLNode::XMLNode(const char *name, Vector &content, NodeType type) :
	name_(name), parent_(0), type_(type), useContentNodes_(false)
{
	XMLNode *nodeA = new XMLNode("a");
	addChild(nodeA);
	XMLNode *nodeB = new XMLNode("b");
	addChild(nodeB);
	XMLNode *nodeC = new XMLNode("c");
	addChild(nodeC);

	char buffer[20];
	snprintf(buffer, 20, "%.2f", content[0]);
	nodeA->addContent(buffer, strlen(buffer));

	snprintf(buffer, 20, "%.2f", content[1]);
	nodeB->addContent(buffer, strlen(buffer));

	snprintf(buffer, 20, "%.2f", content[2]);
	nodeC->addContent(buffer, strlen(buffer));
}

XMLNode::~XMLNode()
{
	while (!children_.empty())
	{
		XMLNode *node = children_.front();
		children_.pop_front();
		delete node;
	}
	while (!removedChildren_.empty())
	{
		XMLNode *node = removedChildren_.front();
		removedChildren_.pop_front();
		delete node;
	}
	while (!parameters_.empty())
	{
		XMLNode *node = parameters_.front();
		parameters_.pop_front();
		delete node;
	}
	while (!removedParameters_.empty())
	{
		XMLNode *node = removedParameters_.front();
		removedParameters_.pop_front();
		delete node;
	}
}

bool XMLNode::writeToFile(const char *filename)
{
	FileLines lines;
	addNodeToFile(lines, 0);
	return lines.writeFile((char *) filename);
}

void XMLNode::addNodeToFile(FileLines &lines, int spacing)
{
	if (type_ == XMLNodeType)
	{
		std::string params;
		std::list<XMLNode *>::iterator pitor;
		for (pitor = parameters_.begin();
			pitor != parameters_.end();
			pitor++)
		{
			XMLNode *node = (*pitor);
			DIALOG_ASSERT(node->type_ == XMLParameterType);
			
			std::string oldContent(node->getContent());
			std::string newContent;
			removeSpecialChars(oldContent, newContent);
			
			params += " " + node->name_ + "='" + newContent + "'";
		}

		if (children_.empty())
		{
			std::string oldContent(getContent());
			std::string newContent;
			removeSpecialChars(oldContent, newContent);
			
			lines.addLine(formatString("%s<%s%s>%s</%s>", 
				getSpacer(spacing),
				name_.c_str(), params.c_str(), 
				newContent.c_str(), name_.c_str()));
		}
		else
		{
			lines.addLine(formatString("%s<%s%s>", 
				getSpacer(spacing),
				name_.c_str(), params.c_str()));

			std::list<XMLNode *>::iterator itor;
			for (itor = children_.begin();
				itor != children_.end();
				itor++)
			{
				XMLNode *node = (*itor);
				node->addNodeToFile(lines, spacing + 1);
			}

			lines.addLine(formatString("%s</%s>", 
				getSpacer(spacing), name_.c_str()));
		}
	}
	else if (type_ == XMLCommentType)
	{
		lines.addLine(formatString("%s<!-- %s -->", 
			getSpacer(spacing), getContent()));
	}
}

bool XMLNode::failChildren()
{
	if (useContentNodes_)
	{
		std::list<XMLNode *>::iterator itor;
		for (itor = getChildren().begin();
			itor != getChildren().end();
			itor++)
		{
			XMLNode	*node = (*itor);
			if (node->getType() == XMLNodeType)
			{
				node->returnError(formatString("Unrecognised node."));
				return false;
			}
		}
	}
	else
	{
		if (!children_.empty())
		{
			XMLNode *node = children_.front();
			node->returnError(formatString("Unrecognised node."));
			return false;
		}
	}

	return true;
}

bool XMLNode::failContent()
{
	for (const char *c=getContent(); *c; c++)
	{
		if (*c != '\n' &&
			*c != '\r' &&
			*c != '\t' &&
			*c != ' ')
		{
			returnError(formatString("Unexpected context : %s", 
					getContent()));
			return false;
		}
	}
	return true;
}

void XMLNode::resurrectRemovedChildren()
{
	while (!removedChildren_.empty())
	{
		XMLNode *node = removedChildren_.front();
		removedChildren_.pop_front();
		children_.push_back(node);
		
		node->resurrectRemovedChildren();
	}
	while (!removedParameters_.empty())
	{
		XMLNode *node = removedParameters_.front();
		removedParameters_.pop_front();
		parameters_.push_back(node);
	}
}

bool XMLNode::getNamedChild(const char *name, XMLNode *&value,
	bool failOnError, bool remove)
{
	std::list<XMLNode *>::iterator itor;
	for (itor = children_.begin();
		itor != children_.end();
		itor++)
	{
		XMLNode *node = (*itor);
		if (strcmp(name, node->getName()) == 0) 
		{
			if (remove)
			{
				removedChildren_.push_back(node);
				children_.erase(itor);
			}
			value = node;
			return true;
		}
	}

	if (failOnError)
	{
		returnError(formatString("Failed to find \"%s\" node", name));
	}
	return false;
}

bool XMLNode::getNamedParameter(const char *name, XMLNode *&value,
	bool failOnError, bool remove)
{
	std::list<XMLNode *>::iterator itor;
	for (itor = parameters_.begin();
		itor != parameters_.end();
		itor++)
	{
		XMLNode *node = (*itor);
		if (strcmp(name, node->getName()) == 0)
		{
			if (remove)
			{
				removedParameters_.push_back(node);
				parameters_.erase(itor);
			}
			value = node;
			return true;
		}
	}

	if (failOnError)
	{
		returnError(formatString("Failed to find \"%s\" parameter", name));
	}
	return false;
}

bool XMLNode::getNamedParameter(const char *name, std::string &value,
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedParameter(name, node, failOnError, remove)) return false;
	value = node->getContent();
	return true;
}

bool XMLNode::getNamedChild(const char *name, std::string &value,
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedChild(name, node, failOnError, remove)) return false;
	value = node->getContent();
	return true;
}

bool XMLNode::getNamedChild(const char *name, bool &value,
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedChild(name, node, failOnError, remove)) return false;

	if (0 == strcmp(node->getContent(), "true")) value = true;
	else if (0 == strcmp(node->getContent(), "false")) value = false;
	else 
	{
		return node->returnError(
			"Failed to parse boolean value (should be true or false)");
	}
	return true;
}

bool XMLNode::getNamedChild(const char *name, NumberParser &value,
        bool failOnError, bool remove)
{
        XMLNode *node;
        if (!getNamedChild(name, node, failOnError, remove)) return false;

        if (!value.setExpression(node->getContent()))
                return node->returnError("Failed to parse expression");

        return true;
}

bool XMLNode::getNamedChild(const char *name, float &value,
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedChild(name, node, failOnError, remove)) return false;

	if (sscanf(node->getContent(), "%f", &value) != 1) 
		return node->returnError("Failed to parse float value");
	return true;
}

bool XMLNode::getNamedChild(const char *name, int &value,
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedChild(name, node, failOnError, remove)) return false;

	if (sscanf(node->getContent(), "%i", &value) != 1)
		return node->returnError("Failed to parse int value");
	return true;
}

bool XMLNode::getNamedChild(const char *name, unsigned int &value,
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedChild(name, node, failOnError, remove)) return false;

	if (sscanf(node->getContent(), "%u", &value) != 1)
		return node->returnError("Failed to parse unsigned int value");
	return true;
}

bool XMLNode::getNamedChild(const char *name, Vector &value, 
	bool failOnError, bool remove)
{
	XMLNode *node;
	if (!getNamedChild(name, node, failOnError, remove)) return false;

	Vector tmpValue;
	if (!node->getNamedChild("A", tmpValue[0], false, true) &&
		!node->getNamedChild("a", tmpValue[0], false, true))
	{
		if (failOnError) node->returnError("Failed to find a node");
		return false;
	}
	if (!node->getNamedChild("B", tmpValue[1], false, true) &&
		!node->getNamedChild("b", tmpValue[1], false, true))
	{
		if (failOnError) node->returnError("Failed to find b node");
		return false;
	}
	if (!node->getNamedChild("C", tmpValue[2], false, true) &&
		!node->getNamedChild("c", tmpValue[2], false, true))
	{
		if (failOnError) node->returnError("Failed to find c node");
		return false;
	}
	if (failOnError && !node->failChildren()) return false;

	value = tmpValue;
	return true;
}

const char *XMLNode::getContent()
{
	if (useContentNodes_ &&
		getType() == XMLNodeType)
	{
		static std::string result;

		result = "";
		std::list<XMLNode *>::iterator itor;
		for (itor = getChildren().begin();
			itor != getChildren().end();
			itor++)
		{
			XMLNode *node = (*itor);
			if (node->getType() == XMLContentType)
			{
				result += node->content_.c_str();
			}
		}
		return result.c_str();
	}
	else
	{
		return content_.c_str();
	}
}

void XMLNode::setSource(const char *source)
{ 
	source_ = source; 
}

void XMLNode::setLine(int line, int col)
{
	line_ = line;
	col_ = col;
}

void XMLNode::addChild(XMLNode *node) 
{ 
	children_.push_back(node); 
	node->setUseContentNodes(useContentNodes_);
	node->parent_ = this; 
	node->source_ = source_; 
}

void XMLNode::addParameter(XMLNode *node) 
{ 
	parameters_.push_back(node); 
	node->parent_ = this; 
	node->source_ = source_; 
}

void XMLNode::addContent(const char *data, int len)
{
	std::string oldContent, newContent;
	oldContent.append(data, len);
	addSpecialChars(oldContent, newContent);

	if (useContentNodes_)
	{
		XMLNode *newNode = 
			new XMLNode("__TEXT__", "", XMLNode::XMLContentType);
		newNode->setLine(line_, col_);
		newNode->content_.append(newContent);
		addChild(newNode);
	}
	else
	{
		content_.append(newContent); 
	}
}

bool XMLNode::returnError(const char *error)
{
	dialogMessage("XMLNode",
		formatString("Parse Error, File:%s Line:%i Col:%i Node:%s Error:%s",
		source_.c_str(), line_, col_, getName(), error));
	return false;
}




See more files for this project here

Scorched 3D

Scorched3D is a 3D remake of the popular 2D artillery game Scorched Earth.\r\nScorched3D can be played against the computer, other players and remotely across the internet or LAN.

Project homepage: http://sourceforge.net/projects/scorched3d
Programming language(s): C,C++,XML
License: gpl2

  XMLFile.cpp
  XMLFile.h
  XMLNode.cpp
  XMLNode.h
  XMLParser.cpp
  XMLParser.h
  XMLStringBuffer.cpp
  XMLStringBuffer.h
  ascii.h
  asciitab.h
  expat.h
  expat_config.h
  iasciitab.h
  internal.h
  latin1tab.h
  nametab.h
  utf8tab.h
  winconfig.h
  xmlparse.c
  xmlrole.c
  xmlrole.h
  xmltok.c
  xmltok.h
  xmltok_impl.c
  xmltok_impl.h
  xmltok_ns.c