#include #include #include #include #include #include #include #include #include #include using namespace mbgl; using namespace mbgl::style; using namespace mbgl::style::expression; using namespace mbgl::style::expression::dsl; using namespace std::string_literals; static const StubGeometryTileFeature oneInteger{PropertyMap{{"property", uint64_t(1)}}}; static const StubGeometryTileFeature oneDouble{PropertyMap{{"property", 1.0}}}; static const StubGeometryTileFeature oneString{PropertyMap{{"property", "1"s}}}; static const StubGeometryTileFeature oneColor{PropertyMap{{"color", "red"s}}}; static const StubGeometryTileFeature oneImage{PropertyMap{{"image_name", "maki-11"s}}}; static const StubGeometryTileFeature emptyTileFeature{PropertyMap{}}; float evaluate(PropertyValue value, float zoom) { return value.evaluate(PropertyEvaluator(PropertyEvaluationParameters(zoom), 0)); } template auto createOverride(expression::type::Type exprType, PossiblyEvaluatedPropertyValue propValue, std::string propName) { return std::make_unique>(std::move(exprType), std::move(propValue), std::move(propName)); } TEST(PropertyExpression, Constant) { EXPECT_EQ(2.0f, evaluate(PropertyValue(2.0), 0)); EXPECT_EQ(3.8f, evaluate(PropertyValue(3.8), 0)); EXPECT_EQ(22.0f, evaluate(PropertyValue(22.0), 0)); EXPECT_EQ(2.0f, evaluate(PropertyValue(2.0), 4)); EXPECT_EQ(3.8f, evaluate(PropertyValue(3.8), 4)); EXPECT_EQ(22.0f, evaluate(PropertyValue(22.0), 4)); EXPECT_EQ(2.0f, evaluate(PropertyValue(2.0), 22)); EXPECT_EQ(3.8f, evaluate(PropertyValue(3.8), 22)); EXPECT_EQ(22.0f, evaluate(PropertyValue(22.0), 22)); } TEST(PropertyExpression, Expression) { PropertyExpression expression(interpolate(linear(), zoom(), 0.0, literal(0.0), 1.0, literal(1.0))); EXPECT_EQ(0.0, evaluate(expression, 0.0)); EXPECT_EQ(0.5, evaluate(expression, 0.5)); } TEST(PropertyExpression, Defaults) { EXPECT_EQ(1.0f, PropertyExpression(number(get("property")), 0.0) .evaluate(oneInteger, 2.0f)); EXPECT_EQ(1.0f, PropertyExpression(number(get("property")), 0.0) .evaluate(oneDouble, 2.0f)); EXPECT_EQ(0.0f, PropertyExpression(number(get("property")), 0.0) .evaluate(oneString, 2.0f)); EXPECT_EQ(2.0f, PropertyExpression(number(get("property"))) .evaluate(oneString, 2.0f)); } TEST(PropertyExpression, ZoomInterpolation) { EXPECT_EQ(40.0f, PropertyExpression( interpolate(linear(), zoom(), 0.0, interpolate(linear(), number(get("property")), 1.0, literal(24.0)), 1.5, interpolate(linear(), number(get("property")), 1.0, literal(36.0)), 3.0, interpolate(linear(), number(get("property")), 1.0, literal(48.0)) ), 0.0f) .evaluate(2.0f, oneInteger, -1.0f)) << "Should interpolate between stops"; EXPECT_EQ(33.0, PropertyExpression( interpolate(linear(), zoom(), 5.0, interpolate(linear(), number(get("property")), 1.0, literal(33.0)), 10.0, interpolate(linear(), number(get("property")), 1.0, literal(66.0)) ), 0.0f) .evaluate(0.0f, oneInteger, -1.0f)) << "Use first stop output for input values from -inf to first stop"; EXPECT_EQ(66.0, PropertyExpression( interpolate(linear(), zoom(), 0.0, interpolate(linear(), number(get("property")), 1.0, literal(33.0)), 10.0, interpolate(linear(), number(get("property")), 1.0, literal(66.0)) ), 0.0f) .evaluate(20.0f, oneInteger, -1.0f)) << "Use last stop output for input values from last stop to +inf"; EXPECT_EQ(66.0f, PropertyExpression( interpolate(linear(), zoom(), 0.0, interpolate(linear(), number(get("property")), 1.0, literal(33.0)), 10.0, interpolate(linear(), number(get("property")), 1.0, literal(66.0)) ), 0.0f) .evaluate(10.0f, oneInteger, -1.0f)) << "Should interpolate TO the last stop."; EXPECT_EQ(33.0f, PropertyExpression( interpolate(linear(), zoom(), 0.0, interpolate(linear(), number(get("property")), 1.0, literal(33.0)), 10.0, interpolate(linear(), number(get("property")), 1.0, literal(66.0)) ), 0.0f) .evaluate(0.0f, oneInteger, -1.0f)) << "Should interpolate TO the first stop"; } TEST(PropertyExpression, Issue8460) { PropertyExpression fn1( interpolate(linear(), zoom(), 15.0, interpolate(linear(), number(get("property")), 1.0, literal(0.0)), 15.2, interpolate(linear(), number(get("property")), 1.0, literal(600.0)) ), 0.0f); EXPECT_NEAR( 0.0f, fn1.evaluate(15.0f, oneInteger, -1.0f), 0.00); EXPECT_NEAR(300.0f, fn1.evaluate(15.1f, oneInteger, -1.0f), 0.01); EXPECT_NEAR(600.0f, fn1.evaluate(15.2f, oneInteger, -1.0f), 0.00); EXPECT_NEAR(600.0f, fn1.evaluate(16.0f, oneInteger, -1.0f), 0.00); PropertyExpression fn2( interpolate(linear(), zoom(), 15.0, interpolate(linear(), number(get("property")), 1.0, literal(0.0)), 15.2, interpolate(linear(), number(get("property")), 1.0, literal(300.0)), 18.0, interpolate(linear(), number(get("property")), 1.0, literal(600.0)) ), 0.0f); EXPECT_NEAR( 0.0f, fn2.evaluate(15.0f, oneInteger, -1.0f), 0.00); EXPECT_NEAR(150.0f, fn2.evaluate(15.1f, oneInteger, -1.0f), 0.01); EXPECT_NEAR(300.0f, fn2.evaluate(15.2f, oneInteger, -1.0f), 0.00); EXPECT_NEAR(385.71f, fn2.evaluate(16.0f, oneInteger, -1.0f), 0.01); EXPECT_NEAR(600.0f, fn2.evaluate(18.0f, oneInteger, -1.0f), 0.00); EXPECT_NEAR(600.0f, fn2.evaluate(19.0f, oneInteger, -1.0f), 0.00); } TEST(PropertyExpression, FormatSectionOverride) { using Value = expression::Value; Value formattedSection = std::unordered_map{ {"text-color", Value{Color::blue()}} }; auto ctx = expression::EvaluationContext(&oneDouble).withFormattedSection(&formattedSection); PossiblyEvaluatedPropertyValue constantValueRed(Color::red()); PossiblyEvaluatedPropertyValue constantValueGreen(Color::green()); PossiblyEvaluatedPropertyValue ddsValueRed(toColor(string(get("color")))); // Evaluation test { auto override1 = createOverride(expression::type::Color, constantValueGreen, "text-color"); PropertyExpression propExpr(std::move(override1)); EXPECT_EQ(Color::green(), propExpr.evaluate(15.0f, oneDouble, Color())); EXPECT_EQ(Color::green(), propExpr.evaluate(oneDouble, Color())); EXPECT_EQ(Color::blue(), propExpr.evaluate(ctx)); auto override2 = createOverride(expression::type::Color, ddsValueRed, "text-color"); PropertyExpression propExprDDS(std::move(override2)); EXPECT_EQ(Color::red(), propExprDDS.evaluate(oneColor, Color())); EXPECT_EQ(Color::blue(), propExprDDS.evaluate(ctx)); } // Equality test { auto override1 = createOverride(expression::type::Color, constantValueRed, "text-color"); auto override2 = createOverride(expression::type::Color, constantValueGreen, "text-color"); auto override3 = createOverride(expression::type::Color, constantValueGreen, "text-halo-color"); auto override4 = createOverride(expression::type::Color, ddsValueRed, "text-color"); EXPECT_TRUE(*override1 != *override2); EXPECT_TRUE(*override2 != *override3); EXPECT_TRUE(*override1 != *override4); } } TEST(PropertyExpression, ImageExpression) { const std::set emptySet; const std::set availableImages = {"maki-11", "airport-11", "bicycle-15"}; // evaluation test without available images { PropertyExpression propExpr(image(literal("airport-11"))); auto evaluatedImage = propExpr.evaluate(emptyTileFeature, emptySet, expression::Image()); EXPECT_FALSE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), "airport-11"s); PropertyExpression ddPropExpr(image(get(literal("image_name"s)))); evaluatedImage = ddPropExpr.evaluate(oneImage, emptySet, expression::Image()); EXPECT_FALSE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), "maki-11"s); evaluatedImage = ddPropExpr.evaluate(emptyTileFeature, emptySet, expression::Image()); EXPECT_FALSE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), ""s); } // evaluation test with available images { PropertyExpression propExpr(image(literal("airport-11"))); auto evaluatedImage = propExpr.evaluate(emptyTileFeature, availableImages, expression::Image()); EXPECT_TRUE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), "airport-11"s); PropertyExpression ddPropExpr(image(get(literal("image_name"s)))); evaluatedImage = ddPropExpr.evaluate(oneImage, availableImages, expression::Image()); EXPECT_TRUE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), "maki-11"s); evaluatedImage = ddPropExpr.evaluate(emptyTileFeature, availableImages, expression::Image()); EXPECT_FALSE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), ""s); } // evaluation with zoom { auto expr = step(zoom(), image(literal("airport-11")), 18.0, image(literal("bicycle-15"))); PropertyExpression propExpr(std::move(expr)); auto evaluatedImage = propExpr.evaluate(0.0, emptyTileFeature, availableImages, expression::Image()); EXPECT_TRUE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), "airport-11"s); evaluatedImage = propExpr.evaluate(18.0, emptyTileFeature, availableImages, expression::Image()); EXPECT_TRUE(evaluatedImage.isAvailable()); EXPECT_EQ(evaluatedImage.id(), "bicycle-15"s); } } TEST(PropertyExpression, WithinExpression) { CanonicalTileID canonicalTileID(3, 3, 3); // geoJSON source must be Polygon.(Currently only supports Polygon) static const std::string invalidGeoSource = R"({ "type": "LineString", "coordinates": [[0, 0], [0, 5], [5, 5], [5, 0]] })"; static const std::string validGeoSource = R"data( { "type": "Polygon", "coordinates": [ [ [-11.689453125, -9.79567758282973], [2.021484375, -10.012129557908128], [-15.99609375, -17.392579271057766], [-5.9765625, -5.659718554577273], [-16.259765625, -3.7327083213358336], [-17.75390625, -12.897489183755892], [-17.138671875, -21.002471054356715], [4.482421875, -16.8886597873816], [3.076171875, -7.01366792756663], [ -5.9326171875, 0.6591651462894632], [-16.1279296875, 1.4939713066293239], [-11.689453125, -9.79567758282973] ] ] })data"; // evaluation test with invalid geojson source { std::stringstream ss; ss << std::string(R"(["within", )") << invalidGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_FALSE(expression); } // evaluation test with valid geojson source std::stringstream ss; ss << std::string(R"(["within", )") << validGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); // evaluation test with valid geojson source but FeatureType is not Point/LineString (currently only support // FeatureType::Point and FeatureType::LineString) { // testPoly is inside polygon, but will return false Polygon testRing{{{-9.228515625, -17.560246503294888}, {-2.4609375, -16.04581345375217}, {-9.228515625, -17.560246503294888}}}; auto geoPoly = convertGeometry(testRing, canonicalTileID); StubGeometryTileFeature polyFeature(FeatureType::Polygon, geoPoly); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&polyFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); } // evaluation test with valid geojson source and valid linestring features { // testLine is inside polygon LineString testLine0{{-9.228515625, -17.560246503294888}, {-2.4609375, -16.04581345375217}}; auto geoLine0 = convertGeometry(testLine0, canonicalTileID); StubGeometryTileFeature lineFeature0(FeatureType::LineString, geoLine0); // testLine is intersecting polygon even though end points are all inside polygon, will return false LineString testLine1{{-10.4150390625, -10.082445532162465}, {-8.8275146484375, -9.194292714912638}}; auto geoLine1 = convertGeometry(testLine1, canonicalTileID); StubGeometryTileFeature lineFeature1(FeatureType::LineString, geoLine1); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature0).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature1).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); } // evaluation test with valid geojson source and valid point features { auto getPointFeature = [&canonicalTileID](const Point& testPoint) -> StubGeometryTileFeature { auto geoPoint = convertGeometry(testPoint, canonicalTileID); StubGeometryTileFeature pointFeature(FeatureType::Point, geoPoint); return pointFeature; }; // check `within` algorithm auto pointFeature = getPointFeature(Point(-10.72265625, -7.27529233637217)); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-7.646484374999999, -12.382928338487396)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-15.2490234375, -2.986927393334863)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-10.590820312499998, 2.4601811810210052)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-3.9990234375, -4.915832801313164)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); pointFeature = getPointFeature(Point(-1.1865234375, -16.63619187839765)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); pointFeature = getPointFeature(Point(-15.5126953125, -11.73830237143684)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); // On the boundary pointFeature = getPointFeature(Point(3.076171875, -7.01366792756663)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-11.689453125, -9.79567758282973)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(2.021484375, -10.012129557908128)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-15.99609375, -17.392579271057766)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-5.9765625, -5.659718554577273)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-16.259765625, -3.7327083213358336)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-17.75390625, -12.897489183755892)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-17.138671875, -21.002471054356715)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(4.482421875, -16.8886597873816)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(3.076171875, -7.01366792756663)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-5.9326171875, 0.6591651462894632)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-16.1279296875, 1.4939713066293239)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); pointFeature = getPointFeature(Point(-11.689453125, -9.79567758282973)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); } } TEST(PropertyExpression, WithinExpressionAntiMeridian) { CanonicalTileID canonicalTileID(0, 0, 0); // evaluation test with line geometries { const std::string polygon1 = R"data( { "type": "Polygon", "coordinates": [[[-190, 0], [-178, 0], [-178, 10], [-190, 10], [-190, 0]]] })data"; // evaluation test with valid geojson source std::stringstream ss; ss << std::string(R"(["within", )") << polygon1 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); LineString testLine0{{-183, 5}, {-179, 1}}; auto geoLine0 = convertGeometry(testLine0, canonicalTileID); StubGeometryTileFeature lineFeature0(FeatureType::LineString, geoLine0); LineString testLine1{{183, 5}, {181, 1}}; auto geoLine1 = convertGeometry(testLine1, canonicalTileID); StubGeometryTileFeature lineFeature1(FeatureType::LineString, geoLine1); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature0).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature1).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); } // evaluation test with point geometries { static const std::string polygon2 = R"data( { "type": "Polygon", "coordinates": [[[-185.0, 60.0], [-175.0, 60.0], [-175.0, 65.0], [-185.0, 65.0], [-185.0, 60.0]]] })data"; // evaluation test with valid geojson source std::stringstream ss; ss << std::string(R"(["within", )") << polygon2 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto getPointFeature = [&canonicalTileID](const Point& testPoint) -> StubGeometryTileFeature { auto geoPoint = convertGeometry(testPoint, canonicalTileID); StubGeometryTileFeature pointFeature(FeatureType::Point, geoPoint); return pointFeature; }; // check `within` algorithm auto pointFeature = getPointFeature(Point(177, 62.5)); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); pointFeature = getPointFeature(Point(180, 62.5)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); pointFeature = getPointFeature(Point(-180, 62.5)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_TRUE(evaluatedResult); pointFeature = getPointFeature(Point(-190, 62.5)); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID)); EXPECT_FALSE(evaluatedResult); } } TEST(PropertyExpression, DistanceExpression) { static const double invalidResult = std::numeric_limits::quiet_NaN(); static const CanonicalTileID canonicalTileID(15, 18653, 9484); // Parsing error with invalid geojson source { // geoJSON source must be Point or LineString. const std::string invalidGeoSource = R"({ "type": "Polygon", "coordinates": [[[0, 0], [0, 5], [5, 5], [5, 0], [0, 0]]] })"; std::stringstream ss; ss << std::string(R"(["distance", )") << invalidGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_FALSE(expression); } // Parsing error with invalid unit { const std::string invalidGeoSource = R"({ "type": "Point", "coordinates": [0, 0], })"; std::stringstream ss; ss << std::string(R"(["distance", )") << invalidGeoSource << std::string(R"(, "InvalidUnits"])"); auto expression = createExpression(ss.str().c_str()); ASSERT_FALSE(expression); } // Evaluation test with non-supported geometry type { const std::string pointGeoSource = R"({ "type": "Point", "coordinates": [ 24.938492774963375, 60.16980226227959 ] })"; std::stringstream ss; ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); Polygon testRing{{{24.937505722045895, 60.16815846879986}, {24.940595626831055, 60.16815846879986}, {24.940595626831055, 60.169268572114106}, {24.937505722045895, 60.169268572114106}, {24.937505722045895, 60.16815846879986}}}; auto geoPoly = convertGeometry(testRing, canonicalTileID); StubGeometryTileFeature polyFeature(FeatureType::Polygon, geoPoly); // return evaluation error auto evaluatedResult = propExpr.evaluate(EvaluationContext(&polyFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_TRUE(std::isnan(evaluatedResult)); } // Evaluation test with Point to Point distance by using different units { const std::string pointGeoSource = R"({ "type": "Point", "coordinates": [ 24.938492774963375, 60.16980226227959 ] })"; Point featurePoint{24.931159615516663, 60.16733387258923}; auto geoPoint = convertGeometry(featurePoint, canonicalTileID); StubGeometryTileFeature pointFeature(FeatureType::Point, geoPoint); std::stringstream ss; // Default Unit: meters ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(491.307, evaluatedResult, 0.01); // Unit: meters ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "meters" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(491.307, evaluatedResult, 0.01); // Unit: kilometers ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "kilometers" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(0.491307, evaluatedResult, 0.00001); // Unit: miles ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "miles" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(0.305284, evaluatedResult, 0.00001); // Unit: nauticalmiles ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "nauticalmiles" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(0.265284, evaluatedResult, 0.00001); // Unit: yards ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "yards" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(537.299, evaluatedResult, 0.01); // Unit: feet ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "feet" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(1611.898, evaluatedResult, 0.01); // Unit: inches ss.str(std::string()); ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"(, "inches" ])"); expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); propExpr = std::move(expression); evaluatedResult = propExpr.evaluate(EvaluationContext(&pointFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(19342.781, evaluatedResult, 0.01); } // Evaluation test with Point to MultiPoint distance { const std::string pointGeoSource = R"({ "type": "Point", "coordinates": [ 24.938492774963375, 60.16980226227959 ] })"; std::stringstream ss; ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); MultiPoint featurePoint{{24.931159615516663, 60.16733387258923}, {24.936089515686035, 60.169033745694826}}; auto geoPoint = convertGeometry(featurePoint, canonicalTileID); StubGeometryTileFeature feature(FeatureType::Point, geoPoint); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&feature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(158.541, evaluatedResult, 0.01); } // Evaluation test with Point to LineString distance { const std::string pointGeoSource = R"({ "type": "Point", "coordinates": [ 24.938492774963375, 60.16980226227959 ] })"; std::stringstream ss; ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); LineString line{{24.93621826171875, 60.17131789507754}, {24.941453933715817, 60.170229208170014}, {24.941110610961914, 60.16850004304534}}; auto geoLine = convertGeometry(line, canonicalTileID); StubGeometryTileFeature lineFeature(FeatureType::LineString, geoLine); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(107.211, evaluatedResult, 0.01); } // Evaluation test with Point to MultiLineString distance { const std::string pointGeoSource = R"({ "type": "Point", "coordinates": [ 24.938492774963375, 60.16980226227959 ] })"; std::stringstream ss; ss << std::string(R"(["distance", )") << pointGeoSource << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); MultiLineString lines{ {{24.93621826171875, 60.17131789507754}, {24.941453933715817, 60.170229208170014}, {24.941110610961914, 60.16850004304534}}, {{24.937773942947388, 60.16940199546824}, {24.933900833129883, 60.17007977773949}}}; auto geoLine = convertGeometry(lines, canonicalTileID); StubGeometryTileFeature lineFeature(FeatureType::LineString, geoLine); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(59.845, evaluatedResult, 0.01); } const auto getFeature = [](const std::string &name, FeatureType type, const CanonicalTileID &canonical) { const auto geometryInput = mbgl::util::read_file(std::string("test/fixtures/geometry_data/" + name)); mapbox::geojson::geojson geojson{mapbox::geojson::parse(geometryInput)}; const auto &geometry = geojson.match( [](const mapbox::geometry::geometry &geometrySet) { return mbgl::Feature(geometrySet).geometry; }, [](const mapbox::feature::feature &feature) { return mbgl::Feature(feature).geometry; }, [](const mapbox::feature::feature_collection &features) { return mbgl::Feature(features.front()).geometry; }, [](const auto &) { return mapbox::geometry::empty(); }); return StubGeometryTileFeature(type, convertGeometry(geometry, canonical)); }; // Evaluation test with MultiPoint to MultiPoint distance { const auto multiPoints1 = mbgl::util::read_file("test/fixtures/geometry_data/multi_point_1.geojson"); const auto multiPointsFeature = getFeature("multi_point_2.geojson", FeatureType::Point, canonicalTileID); std::stringstream ss; ss << std::string(R"(["distance", )") << multiPoints1 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate( EvaluationContext(&multiPointsFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(283.60, evaluatedResult, 0.1); } // Evaluation test with MultiPoint to LineString distance { const auto multiPoints1 = mbgl::util::read_file("test/fixtures/geometry_data/multi_point_1.geojson"); const auto multiLineFeature = getFeature("line_string_2.geojson", FeatureType::LineString, canonicalTileID); std::stringstream ss; ss << std::string(R"(["distance", )") << multiPoints1 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate( EvaluationContext(&multiLineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(258.691, evaluatedResult, 0.01); // ----------------- switch feature and argument ----------------- const auto multiLine2 = mbgl::util::read_file("test/fixtures/geometry_data/line_string_2.geojson"); const auto multiPointsFeature = getFeature("multi_point_1.geojson", FeatureType::Point, canonicalTileID); std::stringstream ss1; ss1 << std::string(R"(["distance", )") << multiLine2 << std::string(R"( ])"); auto expression1 = createExpression(ss1.str().c_str()); ASSERT_TRUE(expression1); PropertyExpression propExpr1(std::move(expression1)); auto evaluatedResult1 = propExpr1.evaluate( EvaluationContext(&multiPointsFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(258.691, evaluatedResult1, 0.01); } // Evaluation test with LineString to LineString distance { const auto lineString1 = mbgl::util::read_file("test/fixtures/geometry_data/line_string_1.geojson"); const auto lineFeature = getFeature("line_string_2.geojson", FeatureType::LineString, canonicalTileID); std::stringstream ss; ss << std::string(R"(["distance", )") << lineString1 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate(EvaluationContext(&lineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(180.408, evaluatedResult, 0.01); } // Evaluation test with MultiPoints to MutltiLineString distance { const auto multiPoints = mbgl::util::read_file("test/fixtures/geometry_data/multi_point_2.geojson"); const auto multiLineFeature = getFeature("multi_line_string_1.geojson", FeatureType::LineString, canonicalTileID); std::stringstream ss; ss << std::string(R"(["distance", )") << multiPoints << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate( EvaluationContext(&multiLineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(937.632, evaluatedResult, 0.01); } // Evaluation test with LineString to MutltiLineString distance { const auto LineString1 = mbgl::util::read_file("test/fixtures/geometry_data/line_string_1.geojson"); const auto multiLineFeature = getFeature("multi_line_string_2.geojson", FeatureType::LineString, canonicalTileID); std::stringstream ss; ss << std::string(R"(["distance", )") << LineString1 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate( EvaluationContext(&multiLineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(17.511, evaluatedResult, 0.01); } // Evaluation test with MultiLineString to MutltiLineString distance { const auto multiLineString1 = mbgl::util::read_file("test/fixtures/geometry_data/multi_line_string_1.geojson"); const auto multiLineFeature = getFeature("multi_line_string_2.geojson", FeatureType::LineString, canonicalTileID); std::stringstream ss; ss << std::string(R"(["distance", )") << multiLineString1 << std::string(R"( ])"); auto expression = createExpression(ss.str().c_str()); ASSERT_TRUE(expression); PropertyExpression propExpr(std::move(expression)); auto evaluatedResult = propExpr.evaluate( EvaluationContext(&multiLineFeature).withCanonicalTileID(&canonicalTileID), invalidResult); EXPECT_NEAR(384.109, evaluatedResult, 0.01); } }