2008-06-12 19:06:37 +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$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/util.h"
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "common/events.h"
|
2008-06-12 23:13:58 +00:00
|
|
|
#include "common/hashmap.h"
|
|
|
|
#include "common/hash-str.h"
|
2008-06-12 19:06:37 +00:00
|
|
|
|
|
|
|
#include "gui/ThemeParser.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
<drawdata = "background_default" cache = true>
|
|
|
|
<draw func = "roundedsq" fill = "gradient" gradient_start = "255, 255, 128" gradient_end = "128, 128, 128" size = "auto">
|
|
|
|
<draw func = "roundedsq" fill = "none" color = "0, 0, 0" size = "auto">
|
|
|
|
</drawdata>
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace GUI {
|
|
|
|
|
2008-06-12 23:13:58 +00:00
|
|
|
void ThemeParser::debug_testEval() {
|
2008-06-14 17:45:26 +00:00
|
|
|
static const char *debugConfigText =
|
2008-06-14 16:14:50 +00:00
|
|
|
"</* lol this is just a moronic test */drawdata id = \"background_default\" cache = true>"
|
2008-06-12 23:13:58 +00:00
|
|
|
"<draw func = \"roundedsq\" /*/fill = \"gradient\" gradient_start = \"255, 255, 128\" gradient_end = \"128, 128, 128\" size = \"auto\"/>"
|
2008-06-13 22:05:21 +00:00
|
|
|
"<draw func = \"roundedsq\" fill = \"none\" color = /*\"0, 0, 0\"*/\"0, 1, 2\" size = \"auto\"/>"
|
2008-06-14 16:14:50 +00:00
|
|
|
"</ drawdata>/* lol this is just a simple test*/";
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
_text = strdup(debugConfigText);
|
2008-06-13 09:24:41 +00:00
|
|
|
parse();
|
2008-06-12 23:13:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-06-12 19:06:37 +00:00
|
|
|
void ThemeParser::parserError(const char *error_string) {
|
|
|
|
_state = kParserError;
|
2008-06-12 23:13:58 +00:00
|
|
|
printf("PARSER ERROR: %s\n", error_string);
|
|
|
|
}
|
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
void ThemeParser::parserCallback_DRAW() {
|
|
|
|
printf("Draw callback!\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThemeParser::parserCallback_DRAWDATA() {
|
|
|
|
printf("Drawdata callback!\n");
|
|
|
|
}
|
|
|
|
|
2008-06-12 23:13:58 +00:00
|
|
|
void ThemeParser::parseActiveKey(bool closed) {
|
|
|
|
printf("Parsed key %s.\n", _activeKey.top().c_str());
|
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
if (!_callbacks.contains(_activeKey.top())) {
|
|
|
|
parserError("Unhandled value inside key.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't you just love C++ syntax? Water clear.
|
|
|
|
(this->*(_callbacks[_activeKey.top()]))();
|
|
|
|
|
|
|
|
for (Common::StringMap::const_iterator t = _keyValues.top().begin(); t != _keyValues.top().end(); ++t)
|
2008-06-12 23:13:58 +00:00
|
|
|
printf(" Key %s = %s\n", t->_key.c_str(), t->_value.c_str());
|
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
if (closed) {
|
|
|
|
_keyValues.pop();
|
2008-06-12 23:13:58 +00:00
|
|
|
_activeKey.pop();
|
2008-06-13 09:24:41 +00:00
|
|
|
}
|
2008-06-12 23:13:58 +00:00
|
|
|
}
|
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
bool ThemeParser::parseKeyValue(Common::String keyName) {
|
2008-06-13 09:24:41 +00:00
|
|
|
assert(_keyValues.empty() == false);
|
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
if (_keyValues.top().contains(keyName))
|
2008-06-13 22:05:21 +00:00
|
|
|
return false;
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
_token.clear();
|
2008-06-14 17:45:26 +00:00
|
|
|
char stringStart;
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-13 17:47:56 +00:00
|
|
|
if (_text[_pos] == '"' || _text[_pos] == '\'') {
|
2008-06-14 17:45:26 +00:00
|
|
|
stringStart = _text[_pos++];
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
while (_text[_pos] && _text[_pos] != stringStart)
|
2008-06-13 22:05:21 +00:00
|
|
|
_token += _text[_pos++];
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
if (_text[_pos++] == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
} else if (!parseToken()) {
|
|
|
|
return false;
|
2008-06-12 23:13:58 +00:00
|
|
|
}
|
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
_keyValues.top()[keyName] = _token;
|
2008-06-13 22:05:21 +00:00
|
|
|
return true;
|
2008-06-12 19:06:37 +00:00
|
|
|
}
|
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
bool ThemeParser::parse() {
|
2008-06-12 19:06:37 +00:00
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
bool activeClosure = false;
|
|
|
|
bool selfClosure = false;
|
2008-06-13 22:05:21 +00:00
|
|
|
|
2008-06-12 19:06:37 +00:00
|
|
|
_state = kParserNeedKey;
|
2008-06-12 23:13:58 +00:00
|
|
|
_pos = 0;
|
|
|
|
_keyValues.clear();
|
2008-06-13 09:24:41 +00:00
|
|
|
_activeKey.clear();
|
2008-06-12 19:06:37 +00:00
|
|
|
|
2008-06-12 23:13:58 +00:00
|
|
|
while (_text[_pos]) {
|
2008-06-13 09:24:41 +00:00
|
|
|
if (_state == kParserError)
|
|
|
|
break;
|
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
if (skipSpaces())
|
|
|
|
continue;
|
2008-06-13 17:47:56 +00:00
|
|
|
|
|
|
|
if (skipComments())
|
|
|
|
continue;
|
2008-06-12 19:06:37 +00:00
|
|
|
|
|
|
|
switch (_state) {
|
|
|
|
case kParserNeedKey:
|
2008-06-14 17:45:26 +00:00
|
|
|
if (_text[_pos++] != '<') {
|
2008-06-14 16:14:50 +00:00
|
|
|
parserError("Expecting key start.");
|
|
|
|
break;
|
|
|
|
}
|
2008-06-12 19:06:37 +00:00
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
if (_text[_pos] == 0) {
|
|
|
|
parserError("Unexpected end of file.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-06-14 16:14:50 +00:00
|
|
|
if (_text[_pos] == '/' && _text[_pos + 1] != '*') {
|
2008-06-12 23:13:58 +00:00
|
|
|
_pos++;
|
2008-06-14 17:45:26 +00:00
|
|
|
activeClosure = true;
|
2008-06-13 22:05:21 +00:00
|
|
|
}
|
2008-06-14 16:14:50 +00:00
|
|
|
|
|
|
|
_state = kParserNeedKeyName;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kParserNeedKeyName:
|
2008-06-13 22:05:21 +00:00
|
|
|
if (!parseToken()) {
|
2008-06-14 16:14:50 +00:00
|
|
|
parserError("Invalid key name.");
|
2008-06-13 22:05:21 +00:00
|
|
|
break;
|
|
|
|
}
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
if (activeClosure) {
|
2008-06-13 09:39:13 +00:00
|
|
|
if (_activeKey.empty() || _token != _activeKey.top())
|
|
|
|
parserError("Unexpected closure.");
|
2008-06-13 22:05:21 +00:00
|
|
|
} else {
|
|
|
|
_keyValues.push(Common::StringMap());
|
|
|
|
_activeKey.push(_token);
|
2008-06-12 23:13:58 +00:00
|
|
|
}
|
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
_state = kParserNeedPropertyName;
|
2008-06-12 19:06:37 +00:00
|
|
|
break;
|
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
case kParserNeedPropertyName:
|
2008-06-14 17:45:26 +00:00
|
|
|
if (activeClosure) {
|
|
|
|
activeClosure = false;
|
2008-06-13 22:05:21 +00:00
|
|
|
_activeKey.pop();
|
|
|
|
_keyValues.pop();
|
2008-06-12 19:06:37 +00:00
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
if (_text[_pos++] != '>')
|
|
|
|
parserError("Invalid syntax in key closure.");
|
2008-06-14 16:14:50 +00:00
|
|
|
else
|
|
|
|
_state = kParserNeedKey;
|
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
break;
|
2008-06-12 19:06:37 +00:00
|
|
|
}
|
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
selfClosure = (_text[_pos] == '/');
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-14 17:45:26 +00:00
|
|
|
if ((selfClosure && _text[_pos + 1] == '>') || _text[_pos] == '>') {
|
|
|
|
parseActiveKey(selfClosure);
|
|
|
|
_pos += selfClosure ? 2 : 1;
|
2008-06-12 23:13:58 +00:00
|
|
|
_state = kParserNeedKey;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-06-14 16:14:50 +00:00
|
|
|
if (!parseToken())
|
|
|
|
parserError("Error when parsing key value.");
|
|
|
|
else
|
|
|
|
_state = kParserNeedPropertyOperator;
|
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
break;
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
case kParserNeedPropertyOperator:
|
2008-06-14 16:14:50 +00:00
|
|
|
if (_text[_pos++] != '=')
|
|
|
|
parserError("Unexpected character after key name.");
|
|
|
|
else
|
|
|
|
_state = kParserNeedPropertyValue;
|
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
break;
|
2008-06-12 23:13:58 +00:00
|
|
|
|
2008-06-13 22:05:21 +00:00
|
|
|
case kParserNeedPropertyValue:
|
2008-06-14 16:14:50 +00:00
|
|
|
if (!parseKeyValue(_token))
|
|
|
|
parserError("Unable to parse key value.");
|
|
|
|
else
|
|
|
|
_state = kParserNeedPropertyName;
|
2008-06-12 19:06:37 +00:00
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
break;
|
2008-06-12 19:06:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-13 09:24:41 +00:00
|
|
|
if (_state == kParserError) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-06-12 23:13:58 +00:00
|
|
|
if (_state != kParserNeedKey || !_activeKey.empty()) {
|
|
|
|
parserError("Unexpected end of file.");
|
2008-06-13 22:05:21 +00:00
|
|
|
return false;
|
2008-06-12 23:13:58 +00:00
|
|
|
}
|
|
|
|
|
2008-06-12 19:06:37 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|