2008-06-18 23:49:47 +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.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef XML_PARSER_H
|
|
|
|
#define XML_PARSER_H
|
|
|
|
|
|
|
|
#include "common/scummsys.h"
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "common/xmlparser.h"
|
2008-06-25 11:34:58 +00:00
|
|
|
#include "common/stream.h"
|
2008-06-25 14:19:56 +00:00
|
|
|
#include "common/file.h"
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
#include "common/hashmap.h"
|
|
|
|
#include "common/hash-str.h"
|
|
|
|
#include "common/stack.h"
|
|
|
|
|
|
|
|
namespace Common {
|
2008-07-30 13:33:32 +00:00
|
|
|
|
|
|
|
/***********************************************
|
|
|
|
**** XMLParser.cpp/h -- Generic XML Parser ****
|
|
|
|
***********************************************
|
|
|
|
|
|
|
|
This is a simple implementation of a generic parser which is able to
|
|
|
|
interpret a subset of the XML language.
|
|
|
|
|
|
|
|
The XMLParser class is virtual, and must be derived into a child class,
|
|
|
|
called a Custom Parser Class, which will manage the parsed data for your
|
|
|
|
specific needs.
|
|
|
|
|
|
|
|
Custom Parser Classes have two basic requirements:
|
|
|
|
They must inherit directly the XMLParser class, and they must define the
|
|
|
|
parsing layout of the XML file.
|
|
|
|
|
|
|
|
Declaring the XML layout is done with the help of the CUSTOM_XML_PARSER()
|
|
|
|
macro: this macro must appear once inside the Custom Parser Class declaration,
|
|
|
|
and takes a single parameter, the name of the Custom Parser Class.
|
|
|
|
|
|
|
|
The macro must be followed by the actual layout of the XML files to be parsed,
|
|
|
|
and closed with the PARSER_END() macro. The layout of XML files is defined by
|
|
|
|
the use of 3 helper macros: XML_KEY(), KEY_END() and XML_PROP().
|
|
|
|
|
|
|
|
Here's a sample of its usage:
|
|
|
|
|
|
|
|
=========== =========== =========== =========== =========== ===========
|
|
|
|
|
|
|
|
CUSTOM_XML_PARSER(ThemeParser) {
|
|
|
|
XML_KEY(render_info)
|
|
|
|
XML_KEY(palette)
|
|
|
|
XML_KEY(color)
|
|
|
|
XML_PROP(name, true)
|
|
|
|
XML_PROP(rgb, true)
|
|
|
|
KEY_END()
|
|
|
|
KEY_END()
|
|
|
|
|
|
|
|
XML_KEY(fonts)
|
|
|
|
XML_KEY(font)
|
|
|
|
XML_PROP(id, true)
|
|
|
|
XML_PROP(type, true)
|
|
|
|
XML_PROP(color, true)
|
|
|
|
KEY_END()
|
|
|
|
KEY_END()
|
|
|
|
|
|
|
|
XML_KEY(defaults)
|
|
|
|
XML_PROP(stroke, false)
|
|
|
|
XML_PROP(shadow, false)
|
|
|
|
XML_PROP(factor, false)
|
|
|
|
XML_PROP(fg_color, false)
|
|
|
|
XML_PROP(bg_color, false)
|
|
|
|
XML_PROP(gradient_start, false)
|
|
|
|
XML_PROP(gradient_end, false)
|
|
|
|
XML_PROP(gradient_factor, false)
|
|
|
|
XML_PROP(fill, false)
|
|
|
|
KEY_END()
|
|
|
|
KEY_END()
|
|
|
|
} PARSER_END()
|
|
|
|
|
|
|
|
=========== =========== =========== =========== =========== ===========
|
|
|
|
|
|
|
|
The XML_KEY() macro takes a single argument, the name of the expected key.
|
|
|
|
Inside the scope of each key, you may define properties for the given key
|
|
|
|
with the XML_PROP() macro, which takes as parameters the name of the property
|
|
|
|
and whether it's optional or required. You might also define the contained
|
|
|
|
children keys, using the XML_KEY() macro again.
|
|
|
|
The scope of a XML key is closed with the KEY_END() macro.
|
|
|
|
|
|
|
|
As an example, the following XML layout:
|
|
|
|
|
|
|
|
XML_KEY(palette)
|
|
|
|
XML_KEY(color)
|
|
|
|
XML_PROP(name, true)
|
|
|
|
XML_PROP(rgb, true)
|
|
|
|
XML_PROP(optional_param, false)
|
|
|
|
KEY_END()
|
|
|
|
KEY_END()
|
|
|
|
|
|
|
|
will expect to parse a syntax like this:
|
|
|
|
|
|
|
|
<palette>
|
|
|
|
<color name = "red" rgb = "255, 0, 0" />
|
|
|
|
<color name = "blue" rgb = "0, 0, 255" optional_param = "565" />
|
|
|
|
</palette>
|
|
|
|
|
|
|
|
TODO: documentation on callbacks
|
|
|
|
|
|
|
|
Note that the XML parser doesn't take into account the actual order of the keys and
|
|
|
|
properties in the XML layout definition, only its layout and relationships.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define XML_KEY(keyName) {\
|
|
|
|
lay = new XMLKeyLayout; \
|
|
|
|
lay->custom = new kLocalParserName::CustomParserCallback; \
|
|
|
|
((kLocalParserName::CustomParserCallback*)(lay->custom))->callback = (&kLocalParserName::parserCallback_##keyName); \
|
|
|
|
layout.top()->children[#keyName] = lay; \
|
|
|
|
layout.push(lay);
|
|
|
|
|
|
|
|
#define KEY_END() layout.pop(); }
|
|
|
|
|
|
|
|
#define XML_PROP(propName, req) {\
|
|
|
|
prop.name = #propName; \
|
|
|
|
prop.required = req; \
|
|
|
|
layout.top()->properties.push_back(prop); }\
|
|
|
|
|
|
|
|
#define CUSTOM_XML_PARSER(parserName) \
|
|
|
|
protected: \
|
|
|
|
typedef bool (parserName::*ParserCallback)(ParserNode *node); \
|
|
|
|
typedef parserName kLocalParserName; \
|
|
|
|
struct CustomParserCallback { ParserCallback callback; }; \
|
|
|
|
bool keyCallback(ParserNode *node) {return (this->*(((parserName::CustomParserCallback*)(node->layout->custom))->callback))(node);}\
|
|
|
|
virtual void buildLayout() { \
|
|
|
|
Common::Stack<XMLKeyLayout*> layout; \
|
|
|
|
XMLKeyLayout *lay = 0; \
|
|
|
|
XMLKeyLayout::XMLKeyProperty prop; \
|
|
|
|
_XMLkeys = new XMLKeyLayout; \
|
|
|
|
layout.push(_XMLkeys);
|
|
|
|
|
|
|
|
#define PARSER_END() layout.clear(); }
|
2008-06-18 23:49:47 +00:00
|
|
|
|
2008-06-25 11:34:58 +00:00
|
|
|
class XMLStream {
|
|
|
|
protected:
|
|
|
|
SeekableReadStream *_stream;
|
|
|
|
int _pos;
|
|
|
|
|
|
|
|
public:
|
|
|
|
XMLStream() : _stream(0), _pos(0) {}
|
|
|
|
|
|
|
|
~XMLStream() {
|
|
|
|
delete _stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
SeekableReadStream *stream() {
|
|
|
|
return _stream;
|
|
|
|
}
|
|
|
|
|
2008-06-28 00:02:54 +00:00
|
|
|
char operator [](int idx) {
|
2008-06-25 11:34:58 +00:00
|
|
|
assert(_stream && idx >= 0);
|
|
|
|
|
|
|
|
if (_pos + 1 != idx)
|
|
|
|
_stream->seek(idx, SEEK_SET);
|
|
|
|
|
|
|
|
_pos = idx;
|
|
|
|
|
2008-07-03 19:42:04 +00:00
|
|
|
return _stream->readByte();
|
2008-06-25 11:34:58 +00:00
|
|
|
}
|
|
|
|
|
2008-06-25 18:11:17 +00:00
|
|
|
void loadStream(SeekableReadStream *s) {
|
2008-06-25 11:34:58 +00:00
|
|
|
delete _stream;
|
2008-06-25 18:11:17 +00:00
|
|
|
_stream = s;
|
2008-06-25 11:34:58 +00:00
|
|
|
}
|
2008-06-25 22:30:28 +00:00
|
|
|
|
|
|
|
bool ready() {
|
|
|
|
return _stream != 0;
|
|
|
|
}
|
2008-06-25 11:34:58 +00:00
|
|
|
};
|
|
|
|
|
2008-06-18 23:49:47 +00:00
|
|
|
/**
|
|
|
|
* The base XMLParser class implements generic functionality for parsing
|
|
|
|
* XML-like files.
|
|
|
|
*
|
|
|
|
* In order to use it, it must be inherited with a child class that implements
|
|
|
|
* the XMLParser::keyCallback() function.
|
|
|
|
*
|
|
|
|
* @see XMLParser::keyCallback()
|
|
|
|
*/
|
|
|
|
class XMLParser {
|
|
|
|
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Parser constructor.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
XMLParser() : _XMLkeys(0) {}
|
2008-06-18 23:49:47 +00:00
|
|
|
|
2008-06-25 14:19:56 +00:00
|
|
|
virtual ~XMLParser() {
|
|
|
|
while (!_activeKey.empty())
|
|
|
|
delete _activeKey.pop();
|
|
|
|
}
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
/** Active state for the parser */
|
|
|
|
enum ParserState {
|
|
|
|
kParserNeedKey,
|
|
|
|
kParserNeedKeyName,
|
|
|
|
|
|
|
|
kParserNeedPropertyName,
|
|
|
|
kParserNeedPropertyOperator,
|
|
|
|
kParserNeedPropertyValue,
|
|
|
|
|
|
|
|
kParserError
|
|
|
|
};
|
2008-07-30 13:33:32 +00:00
|
|
|
|
|
|
|
struct XMLKeyLayout;
|
|
|
|
|
|
|
|
typedef Common::HashMap<Common::String, XMLParser::XMLKeyLayout*, Common::IgnoreCase_Hash, Common::IgnoreCase_EqualTo> ChildMap;
|
|
|
|
|
|
|
|
/** nested struct representing the layout of the XML file */
|
|
|
|
struct XMLKeyLayout {
|
|
|
|
void *custom;
|
|
|
|
struct XMLKeyProperty {
|
|
|
|
Common::String name;
|
|
|
|
bool required;
|
|
|
|
};
|
|
|
|
|
|
|
|
Common::List<XMLKeyProperty> properties;
|
|
|
|
ChildMap children;
|
|
|
|
} *_XMLkeys;
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
/** Struct representing a parsed node */
|
|
|
|
struct ParserNode {
|
|
|
|
Common::String name;
|
|
|
|
Common::StringMap values;
|
2008-06-24 19:48:01 +00:00
|
|
|
bool ignore;
|
2008-06-25 17:56:00 +00:00
|
|
|
int depth;
|
2008-07-30 13:33:32 +00:00
|
|
|
XMLKeyLayout *layout;
|
2008-06-18 23:49:47 +00:00
|
|
|
};
|
|
|
|
|
2008-06-26 13:50:16 +00:00
|
|
|
/**
|
|
|
|
* Loads a file into the parser.
|
|
|
|
* Used for the loading of Theme Description files
|
|
|
|
* straight from the filesystem.
|
|
|
|
*
|
|
|
|
* @param filename Name of the file to load.
|
|
|
|
*/
|
2008-06-25 22:30:28 +00:00
|
|
|
virtual bool loadFile(Common::String filename) {
|
2008-06-25 14:19:56 +00:00
|
|
|
Common::File *f = new Common::File;
|
|
|
|
|
|
|
|
if (!f->open(filename, Common::File::kFileReadMode))
|
|
|
|
return false;
|
|
|
|
|
2008-06-26 13:50:16 +00:00
|
|
|
_fileName = filename;
|
2008-06-25 14:19:56 +00:00
|
|
|
_text.loadStream(f);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-06-26 13:50:16 +00:00
|
|
|
/**
|
|
|
|
* Loads a memory buffer into the parser.
|
|
|
|
* Used for loading the default theme fallback directly
|
|
|
|
* from memory if no themes can be found.
|
|
|
|
*
|
|
|
|
* @param buffer Pointer to the buffer.
|
2008-07-03 16:09:54 +00:00
|
|
|
* @param size Size of the buffer
|
2008-06-26 13:50:16 +00:00
|
|
|
* @param disposable Sets if the XMLParser owns the buffer,
|
|
|
|
* i.e. if it can be freed safely after it's
|
|
|
|
* no longer needed by the parser.
|
|
|
|
*/
|
2008-07-03 16:09:54 +00:00
|
|
|
virtual bool loadBuffer(const byte *buffer, uint32 size, bool disposable = false) {
|
|
|
|
_text.loadStream(new MemoryReadStream(buffer, size, disposable));
|
2008-06-26 13:50:16 +00:00
|
|
|
_fileName = "Memory Stream";
|
2008-06-25 14:19:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-06-26 13:50:16 +00:00
|
|
|
/**
|
|
|
|
* The actual parsing function.
|
|
|
|
* Parses the loaded data stream, returns true if successful.
|
|
|
|
*/
|
2008-06-18 23:49:47 +00:00
|
|
|
virtual bool parse();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the active node being parsed (the one on top of
|
|
|
|
* the node stack).
|
|
|
|
*/
|
2008-06-25 17:56:00 +00:00
|
|
|
ParserNode *getActiveNode() {
|
2008-06-18 23:49:47 +00:00
|
|
|
if (!_activeKey.empty())
|
|
|
|
return _activeKey.top();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-06-25 17:56:00 +00:00
|
|
|
/**
|
|
|
|
* Returns the parent of a given node in the stack.
|
|
|
|
*/
|
|
|
|
ParserNode *getParentNode(ParserNode *child) {
|
|
|
|
return child->depth > 0 ? _activeKey[child->depth - 1] : 0;
|
|
|
|
}
|
|
|
|
|
2008-06-18 23:49:47 +00:00
|
|
|
protected:
|
2008-07-30 13:33:32 +00:00
|
|
|
|
2008-06-18 23:49:47 +00:00
|
|
|
/**
|
2008-07-30 13:33:32 +00:00
|
|
|
* The buildLayout function builds the layout for the parser to use
|
|
|
|
* based on a series of helper macros. This function is automatically
|
|
|
|
* generated by the CUSTOM_XML_PARSER() macro on custom parsers.
|
2008-06-18 23:49:47 +00:00
|
|
|
*
|
2008-07-30 13:33:32 +00:00
|
|
|
* See the documentation regarding XML layouts.
|
|
|
|
*/
|
|
|
|
virtual void buildLayout() = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The keycallback function is automatically overloaded on custom parsers
|
|
|
|
* when using the CUSTOM_XML_PARSER() macro.
|
2008-06-18 23:49:47 +00:00
|
|
|
*
|
2008-07-30 13:33:32 +00:00
|
|
|
* Its job is to call the corresponding Callback function for the given node.
|
|
|
|
* A function for each key type must be declared separately. See the custom
|
|
|
|
* parser creation instructions.
|
2008-06-18 23:49:47 +00:00
|
|
|
*
|
2008-07-30 13:33:32 +00:00
|
|
|
* When parsing a key in such function, one may chose to skip it, e.g. because it's not needed
|
2008-06-24 19:48:01 +00:00
|
|
|
* on the current configuration. In order to ignore a key, you must set
|
|
|
|
* the "ignore" field of its KeyNode struct to "true": The key and all its children
|
|
|
|
* will then be automatically ignored by the parser.
|
|
|
|
*
|
2008-07-30 13:33:32 +00:00
|
|
|
* The callback function must return true if the key was properly handled (this includes the case when the
|
|
|
|
* key is being ignored). False otherwise. The return of keyCallback() is the same as
|
|
|
|
* the callback function's.
|
2008-06-18 23:49:47 +00:00
|
|
|
* See the sample implementation in GUI::ThemeParser.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
virtual bool keyCallback(ParserNode *node) = 0;
|
2008-06-18 23:49:47 +00:00
|
|
|
|
2008-07-04 23:51:23 +00:00
|
|
|
/**
|
2008-07-30 13:33:32 +00:00
|
|
|
* The closed key callback function MAY be overloaded by inheriting classes to
|
2008-07-04 23:51:23 +00:00
|
|
|
* implement parser-specific functions.
|
|
|
|
*
|
|
|
|
* The closedKeyCallback is issued once a key has been finished parsing, to let
|
|
|
|
* the parser verify that all the required subkeys, etc, were included.
|
|
|
|
*
|
|
|
|
* Returns true if the key was properly closed, false otherwise.
|
|
|
|
* By default, all keys are properly closed.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
virtual bool closedKeyCallback(ParserNode *node) {
|
2008-07-04 23:51:23 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-06-18 23:49:47 +00:00
|
|
|
/**
|
|
|
|
* Parses the value of a given key. There's no reason to overload this.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
bool parseKeyValue(Common::String keyName);
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Called once a key has been parsed. It handles the closing/cleanup of the
|
|
|
|
* node stack and calls the keyCallback.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
bool parseActiveKey(bool closed);
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Prints an error message when parsing fails and stops the parser.
|
2008-06-25 14:19:56 +00:00
|
|
|
* Parser error always returns "false" so we can pass the return value directly
|
|
|
|
* and break down the parsing.
|
2008-06-18 23:49:47 +00:00
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
bool parserError(const char *errorString, ...) GCC_PRINTF(2, 3);
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Skips spaces/whitelines etc. Returns true if any spaces were skipped.
|
|
|
|
* Overload this if you want to make your parser depend on newlines or
|
|
|
|
* whatever.
|
|
|
|
*/
|
|
|
|
virtual bool skipSpaces() {
|
|
|
|
if (!isspace(_text[_pos]))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
while (_text[_pos] && isspace(_text[_pos]))
|
|
|
|
_pos++;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Skips comment blocks and comment lines.
|
|
|
|
* Returns true if any comments were skipped.
|
|
|
|
* Overload this if you want to disable comments on your XML syntax
|
|
|
|
* or to change the commenting syntax.
|
|
|
|
*/
|
|
|
|
virtual bool skipComments() {
|
|
|
|
if (_text[_pos] == '/' && _text[_pos + 1] == '*') {
|
|
|
|
_pos += 2;
|
|
|
|
while (_text[_pos++]) {
|
|
|
|
if (_text[_pos - 2] == '*' && _text[_pos - 1] == '/')
|
|
|
|
break;
|
|
|
|
if (_text[_pos] == 0)
|
|
|
|
parserError("Comment has no closure.");
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_text[_pos] == '/' && _text[_pos + 1] == '/') {
|
|
|
|
_pos += 2;
|
2008-06-24 19:48:01 +00:00
|
|
|
while (_text[_pos] && _text[_pos] != '\n' && _text[_pos] != '\r')
|
2008-06-18 23:49:47 +00:00
|
|
|
_pos++;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if a given character can be part of a KEY or VALUE name.
|
|
|
|
* Overload this if you want to support keys with strange characters
|
|
|
|
* in their name.
|
|
|
|
*/
|
|
|
|
virtual bool isValidNameChar(char c) {
|
|
|
|
return isalnum(c) || c == '_';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parses a the first textual token found.
|
|
|
|
* There's no reason to overload this.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
bool parseToken() {
|
2008-06-18 23:49:47 +00:00
|
|
|
_token.clear();
|
|
|
|
while (isValidNameChar(_text[_pos]))
|
|
|
|
_token += _text[_pos++];
|
|
|
|
|
2008-07-04 11:56:31 +00:00
|
|
|
return isspace(_text[_pos]) != 0 || _text[_pos] == '>' || _text[_pos] == '=';
|
2008-06-18 23:49:47 +00:00
|
|
|
}
|
|
|
|
|
2008-06-28 00:02:54 +00:00
|
|
|
/**
|
|
|
|
* Parses the values inside an integer key.
|
|
|
|
* The count parameter specifies the number of values inside
|
|
|
|
* the key, which are expected to be separated with commas.
|
|
|
|
*
|
|
|
|
* Sample usage:
|
|
|
|
* parseIntegerKey("255, 255, 255", 3, &red, &green, &blue);
|
|
|
|
* [will parse each field into its own integer]
|
|
|
|
*
|
|
|
|
* parseIntegerKey("1234", 1, &number);
|
|
|
|
* [will parse the single number into the variable]
|
|
|
|
*
|
|
|
|
* @param key String containing the integers to be parsed.
|
|
|
|
* @param count Number of comma-separated ints in the string.
|
|
|
|
* @param ... Integer variables to store the parsed ints, passed
|
|
|
|
* by reference.
|
|
|
|
* @returns True if the parsing succeeded.
|
|
|
|
*/
|
2008-07-30 13:33:32 +00:00
|
|
|
bool parseIntegerKey(const char *key, int count, ...) {
|
2008-06-28 13:44:10 +00:00
|
|
|
char *parseEnd;
|
2008-06-28 00:02:54 +00:00
|
|
|
int *num_ptr;
|
|
|
|
|
|
|
|
va_list args;
|
|
|
|
va_start(args, count);
|
|
|
|
|
|
|
|
while (count--) {
|
|
|
|
while (isspace(*key))
|
|
|
|
key++;
|
|
|
|
|
|
|
|
num_ptr = va_arg(args, int*);
|
|
|
|
*num_ptr = strtol(key, &parseEnd, 10);
|
|
|
|
|
2008-06-28 13:44:10 +00:00
|
|
|
key = parseEnd;
|
2008-06-28 00:02:54 +00:00
|
|
|
|
2008-06-28 13:44:10 +00:00
|
|
|
while (isspace(*key))
|
|
|
|
key++;
|
2008-06-28 00:02:54 +00:00
|
|
|
|
2008-06-28 13:44:10 +00:00
|
|
|
if (count && *key++ != ',')
|
|
|
|
return false;
|
2008-06-28 00:02:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
va_end(args);
|
2008-06-28 13:44:10 +00:00
|
|
|
return (*key == 0);
|
2008-06-28 00:02:54 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 19:42:04 +00:00
|
|
|
/**
|
|
|
|
* Overload if your parser needs to support parsing the same file
|
|
|
|
* several times, so you can clean up the internal state of the
|
|
|
|
* parser before each parse.
|
|
|
|
*/
|
|
|
|
virtual void cleanup() {}
|
|
|
|
|
2008-06-18 23:49:47 +00:00
|
|
|
int _pos; /** Current position on the XML buffer. */
|
2008-06-25 11:34:58 +00:00
|
|
|
XMLStream _text; /** Buffer with the text being parsed */
|
2008-06-25 17:56:00 +00:00
|
|
|
Common::String _fileName;
|
2008-06-18 23:49:47 +00:00
|
|
|
|
|
|
|
ParserState _state; /** Internal state of the parser */
|
|
|
|
|
|
|
|
Common::String _error; /** Current error message */
|
|
|
|
Common::String _token; /** Current text token */
|
|
|
|
|
2008-07-03 19:32:57 +00:00
|
|
|
Common::Stack<ParserNode*> _activeKey; /** Node stack of the parsed keys */
|
2008-06-18 23:49:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|