Add special-case curve tests

This commit is contained in:
Matthias Clasen
2022-04-07 14:48:19 -04:00
parent af6ae3fd74
commit bdd8c87f9c

View File

@@ -121,6 +121,163 @@ test_curve_tangents (void)
g_assert_true (graphene_vec2_near (&t, graphene_vec2_y_axis (), 0.0001));
}
static gboolean
near_one_point (const graphene_point_t *p,
const graphene_point_t *q,
int n,
float epsilon)
{
for (int i = 0; i < n; i++)
{
if (graphene_point_near (p, &q[i], epsilon))
return TRUE;
}
return FALSE;
}
static gboolean
pathop_cb (GskPathOperation op,
const graphene_point_t *pts,
gsize n_pts,
float weight,
gpointer user_data)
{
GskCurve *curve = user_data;
g_assert (op != GSK_PATH_CLOSE);
if (op == GSK_PATH_MOVE)
return TRUE;
gsk_curve_init_foreach (curve, op, pts, n_pts, weight);
return FALSE;
}
static void
parse_curve (GskCurve *c,
const char *str)
{
GskPath *path = gsk_path_parse (str);
gsk_path_foreach (path, -1, pathop_cb, c);
gsk_path_unref (path);
}
static void
test_curve_intersections (void)
{
struct {
const char *c1;
const char *c2;
int n;
graphene_point_t p[9];
} tests[] = {
{ "M 0 100 L 100 100",
"M 0 110 L 100 110",
0, { { 0, 0 }, },
},
{ "M 0 100 L 100 100",
"M 110 100 L 210 100",
0, { { 0, 0 }, },
},
{ "M 0 100 L 100 100",
"M 0 100 L -100 100",
1, { { 0, 100 }, },
},
{ "M 0 100 L 100 100",
"M 20 100 L 80 100",
2, { { 20, 100 }, { 80, 100 }, },
},
{ "M 0 100 L 100 100",
"M 150 100 L 50 100",
2, { { 100, 100 }, { 50, 100 }, },
},
{ "M 888 482 C 999.333313 508.666687 1080.83325 544.333313 1132.5 589",
"M 886 680 L 642 618",
0, { { 0, 0 }, },
},
{ "M 1119.5 772 C 1039.16675 850.666687 925.333313 890 778 890",
"M 1052 1430 734 762",
1, { { 794.851257, 889.825439 }, },
},
{ "M 844.085 271.845 Q 985.723 94.0499 836.718 817.477",
"M 790.206 34.4028 L 965.236 893.041",
1,
{ { 890.685, 527.323 }, }
},
{ "M 521.412 466.917 Q 838.809 472.131 51.3819 192.985",
"M 854.519 682.333 Q 154.655 -50.3073 260.046 627.56",
3, { { 611.932129, 450.019135 }, { 518.727844, 377.701019 }, { 343.737976, 301.792297 } }
},
{ "M 521.412 466.917 Q 838.809 472.131 51.3819 192.985",
"M 854.519 682.333 O 154.655 -50.3073 260.046 627.56 1",
3, { { 611.932129, 450.019135 }, { 518.727844, 377.701019 }, { 343.737976, 301.792297 } }
},
{ "M 521.412 466.917 Q 838.809 472.131 51.3819 192.985",
"M 854.519 682.333 O 154.655 50.3073 260.046 627.56 1.53362",
3, { { 597.725, 460.362 }, { 426.752, 335.879 }, { 310.528, 288.720 }, },
},
};
for (unsigned int i = 0; i < G_N_ELEMENTS (tests); i++)
{
GskCurve c1, c2;
float t1[9], t2[9];
graphene_point_t p[9];
int n;
parse_curve (&c1, tests[i].c1);
parse_curve (&c2, tests[i].c2);
n = gsk_curve_intersect (&c1, &c2, t1, t2, p, 9);
if (g_test_verbose ())
g_print ("expected %d intersections, got %d\n", tests[i].n, n);
if (c1.op == GSK_PATH_CONIC || c2.op == GSK_PATH_CONIC)
{
/* Our conic intersection code can produce duplicate intersections */
for (unsigned int j = 0; j < tests[i].n; j++)
{
if (g_test_verbose ())
g_print ("looking for %f %f\n", tests[i].p[j].x, tests[i].p[j].y);
g_assert_true (near_one_point (&tests[i].p[j], p, n, 0.01));
if (g_test_verbose ())
g_print ("found expected intersection %d\n", j);
}
for (unsigned int j = 0; j < n; j++)
{
if (g_test_verbose ())
g_print ("looking for %f %f\n", p[j].x, p[j].y);
g_assert_true (near_one_point (&p[j], tests[i].p, tests[i].n, 0.01));
if (g_test_verbose ())
g_print ("intersection %d is expected\n", j);
}
}
else
{
g_assert_true (n == tests[i].n);
for (unsigned int j = 0; j < n; j++)
{
if (g_test_verbose ())
g_print ("expected %f %f got %f %f\n",
tests[i].p[j].x, tests[i].p[j].y,
p[j].x, p[j].y);
g_assert_true (graphene_point_near (&p[j], &tests[i].p[j], 0.001));
if (g_test_verbose ())
g_test_message ("intersection %d OK", j);
}
}
if (g_test_verbose ())
g_test_message ("test %d OK", i);
}
}
int
main (int argc,
char *argv[])
@@ -129,6 +286,7 @@ main (int argc,
g_test_add_func ("/curve/special/conic-segment", test_conic_segment);
g_test_add_func ("/curve/special/tangents", test_curve_tangents);
g_test_add_func ("/curve/special/intersections", test_curve_intersections);
return g_test_run ();
}