/*
* This source file is part of RmlUi, the HTML/CSS Interface Middleware
*
* For the latest information, see http://github.com/mikke89/RmlUi
*
* Copyright (c) 2008-2010 CodePoint Ltd, Shift Technology Ltd
* Copyright (c) 2019-2023 The RmlUi Team, and contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
*/
#include "../../Include/RmlUi/Core/PropertyDefinition.h"
#include "../../Include/RmlUi/Core/Log.h"
#include "../../Include/RmlUi/Core/StyleSheetSpecification.h"
namespace Rml {
PropertyDefinition::PropertyDefinition(PropertyId id, const String& _default_value, bool _inherited, bool _forces_layout) :
id(id), default_value(_default_value, Unit::UNKNOWN), relative_target(RelativeTarget::None)
{
inherited = _inherited;
forces_layout = _forces_layout;
default_value.definition = this;
}
PropertyDefinition::~PropertyDefinition() {}
PropertyDefinition& PropertyDefinition::AddParser(const String& parser_name, const String& parser_parameters)
{
ParserState new_parser;
// Fetch the parser.
new_parser.parser = StyleSheetSpecification::GetParser(parser_name);
if (new_parser.parser == nullptr)
{
Log::Message(Log::LT_ERROR, "Property was registered with invalid parser '%s'.", parser_name.c_str());
return *this;
}
// Split the parameter list, and set up the map.
if (!parser_parameters.empty())
{
StringList parameter_list;
StringUtilities::ExpandString(parameter_list, parser_parameters);
int parameter_value = 0;
for (const String& parameter : parameter_list)
{
// Look for an optional parameter value such as in "normal=400".
const size_t i_equal = parameter.find('=');
if (i_equal != String::npos)
{
if (!TypeConverter::Convert(parameter.substr(i_equal + 1), parameter_value))
{
Log::Message(Log::LT_ERROR, "Parser was added with invalid parameter '%s'.", parameter.c_str());
return *this;
}
}
new_parser.parameters[parameter.substr(0, i_equal)] = parameter_value;
parameter_value += 1;
}
}
const int parser_index = (int)parsers.size();
parsers.push_back(new_parser);
// If the default value has not been parsed successfully yet, run it through the new parser.
if (default_value.unit == Unit::UNKNOWN)
{
String unparsed_value = default_value.value.Get();
if (new_parser.parser->ParseValue(default_value, unparsed_value, new_parser.parameters))
{
default_value.parser_index = parser_index;
}
else
{
default_value.value = unparsed_value;
default_value.unit = Unit::UNKNOWN;
}
}
return *this;
}
bool PropertyDefinition::ParseValue(Property& property, const String& value) const
{
for (size_t i = 0; i < parsers.size(); i++)
{
if (parsers[i].parser->ParseValue(property, value, parsers[i].parameters))
{
property.definition = this;
property.parser_index = (int)i;
return true;
}
}
property.unit = Unit::UNKNOWN;
return false;
}
bool PropertyDefinition::GetValue(String& value, const Property& property) const
{
value = property.value.Get();
switch (property.unit)
{
case Unit::KEYWORD:
{
int parser_index = property.parser_index;
if (parser_index < 0 || parser_index >= (int)parsers.size())
{
// Look for the keyword parser in the property's list of parsers
const auto* keyword_parser = StyleSheetSpecification::GetParser("keyword");
for (int i = 0; i < (int)parsers.size(); i++)
{
if (parsers[i].parser == keyword_parser)
{
parser_index = i;
break;
}
}
// If we couldn't find it, exit now
if (parser_index < 0 || parser_index >= (int)parsers.size())
return false;
}
int keyword = property.value.Get();
for (const auto& name_keyword : parsers[parser_index].parameters)
{
if (name_keyword.second == keyword)
{
value = name_keyword.first;
break;
}
}
return false;
}
break;
default: value += ToString(property.unit); break;
}
return true;
}
bool PropertyDefinition::IsInherited() const
{
return inherited;
}
bool PropertyDefinition::IsLayoutForced() const
{
return forces_layout;
}
const Property* PropertyDefinition::GetDefaultValue() const
{
return &default_value;
}
RelativeTarget PropertyDefinition::GetRelativeTarget() const
{
return relative_target;
}
PropertyId PropertyDefinition::GetId() const
{
return id;
}
PropertyDefinition& PropertyDefinition::SetRelativeTarget(RelativeTarget relative_target)
{
this->relative_target = relative_target;
return *this;
}
} // namespace Rml