#define DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN #include #include "../../../Source/Core/DataExpression.cpp" using namespace Rml; DataTypeRegister type_register; DataModel model(type_register.GetTransformFuncRegister()); String TestExpression(String expression) { String result; DataExpressionInterface interface(&model, nullptr); DataParser parser(expression, interface); if (parser.Parse(false)) { Program program = parser.ReleaseProgram(); AddressList addresses = parser.ReleaseAddresses(); DataInterpreter interpreter(program, addresses, interface); if (interpreter.Run()) result = interpreter.Result().Get(); // If it fails, it can be useful to check: interpreter.DumpProgram() } else { FAIL_CHECK("Could not parse expression."); } return result; }; bool TestAssignment(String expression) { bool result = false; DataExpressionInterface interface(&model, nullptr); DataParser parser(expression, interface); if (parser.Parse(true)) { Program program = parser.ReleaseProgram(); AddressList addresses = parser.ReleaseAddresses(); DataInterpreter interpreter(program, addresses, interface); result = interpreter.Run(); } return result; }; TEST_CASE("Data expression parsing and execution") { float radius = 8.7f; String color_name = "color"; Colourb color_value = Colourb(180, 100, 255); DataModelConstructor handle(&model, &type_register); handle.Bind("radius", &radius); handle.Bind("color_name", &color_name); handle.BindFunc("color_value", [&](Variant& variant) { variant = ToString(color_value); }); CHECK(TestExpression("!!10 - 1 ? 'hello' : 'world' | to_upper") == "WORLD"); CHECK(TestExpression("(color_name) + (': rgba(' + color_value + ')')") == "color: rgba(180, 100, 255, 255)"); CHECK(TestExpression("'hello world' | to_upper(5 + 12 == 17 ? 'yes' : 'no', 9*2)") == "HELLO WORLD"); CHECK(TestExpression("true == false") == "0"); CHECK(TestExpression("true != false") == "1"); CHECK(TestExpression("true") == "1"); CHECK(TestExpression("true || false ? true && 3==1+2 ? 'Absolutely!' : 'well..' : 'no'") == "Absolutely!"); CHECK(TestExpression(R"('It\'s a fit')") == R"(It's a fit)"); CHECK(TestExpression("2 * 2") == "4"); CHECK(TestExpression("50000 / 1500") == "33.333"); CHECK(TestExpression("5*1+2") == "7"); CHECK(TestExpression("5*(1+2)") == "15"); CHECK(TestExpression("2*(-2)/4") == "-1"); CHECK(TestExpression("5.2 + 19 + 'px'") == "24.2px"); CHECK(TestExpression("(radius | format(2)) + 'm'") == "8.70m"); CHECK(TestExpression("radius < 10.5 ? 'smaller' : 'larger'") == "smaller"); CHECK(TestAssignment("radius = 15")); CHECK(radius == doctest::Approx(15.f)); CHECK(TestExpression("radius < 10.5 ? 'smaller' : 'larger'") == "larger"); CHECK(TestAssignment("radius = 4; color_name = 'image-color'")); CHECK(radius == doctest::Approx(4.f)); CHECK(color_name == "image-color"); CHECK(TestExpression("radius == 4 && color_name == 'image-color'") == "1"); CHECK(TestExpression("5 == 1 + 2*2 || 8 == 1 + 4 ? 'yes' : 'no'") == "yes"); CHECK(TestExpression("!!('fa' + 'lse')") == "0"); CHECK(TestExpression("!!('tr' + 'ue')") == "1"); CHECK(TestExpression("'fox' + 'dog' ? 'FoxyDog' : 'hot' + 'dog' | to_upper") == "HOTDOG"); CHECK(TestExpression("3.62345 | round") == "4"); CHECK(TestExpression("3.62345 | format(0)") == "4"); CHECK(TestExpression("3.62345 | format(2)") == "3.62"); CHECK(TestExpression("3.62345 | format(10)") == "3.6234500000"); CHECK(TestExpression("3.62345 | format(10, true)") == "3.62345"); CHECK(TestExpression("3.62345 | round | format(2)") == "4.00"); CHECK(TestExpression("3.0001 | format(2, false)") == "3.00"); CHECK(TestExpression("3.0001 | format(2, true)") == "3"); CHECK(TestExpression("0.2 + 3.42345 | round") == "4"); CHECK(TestExpression("(3.42345 | round) + 0.2") == "3.2"); CHECK(TestExpression("(3.42345 | format(0)) + 0.2") == "30.2"); // Here, format(0) returns a string, so the + means string concatenation. }