scummvm/gui/ThemeParser.cpp

1122 lines
33 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "gui/ThemeEngine.h"
#include "gui/ThemeEval.h"
#include "gui/ThemeParser.h"
#include "graphics/VectorRenderer.h"
#include "common/system.h"
#include "common/tokenizer.h"
namespace GUI {
struct TextDataInfo {
TextData id;
const char *name;
};
static const TextDataInfo kTextDataDefaults[] = {
{ kTextDataDefault, "text_default" },
{ kTextDataButton, "text_button" },
{ kTextDataNormalFont, "text_normal" },
{ kTextDataTooltip, "tooltip_normal" },
{ kTextDataConsole, "console" },
{ kTextDataExtraLang, "extra_lang" }
};
#define SCALEVALUE(val) (val > 0 ? val * _scaleFactor : val)
#define FORCESCALEVALUE(val) (val * _scaleFactor)
static TextData parseTextDataId(const Common::String &name) {
for (int i = 0; i < kTextDataMAX; ++i)
if (name.compareToIgnoreCase(kTextDataDefaults[i].name) == 0)
return kTextDataDefaults[i].id;
return kTextDataNone;
}
struct TextColorDataInfo {
TextColor id;
const char *name;
};
static const TextColorDataInfo kTextColorDefaults[] = {
{ kTextColorNormal, "color_normal" },
{ kTextColorNormalInverted, "color_normal_inverted" },
{ kTextColorNormalHover, "color_normal_hover" },
{ kTextColorNormalDisabled, "color_normal_disabled" },
{ kTextColorAlternative, "color_alternative" },
{ kTextColorAlternativeInverted, "color_alternative_inverted" },
{ kTextColorAlternativeHover, "color_alternative_hover" },
{ kTextColorAlternativeDisabled, "color_alternative_disabled" },
{ kTextColorOverride, "color_override" },
{ kTextColorOverrideInverted, "color_override_inverted" },
{ kTextColorOverrideHover, "color_override_hover" },
{ kTextColorOverrideDisabled, "color_override_disabled" },
{ kTextColorButton, "color_button" },
{ kTextColorButtonHover, "color_button_hover" },
{ kTextColorButtonDisabled, "color_button_disabled" }
};
static TextColor parseTextColorId(const Common::String &name) {
for (int i = 0; i < kTextColorMAX; ++i)
if (name.compareToIgnoreCase(kTextColorDefaults[i].name) == 0)
return kTextColorDefaults[i].id;
return kTextColorMAX;
}
Graphics::TextAlign parseTextHAlign(const Common::String &val) {
if (val == "start")
return Graphics::kTextAlignStart;
else if (val == "end")
return Graphics::kTextAlignEnd;
else if (val == "left")
return Graphics::kTextAlignLeft;
else if (val == "right")
return Graphics::kTextAlignRight;
else if (val == "center")
return Graphics::kTextAlignCenter;
else
return Graphics::kTextAlignInvalid;
}
static GUI::ThemeEngine::TextAlignVertical parseTextVAlign(const Common::String &val) {
if (val == "top")
return GUI::ThemeEngine::kTextAlignVTop;
else if (val == "center")
return GUI::ThemeEngine::kTextAlignVCenter;
else if (val == "bottom")
return GUI::ThemeEngine::kTextAlignVBottom;
else
return GUI::ThemeEngine::kTextAlignVInvalid;
}
bool parseBoolean(const Common::String &val) {
if (val == "true")
return true;
else if (val == "yes")
return true;
else if (val == "false")
return false;
else if (val == "no")
return false;
else
warning("Incorrect boolean value %s", val.c_str());
return false;
}
ThemeParser::ThemeParser(ThemeEngine *parent) : XMLParser() {
_defaultStepGlobal = defaultDrawStep();
_defaultStepLocal = nullptr;
_theme = parent;
_baseWidth = _baseHeight = 0;
_scaleFactor = 1.0f;
}
ThemeParser::~ThemeParser() {
delete _defaultStepGlobal;
delete _defaultStepLocal;
}
void ThemeParser::cleanup() {
delete _defaultStepGlobal;
delete _defaultStepLocal;
_defaultStepGlobal = defaultDrawStep();
_defaultStepLocal = nullptr;
_palette.clear();
}
Graphics::DrawStep *ThemeParser::defaultDrawStep() {
Graphics::DrawStep *step = new Graphics::DrawStep;
step->xAlign = Graphics::DrawStep::kVectorAlignManual;
step->yAlign = Graphics::DrawStep::kVectorAlignManual;
step->factor = 1;
step->autoWidth = true;
step->autoHeight = true;
step->fillMode = Graphics::VectorRenderer::kFillDisabled;
step->scale = (1 << 16);
step->radius = 0xFF;
step->shadowIntensity = SCALEVALUE((1 << 16));
return step;
}
Graphics::DrawStep *ThemeParser::newDrawStep() {
assert(_defaultStepGlobal);
Graphics::DrawStep *step = nullptr; //new DrawStep;
if (_defaultStepLocal) {
step = new Graphics::DrawStep(*_defaultStepLocal);
} else {
step = new Graphics::DrawStep(*_defaultStepGlobal);
}
return step;
}
bool ThemeParser::parserCallback_defaults(ParserNode *node) {
ParserNode *parentNode = getParentNode(node);
Graphics::DrawStep *step = nullptr;
if (parentNode->name == "render_info") {
step = _defaultStepGlobal;
} else if (parentNode->name == "drawdata") {
if (_defaultStepLocal == nullptr)
_defaultStepLocal = new Graphics::DrawStep(*_defaultStepGlobal);
step = _defaultStepLocal;
} else {
return parserError("<default> key out of scope. Must be inside <drawdata> or <render_info> keys.");
}
return parseDrawStep(node, step, false);
}
bool ThemeParser::parserCallback_font(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
return true;
}
bool ThemeParser::parserCallback_language(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
TextData textDataId = parseTextDataId(getParentNode(node)->values["id"]);
// Default to a point size of 12.
int pointsize = 12;
Common::String ps;
if (node->values.contains("point_size")) {
ps = node->values["point_size"];
} else if (getParentNode(node)->values.contains("point_size")) {
ps = getParentNode(node)->values["point_size"];
}
if (!ps.empty()) {
if (sscanf(ps.c_str(), "%d", &pointsize) != 1 || pointsize <= 0)
return parserError(Common::String::format("Font \"%s\" has invalid point size \"%s\"", node->values["id"].c_str(), ps.c_str()));
}
pointsize = SCALEVALUE(pointsize);
Common::String file;
if (node->values.contains("file")) {
file = node->values["file"];
} else if (getParentNode(node)->values.contains("file")) {
file = getParentNode(node)->values["file"];
}
if (file.empty()) {
return parserError("Missing required property 'file' in either <font> or <language>");
}
Common::String scalableFile;
if (node->values.contains("scalable_file")) {
scalableFile = node->values["scalable_file"];
} else if (getParentNode(node)->values.contains("scalable_file")) {
scalableFile = getParentNode(node)->values["scalable_file"];
}
_theme->storeFontNames(textDataId, node->values["id"], file, scalableFile, pointsize);
if (!_theme->addFont(textDataId, node->values["id"], file, scalableFile, pointsize))
return parserError("Error loading localized Font in theme engine.");
return true;
}
bool ThemeParser::parserCallback_text_color(ParserNode *node) {
int red, green, blue;
TextColor colorId = parseTextColorId(node->values["id"]);
if (colorId == kTextColorMAX)
return parserError("Error text color is not defined.");
if (_palette.contains(node->values["color"]))
getPaletteColor(node->values["color"], red, green, blue);
else if (!parseIntegerKey(node->values["color"], 3, &red, &green, &blue))
return parserError("Error parsing color value for text color definition.");
if (!_theme->addTextColor(colorId, red, green, blue))
return parserError("Error while adding text color information.");
return true;
}
bool ThemeParser::parserCallback_fonts(ParserNode *node) {
return true;
}
bool ThemeParser::parserCallback_cursor(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
int spotx, spoty;
if (!parseIntegerKey(node->values["hotspot"], 2, &spotx, &spoty))
return parserError("Error parsing cursor Hot Spot coordinates.");
if (!_theme->createCursor(node->values["file"], spotx, spoty))
return parserError("Error creating Bitmap Cursor.");
return true;
}
bool ThemeParser::parserCallback_bitmap(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
Common::String scalableFile;
if (node->values.contains("scalable_file"))
scalableFile = node->values["scalable_file"];
int width = 0, height = 0;
Common::String val;
if (node->values.contains("width")) {
val = node->values["width"];
if (!parseIntegerKey(val, 1, &width))
return parserError("Error parsing width value");
}
if (node->values.contains("height")) {
val = node->values["height"];
if (!parseIntegerKey(val, 1, &height))
return parserError("Error parsing width height");
}
if (!_theme->addBitmap(node->values["filename"], scalableFile, width, height))
return parserError("Error loading Bitmap file '" + node->values["filename"] + "'");
return true;
}
bool ThemeParser::parserCallback_text(ParserNode *node) {
Graphics::TextAlign alignH;
GUI::ThemeEngine::TextAlignVertical alignV;
if ((alignH = parseTextHAlign(node->values["horizontal_align"])) == Graphics::kTextAlignInvalid)
return parserError("Invalid value for text alignment.");
if ((alignV = parseTextVAlign(node->values["vertical_align"])) == GUI::ThemeEngine::kTextAlignVInvalid)
return parserError("Invalid value for text alignment.");
Common::String id = getParentNode(node)->values["id"];
TextData textDataId = parseTextDataId(node->values["font"]);
TextColor textColorId = parseTextColorId(node->values["text_color"]);
if (!_theme->addTextData(id, textDataId, textColorId, alignH, alignV))
return parserError("Error adding Text Data for '" + id + "'.");
return true;
}
bool ThemeParser::parserCallback_render_info(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false)
node->ignore = true;
return true;
}
bool ThemeParser::parserCallback_layout_info(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false)
node->ignore = true;
return true;
}
bool ThemeParser::parserCallback_palette(ParserNode *node) {
return true;
}
bool ThemeParser::parserCallback_color(ParserNode *node) {
Common::String name = node->values["name"];
if (_palette.contains(name))
return parserError("Color '" + name + "' has already been defined.");
int red, green, blue;
if (parseIntegerKey(node->values["rgb"], 3, &red, &green, &blue) == false ||
red < 0 || red > 255 || green < 0 || green > 255 || blue < 0 || blue > 255)
return parserError("Error parsing RGB values for palette color '" + name + "'");
_palette[name].r = red;
_palette[name].g = green;
_palette[name].b = blue;
return true;
}
static Graphics::DrawingFunctionCallback getDrawingFunctionCallback(const Common::String &name) {
if (name == "circle")
return &Graphics::VectorRenderer::drawCallback_CIRCLE;
if (name == "square")
return &Graphics::VectorRenderer::drawCallback_SQUARE;
if (name == "roundedsq")
return &Graphics::VectorRenderer::drawCallback_ROUNDSQ;
if (name == "bevelsq")
return &Graphics::VectorRenderer::drawCallback_BEVELSQ;
if (name == "line")
return &Graphics::VectorRenderer::drawCallback_LINE;
if (name == "triangle")
return &Graphics::VectorRenderer::drawCallback_TRIANGLE;
if (name == "fill")
return &Graphics::VectorRenderer::drawCallback_FILLSURFACE;
if (name == "tab")
return &Graphics::VectorRenderer::drawCallback_TAB;
if (name == "void")
return &Graphics::VectorRenderer::drawCallback_VOID;
if (name == "bitmap")
return &Graphics::VectorRenderer::drawCallback_BITMAP;
if (name == "cross")
return &Graphics::VectorRenderer::drawCallback_CROSS;
return nullptr;
}
bool ThemeParser::parserCallback_drawstep(ParserNode *node) {
Graphics::DrawStep *drawstep = newDrawStep();
Common::String functionName = node->values["func"];
drawstep->drawingCall = getDrawingFunctionCallback(functionName);
if (drawstep->drawingCall == nullptr) {
delete drawstep;
return parserError(functionName + " is not a valid drawing function name");
}
if (!parseDrawStep(node, drawstep, true)) {
delete drawstep;
return false;
}
_theme->addDrawStep(getParentNode(node)->values["id"], *drawstep);
delete drawstep;
return true;
}
bool ThemeParser::parserCallback_drawdata(ParserNode *node) {
bool cached = false;
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
if (node->values.contains("cache")) {
if (!Common::parseBool(node->values["cache"], cached))
return parserError("'Parsed' value must be either true or false.");
}
if (_theme->addDrawData(node->values["id"], cached) == false)
return parserError("Error adding Draw Data set: Invalid DrawData name.");
delete _defaultStepLocal;
_defaultStepLocal = nullptr;
return true;
}
bool ThemeParser::parseDrawStep(ParserNode *stepNode, Graphics::DrawStep *drawstep, bool functionSpecific) {
int red, green, blue, x;
Common::String val;
/**
* Helper macro to sanitize and assign an integer value from a key
* to the draw step.
*
* @param struct_name Name of the field of a DrawStep struct that must be
* assigned.
* @param key_name Name as STRING of the key identifier as it appears in the
* theme description format.
* @param force Sets if the key is optional or necessary.
*/
#define PARSER_ASSIGN_INT(struct_name, key_name, force) \
if (stepNode->values.contains(key_name)) { \
if (!parseIntegerKey(stepNode->values[key_name], 1, &x)) \
return parserError("Error parsing key value for '" + Common::String(key_name) + "'."); \
\
drawstep->struct_name = x; \
} else if (force) { \
return parserError("Missing necessary key '" + Common::String(key_name) + "'."); \
}
#define PARSER_ASSIGN_INT_SCALED(struct_name, key_name, force) \
PARSER_ASSIGN_INT(struct_name, key_name, force); \
drawstep->struct_name = SCALEVALUE(drawstep->struct_name);
/**
* Helper macro to sanitize and assign a RGB value from a key to the draw
* step. RGB values have the following syntax: "R, G, B".
*
* @param struct_name Name of the field of a DrawStep struct that must be
* assigned.
* @param key_name Name as STRING of the key identifier as it appears in the
* theme description format.
*/
#define PARSER_ASSIGN_RGB(struct_name, key_name) \
if (stepNode->values.contains(key_name)) { \
val = stepNode->values[key_name]; \
if (_palette.contains(val)) { \
red = _palette[val].r; \
green = _palette[val].g; \
blue = _palette[val].b; \
} else if (parseIntegerKey(val, 3, &red, &green, &blue) == false || \
red < 0 || red > 255 || green < 0 || green > 255 || blue < 0 || blue > 255) \
return parserError("Error parsing color struct '" + val + "'");\
\
drawstep->struct_name.r = red; \
drawstep->struct_name.g = green; \
drawstep->struct_name.b = blue; \
drawstep->struct_name.set = true; \
}
PARSER_ASSIGN_INT_SCALED(stroke, "stroke", false);
PARSER_ASSIGN_INT_SCALED(bevel, "bevel", false);
PARSER_ASSIGN_INT_SCALED(shadow, "shadow", false);
PARSER_ASSIGN_INT(factor, "gradient_factor", false);
PARSER_ASSIGN_RGB(fgColor, "fg_color");
PARSER_ASSIGN_RGB(bgColor, "bg_color");
PARSER_ASSIGN_RGB(gradColor1, "gradient_start");
PARSER_ASSIGN_RGB(gradColor2, "gradient_end");
PARSER_ASSIGN_RGB(bevelColor, "bevel_color");
if (functionSpecific) {
assert(stepNode->values.contains("func"));
Common::String functionName = stepNode->values["func"];
if (functionName == "bitmap") {
if (!stepNode->values.contains("file"))
return parserError("Need to specify a filename for Bitmap blitting.");
drawstep->blitSrc = _theme->getImageSurface(stepNode->values["file"]);
if (!drawstep->blitSrc)
return parserError("The given filename hasn't been loaded into the GUI.");
drawstep->alphaType = drawstep->blitSrc->detectAlpha();
}
if (functionName == "roundedsq" || functionName == "circle" || functionName == "tab") {
if (stepNode->values.contains("radius") && stepNode->values["radius"] == "auto") {
drawstep->radius = 0xFF;
} else {
PARSER_ASSIGN_INT_SCALED(radius, "radius", true);
}
}
if (functionName == "triangle") {
drawstep->extraData = Graphics::VectorRenderer::kTriangleUp;
if (stepNode->values.contains("orientation")) {
val = stepNode->values["orientation"];
if (val == "top")
drawstep->extraData = Graphics::VectorRenderer::kTriangleUp;
else if (val == "bottom")
drawstep->extraData = Graphics::VectorRenderer::kTriangleDown;
else if (val == "left")
drawstep->extraData = Graphics::VectorRenderer::kTriangleLeft;
else if (val == "right")
drawstep->extraData = Graphics::VectorRenderer::kTriangleRight;
else
return parserError("'" + val + "' is not a valid value for triangle orientation.");
}
}
if (stepNode->values.contains("size")) {
warning("The <size> keyword has been deprecated. Use <width> and <height> instead");
}
if (stepNode->values.contains("width") && stepNode->values["width"] != "auto") {
drawstep->autoWidth = false;
val = stepNode->values["width"];
if (parseIntegerKey(val, 1, &x))
drawstep->w = SCALEVALUE(x);
else if (val == "height")
drawstep->w = -1;
else return parserError("Invalid value for vector width.");
if (stepNode->values.contains("xpos")) {
val = stepNode->values["xpos"];
if (parseIntegerKey(val, 1, &x))
drawstep->x = SCALEVALUE(x);
else if (val == "center")
drawstep->xAlign = Graphics::DrawStep::kVectorAlignCenter;
else if (val == "left")
drawstep->xAlign = Graphics::DrawStep::kVectorAlignLeft;
else if (val == "right")
drawstep->xAlign = Graphics::DrawStep::kVectorAlignRight;
else
return parserError("Invalid value for X Position");
} else {
return parserError("When width is not set to 'auto', a <xpos> tag must be included.");
}
}
if (stepNode->values.contains("height") && stepNode->values["height"] != "auto") {
drawstep->autoHeight = false;
val = stepNode->values["height"];
if (parseIntegerKey(val, 1, &x))
drawstep->h = SCALEVALUE(x);
else if (val == "width")
drawstep->h = -1;
else return parserError("Invalid value for vector height.");
if (stepNode->values.contains("ypos")) {
val = stepNode->values["ypos"];
if (parseIntegerKey(val, 1, &x))
drawstep->y = SCALEVALUE(x);
else if (val == "center")
drawstep->yAlign = Graphics::DrawStep::kVectorAlignCenter;
else if (val == "top")
drawstep->yAlign = Graphics::DrawStep::kVectorAlignTop;
else if (val == "bottom")
drawstep->yAlign = Graphics::DrawStep::kVectorAlignBottom;
else
return parserError("Invalid value for Y Position");
} else {
return parserError("When height is not set to 'auto', a <ypos> tag must be included.");
}
}
if (drawstep->h == -1 && drawstep->w == -1)
return parserError("Cross-reference in Vector Size: Height is set to width and width is set to height.");
}
if (stepNode->values.contains("fill")) {
val = stepNode->values["fill"];
if (val == "none")
drawstep->fillMode = Graphics::VectorRenderer::kFillDisabled;
else if (val == "foreground")
drawstep->fillMode = Graphics::VectorRenderer::kFillForeground;
else if (val == "background")
drawstep->fillMode = Graphics::VectorRenderer::kFillBackground;
else if (val == "gradient")
drawstep->fillMode = Graphics::VectorRenderer::kFillGradient;
else
return parserError("'" + stepNode->values["fill"] + "' is not a valid fill mode for a shape.");
}
if (stepNode->values.contains("padding")) {
val = stepNode->values["padding"];
int pr, pt, pl, pb;
if (parseIntegerKey(val, 4, &pl, &pt, &pr, &pb)) {
drawstep->padding.left = SCALEVALUE(pl);
drawstep->padding.top = SCALEVALUE(pt);
drawstep->padding.right = SCALEVALUE(pr);
drawstep->padding.bottom = SCALEVALUE(pb);
}
}
if (stepNode->values.contains("clip")) {
val = stepNode->values["clip"];
int cl, ct, cr, cb;
if (parseIntegerKey(val, 4, &cl, &ct, &cr, &cb)) {
// Values could be less than 0 which is legit
drawstep->clip.left = FORCESCALEVALUE(cl);
drawstep->clip.top = FORCESCALEVALUE(ct);
drawstep->clip.right = FORCESCALEVALUE(cr);
drawstep->clip.bottom = FORCESCALEVALUE(cb);
}
}
#undef PARSER_ASSIGN_INT
#undef PARSER_ASSIGN_RGB
return true;
}
bool ThemeParser::parserCallback_def(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
Common::String var = "Globals." + node->values["var"];
int value;
bool scalable = false;
if (_theme->getEvaluator()->hasVar(node->values["value"]) == true)
value = _theme->getEvaluator()->getVar(node->values["value"]);
else if (!parseIntegerKey(node->values["value"], 1, &value))
return parserError("Invalid definition for '" + var + "'.");
if (node->values.contains("scalable"))
scalable = parseBoolean(node->values["scalable"]);
if (scalable)
value = SCALEVALUE(value);
_theme->getEvaluator()->setVar(var, value);
return true;
}
bool ThemeParser::parserCallback_widget(ParserNode *node) {
Common::String var;
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
if (getParentNode(node)->name == "globals") {
var = "Globals." + node->values["name"] + ".";
if (!parseCommonLayoutProps(node, var))
return parserError("Error parsing Layout properties of '" + var + "'.");
} else {
// FIXME: Shouldn't we distinguish the name/id and the label of a widget?
var = node->values["name"];
int width = -1;
int height = -1;
bool useRTL = true;
if (node->values.contains("width")) {
if (_theme->getEvaluator()->hasVar(node->values["width"]) == true)
width = _theme->getEvaluator()->getVar(node->values["width"]);
else if (parseIntegerKey(node->values["width"], 1, &width))
width = SCALEVALUE(width);
else
return parserError("Corrupted width value in key for " + var);
}
if (node->values.contains("height")) {
if (_theme->getEvaluator()->hasVar(node->values["height"]) == true)
height = _theme->getEvaluator()->getVar(node->values["height"]);
else if (parseIntegerKey(node->values["height"], 1, &height))
height = SCALEVALUE(height);
else
return parserError("Corrupted height value in key for " + var);
}
Graphics::TextAlign alignH = Graphics::kTextAlignStart;
if (node->values.contains("textalign")) {
if ((alignH = parseTextHAlign(node->values["textalign"])) == Graphics::kTextAlignInvalid)
return parserError("Invalid value for text alignment.");
}
if (node->values.contains("rtl"))
useRTL = parseBoolean(node->values["rtl"]);
_theme->getEvaluator()->addWidget(var, node->values["type"], width, height, alignH, useRTL);
}
return true;
}
bool ThemeParser::parserCallback_dialog(ParserNode *node) {
Common::String name = node->values["name"];
int inset = 0;
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
if (node->values.contains("inset")) {
if (!parseIntegerKey(node->values["inset"], 1, &inset))
return false;
}
Common::String overlays = node->values["overlays"];
// The size of a dialog depends on the value of its 'overlays' property:
// - 'screen', means the dialog fills the whole screen.
// - 'screen_center', means the size of the dialog is determined
// by its contents, unless an explicit size is specified.
// - if it is set to a user interface element name, the dialog takes
// its size.
int width = -1, height = -1;
if (node->values.contains("size")) {
if (overlays != "screen_center")
return parserError("Dialogs can only have an explicit size if they overlay 'screen_center'.");
if (!parseIntegerKey(node->values["size"], 2, &width, &height))
return false;
}
_theme->getEvaluator()->addDialog(name, overlays, SCALEVALUE(width), SCALEVALUE(height), inset);
if (node->values.contains("shading")) {
int shading = 0;
if (node->values["shading"] == "dim")
shading = 1;
else if (node->values["shading"] == "luminance")
shading = 2;
else return parserError("Invalid value for Dialog background shading.");
_theme->getEvaluator()->setVar("Dialog." + name + ".Shading", shading);
}
return true;
}
bool ThemeParser::parserCallback_import(ParserNode *node) {
Common::String importedName = node->values["layout"];
if (!_theme->getEvaluator()->hasDialog(importedName))
return parserError("Imported layout was not found: " + importedName);
_theme->getEvaluator()->addImportedLayout(importedName);
return true;
}
bool ThemeParser::parserCallback_layout(ParserNode *node) {
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
int spacing = -1;
if (node->values.contains("spacing")) {
if (parseIntegerKey(node->values["spacing"], 1, &spacing))
spacing = SCALEVALUE(spacing);
else
return false;
}
ThemeLayout::ItemAlign itemAlign = ThemeLayout::kItemAlignStart;
if (node->values.contains("align")) {
Common::String val = node->values["align"];
if (val == "start") {
itemAlign = ThemeLayout::kItemAlignStart;
} else if (val == "center") {
itemAlign = ThemeLayout::kItemAlignCenter;
} else if (val == "end") {
itemAlign = ThemeLayout::kItemAlignEnd;
} else if (val == "stretch") {
itemAlign = ThemeLayout::kItemAlignStretch;
} else {
return parserError("'" + val + "' is not a valid item alignment for a layout.");
}
}
if (node->values["type"] == "vertical")
_theme->getEvaluator()->addLayout(GUI::ThemeLayout::kLayoutVertical, spacing, itemAlign);
else if (node->values["type"] == "horizontal")
_theme->getEvaluator()->addLayout(GUI::ThemeLayout::kLayoutHorizontal, spacing, itemAlign);
else
return parserError("Invalid layout type. Only 'horizontal' and 'vertical' layouts allowed.");
if (node->values.contains("padding")) {
int paddingL, paddingR, paddingT, paddingB;
if (!parseIntegerKey(node->values["padding"], 4, &paddingL, &paddingR, &paddingT, &paddingB))
return false;
// values are scaled inside this method
_theme->getEvaluator()->addPadding(paddingL, paddingR, paddingT, paddingB);
}
return true;
}
bool ThemeParser::parserCallback_space(ParserNode *node) {
int size = -1;
if (resolutionCheck(node->values["resolution"]) == false) {
node->ignore = true;
return true;
}
if (node->values.contains("size")) {
if (_theme->getEvaluator()->hasVar(node->values["size"]))
size = _theme->getEvaluator()->getVar(node->values["size"]);
else if (parseIntegerKey(node->values["size"], 1, &size))
size = SCALEVALUE(size);
else
return parserError("Invalid value for Spacing size.");
}
_theme->getEvaluator()->addSpace(size);
return true;
}
bool ThemeParser::closedKeyCallback(ParserNode *node) {
if (node->name == "layout")
_theme->getEvaluator()->closeLayout();
else if (node->name == "dialog")
_theme->getEvaluator()->closeDialog();
return true;
}
bool ThemeParser::parseCommonLayoutProps(ParserNode *node, const Common::String &var) {
if (node->values.contains("size")) {
int width, height;
if (parseIntegerKey(node->values["size"], 2, &width, &height)) {
width = SCALEVALUE(width);
height = SCALEVALUE(height);
} else {
Common::StringTokenizer tokenizer(node->values["size"], " ,");
Common::String wtoken, htoken;
char *parseEnd;
wtoken = tokenizer.nextToken();
if (_theme->getEvaluator()->hasVar(wtoken)) {
width = _theme->getEvaluator()->getVar(wtoken);
} else {
width = strtol(wtoken.c_str(), &parseEnd, 10);
if (*parseEnd != 0 && !(*parseEnd == '%' && *(parseEnd + 1) == 0))
return false;
if (wtoken.lastChar() == '%')
width = _baseWidth * width / 100;
else
width = SCALEVALUE(width);
}
htoken = tokenizer.nextToken();
if (_theme->getEvaluator()->hasVar(htoken)) {
height = _theme->getEvaluator()->getVar(htoken);
} else {
height = strtol(htoken.c_str(), &parseEnd, 10);
if (*parseEnd != 0 && !(*parseEnd == '%' && *(parseEnd + 1) == 0))
return false;
if (htoken.lastChar() == '%')
height = _baseHeight * height / 100;
else
height = SCALEVALUE(height);
}
if (!tokenizer.empty())
return false;
}
_theme->getEvaluator()->setVar(var + "Width", width);
_theme->getEvaluator()->setVar(var + "Height", height);
}
if (node->values.contains("pos")) {
int x, y;
if (parseIntegerKey(node->values["pos"], 2, &x, &y)) {
x = SCALEVALUE(x);
y = SCALEVALUE(y);
} else {
Common::StringTokenizer tokenizer(node->values["pos"], " ,");
Common::String xpos, ypos;
char *parseEnd;
xpos = tokenizer.nextToken();
if (xpos == "center") {
if (!_theme->getEvaluator()->hasVar(var + "Width"))
return false;
x = (_baseWidth / 2) - (_theme->getEvaluator()->getVar(var + "Width") / 2);
} else if (_theme->getEvaluator()->hasVar(xpos)) {
x = _theme->getEvaluator()->getVar(xpos);
} else {
x = strtol(xpos.c_str(), &parseEnd, 10);
x = SCALEVALUE(x);
if (*parseEnd != 0 && !(*parseEnd == 'r' && *(parseEnd + 1) == 0))
return false;
if (xpos.lastChar() == 'r')
x = _baseWidth - x;
}
ypos = tokenizer.nextToken();
if (ypos == "center") {
if (!_theme->getEvaluator()->hasVar(var + "Height"))
return false;
y = (_baseHeight / 2) - (_theme->getEvaluator()->getVar(var + "Height") / 2);
} else if (_theme->getEvaluator()->hasVar(ypos)) {
y = _theme->getEvaluator()->getVar(ypos);
} else {
y = strtol(ypos.c_str(), &parseEnd, 10);
y = SCALEVALUE(y);
if (*parseEnd != 0 && !(*parseEnd == 'b' && *(parseEnd + 1) == 0))
return false;
if (ypos.lastChar() == 'b')
y = _baseHeight - y;
}
if (!tokenizer.empty())
return false;
}
_theme->getEvaluator()->setVar(var + "X", x);
_theme->getEvaluator()->setVar(var + "Y", y);
}
if (node->values.contains("padding")) {
int paddingL, paddingR, paddingT, paddingB;
if (!parseIntegerKey(node->values["padding"], 4, &paddingL, &paddingR, &paddingT, &paddingB))
return false;
_theme->getEvaluator()->setVar(var + "Padding.Left", SCALEVALUE(paddingL));
_theme->getEvaluator()->setVar(var + "Padding.Right", SCALEVALUE(paddingR));
_theme->getEvaluator()->setVar(var + "Padding.Top", SCALEVALUE(paddingT));
_theme->getEvaluator()->setVar(var + "Padding.Bottom", SCALEVALUE(paddingB));
}
if (node->values.contains("textalign")) {
Graphics::TextAlign alignH = Graphics::kTextAlignStart;
if ((alignH = parseTextHAlign(node->values["textalign"])) == Graphics::kTextAlignInvalid)
return parserError("Invalid value for text alignment.");
_theme->getEvaluator()->setVar(var + "Align", alignH);
}
return true;
}
bool ThemeParser::resolutionCheck(const Common::String &resolution) {
if (resolution.empty())
return true;
Common::StringTokenizer globTokenizer(resolution, ", ");
Common::String cur;
while (!globTokenizer.empty()) {
cur = globTokenizer.nextToken();
bool lt;
int val;
if (cur.size() < 3) {
warning("Invalid theme 'resolution' token '%s'", resolution.c_str());
return false;
}
if (cur[0] == 'x') {
val = _baseWidth;
} else if (cur[0] == 'y') {
val = _baseHeight;
} else {
warning("Error parsing theme 'resolution' token '%s'", resolution.c_str());
return false;
}
if (cur[1] == '<') {
lt = true;
} else if (cur[1] == '>') {
lt = false;
} else {
warning("Error parsing theme 'resolution' token '%s'", resolution.c_str());
return false;
}
bool eq = false;
int offset = 2;
if (cur[2] == '=') {
eq = true;
offset++;
}
int token;
if (cur[offset] == 'W') { // Reported threshold width
token = 320;
} else if (cur[offset] == 'H') { // Reported threshold height
#ifndef IPHONE
token = 400;
#else
// HACK. Think about API to move it to OSystem?
// iPhone SE (gen3) is 375 × 667 @2, iPhone 14 is 390 × 844 @3
//
// Hence, we are setting height to 375, so we have highres
// theme by default
token = 375;
#endif
} else if (cur[offset] == 'x') {
token = _baseWidth;
} else if (cur[offset] == 'y') {
token = _baseHeight;
} else {
token = atoi(cur.c_str() + offset);
}
if (eq && val == token)
return true;
// check inverse for unfulfilled requirements
if (lt) {
if (val >= token)
return false;
} else {
if (val <= token)
return false;
}
}
return true;
}
} // End of namespace GUI