416 lines
12 KiB
C
416 lines
12 KiB
C
#include <gtk/gtk.h>
|
|
#include "gsk/gskcurveprivate.h"
|
|
|
|
static void
|
|
init_random_point (graphene_point_t *p)
|
|
{
|
|
p->x = g_test_rand_double_range (0, 1000);
|
|
p->y = g_test_rand_double_range (0, 1000);
|
|
}
|
|
|
|
static void
|
|
init_random_curve_with_op (GskCurve *curve,
|
|
GskPathOperation min_op,
|
|
GskPathOperation max_op)
|
|
{
|
|
switch (g_test_rand_int_range (min_op, max_op + 1))
|
|
{
|
|
case GSK_PATH_LINE:
|
|
{
|
|
graphene_point_t p[2];
|
|
|
|
init_random_point (&p[0]);
|
|
init_random_point (&p[1]);
|
|
gsk_curve_init (curve, gsk_pathop_encode (GSK_PATH_LINE, p));
|
|
}
|
|
break;
|
|
|
|
case GSK_PATH_QUAD:
|
|
{
|
|
graphene_point_t p[3];
|
|
|
|
init_random_point (&p[0]);
|
|
init_random_point (&p[1]);
|
|
init_random_point (&p[2]);
|
|
gsk_curve_init (curve, gsk_pathop_encode (GSK_PATH_QUAD, p));
|
|
}
|
|
break;
|
|
|
|
case GSK_PATH_CUBIC:
|
|
{
|
|
graphene_point_t p[4];
|
|
|
|
init_random_point (&p[0]);
|
|
init_random_point (&p[1]);
|
|
init_random_point (&p[2]);
|
|
init_random_point (&p[3]);
|
|
gsk_curve_init (curve, gsk_pathop_encode (GSK_PATH_CUBIC, p));
|
|
}
|
|
break;
|
|
|
|
case GSK_PATH_CONIC:
|
|
{
|
|
graphene_point_t p[4];
|
|
|
|
init_random_point (&p[0]);
|
|
init_random_point (&p[1]);
|
|
p[2].x = g_test_rand_double_range (0.2, 20);
|
|
p[2].y = 0.f;
|
|
init_random_point (&p[3]);
|
|
gsk_curve_init (curve, gsk_pathop_encode (GSK_PATH_CONIC, p));
|
|
}
|
|
break;
|
|
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_random_curve (GskCurve *curve)
|
|
{
|
|
init_random_curve_with_op (curve, GSK_PATH_LINE, GSK_PATH_CONIC);
|
|
}
|
|
|
|
static void
|
|
test_curve_tangents (void)
|
|
{
|
|
for (int i = 0; i < 100; i++)
|
|
{
|
|
GskCurve c;
|
|
graphene_vec2_t vec, exact;
|
|
|
|
init_random_curve (&c);
|
|
|
|
gsk_curve_get_tangent (&c, 0, &vec);
|
|
g_assert_cmpfloat_with_epsilon (graphene_vec2_length (&vec), 1.0f, 0.00001);
|
|
gsk_curve_get_start_tangent (&c, &exact);
|
|
g_assert_cmpfloat_with_epsilon (graphene_vec2_length (&exact), 1.0f, 0.00001);
|
|
g_assert_true (graphene_vec2_near (&vec, &exact, 0.05));
|
|
|
|
gsk_curve_get_tangent (&c, 1, &vec);
|
|
g_assert_cmpfloat_with_epsilon (graphene_vec2_length (&vec), 1.0f, 0.00001);
|
|
gsk_curve_get_end_tangent (&c, &exact);
|
|
g_assert_cmpfloat_with_epsilon (graphene_vec2_length (&exact), 1.0f, 0.00001);
|
|
g_assert_true (graphene_vec2_near (&vec, &exact, 0.05));
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_curve_points (void)
|
|
{
|
|
for (int i = 0; i < 100; i++)
|
|
{
|
|
GskCurve c;
|
|
graphene_point_t p;
|
|
|
|
init_random_curve (&c);
|
|
|
|
/* We could assert equality here because evaluating the polynomials with 0
|
|
* has no effect on accuracy, but for arcs, we use trigonometric functions,
|
|
* so allow a small error.
|
|
*/
|
|
gsk_curve_get_point (&c, 0, &p);
|
|
g_assert_true (graphene_point_near (gsk_curve_get_start_point (&c), &p, 0.001));
|
|
|
|
/* But here we evaluate the polynomials with 1 which gives the highest possible
|
|
* accuracy error. So we'll just be generous here.
|
|
*/
|
|
gsk_curve_get_point (&c, 1, &p);
|
|
g_assert_true (graphene_point_near (gsk_curve_get_end_point (&c), &p, 0.05));
|
|
}
|
|
}
|
|
|
|
/* at this point the subdivision stops and the decomposer
|
|
* violates tolerance rules
|
|
*/
|
|
#define MIN_PROGRESS (1/1024.f)
|
|
|
|
typedef struct
|
|
{
|
|
graphene_point_t p;
|
|
float t;
|
|
} PointOnLine;
|
|
|
|
static gboolean
|
|
add_line_to_array (const graphene_point_t *from,
|
|
const graphene_point_t *to,
|
|
float from_progress,
|
|
float to_progress,
|
|
GskCurveLineReason reason,
|
|
gpointer user_data)
|
|
{
|
|
GArray *array = user_data;
|
|
PointOnLine *last = &g_array_index (array, PointOnLine, array->len - 1);
|
|
|
|
g_assert_true (array->len > 0);
|
|
g_assert_cmpfloat (from_progress, >=, 0.0f);
|
|
g_assert_cmpfloat (from_progress, <, to_progress);
|
|
g_assert_cmpfloat (to_progress, <=, 1.0f);
|
|
|
|
g_assert_true (graphene_point_equal (&last->p, from));
|
|
g_assert_cmpfloat (last->t, ==, from_progress);
|
|
|
|
g_array_append_vals (array, (PointOnLine[1]) { { *to, to_progress } }, 1);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
test_curve_decompose (void)
|
|
{
|
|
static const float tolerance = 0.5;
|
|
|
|
for (int i = 0; i < 100; i++)
|
|
{
|
|
GArray *array;
|
|
GskCurve c;
|
|
|
|
init_random_curve (&c);
|
|
|
|
array = g_array_new (FALSE, FALSE, sizeof (PointOnLine));
|
|
g_array_append_vals (array, (PointOnLine[1]) { { *gsk_curve_get_start_point (&c), 0.f } }, 1);
|
|
|
|
g_assert_true (gsk_curve_decompose (&c, tolerance, add_line_to_array, array));
|
|
|
|
g_assert_cmpint (array->len, >=, 2); /* We at least got a line to the end */
|
|
g_assert_cmpfloat (g_array_index (array, PointOnLine, array->len - 1).t, ==, 1.0);
|
|
|
|
for (int j = 0; j < array->len; j++)
|
|
{
|
|
PointOnLine *pol = &g_array_index (array, PointOnLine, j);
|
|
graphene_point_t p;
|
|
|
|
/* Check that the points we got are actually on the line */
|
|
gsk_curve_get_point (&c, pol->t, &p);
|
|
g_assert_true (graphene_point_near (&pol->p, &p, 0.05));
|
|
|
|
/* Check that the mid point is not further than the tolerance */
|
|
if (j > 0)
|
|
{
|
|
PointOnLine *last = &g_array_index (array, PointOnLine, j - 1);
|
|
graphene_point_t mid;
|
|
|
|
if (pol->t - last->t > MIN_PROGRESS)
|
|
{
|
|
graphene_point_interpolate (&last->p, &pol->p, 0.5, &mid);
|
|
gsk_curve_get_point (&c, (pol->t + last->t) / 2, &p);
|
|
/* The decomposer does this cheaper Manhattan distance test,
|
|
* so graphene_point_near() does not work */
|
|
g_assert_cmpfloat (fabs (mid.x - p.x), <=, tolerance + 0.0002);
|
|
g_assert_cmpfloat (fabs (mid.y - p.y), <=, tolerance + 0.0002);
|
|
}
|
|
}
|
|
}
|
|
|
|
g_array_unref (array);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
add_curve_to_array (GskPathOperation op,
|
|
const graphene_point_t *pts,
|
|
gsize n_pts,
|
|
float weight,
|
|
gpointer user_data)
|
|
{
|
|
GArray *array = user_data;
|
|
GskCurve c;
|
|
|
|
gsk_curve_init_foreach (&c, op, pts, n_pts, weight);
|
|
g_array_append_val (array, c);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
test_curve_decompose_into (GskPathForeachFlags flags)
|
|
{
|
|
for (int i = 0; i < 100; i++)
|
|
{
|
|
GskCurve c;
|
|
GskPathBuilder *builder;
|
|
const graphene_point_t *s;
|
|
GskPath *path;
|
|
GArray *array;
|
|
|
|
init_random_curve (&c);
|
|
|
|
builder = gsk_path_builder_new ();
|
|
|
|
s = gsk_curve_get_start_point (&c);
|
|
gsk_path_builder_move_to (builder, s->x, s->y);
|
|
gsk_curve_builder_to (&c, builder);
|
|
path = gsk_path_builder_free_to_path (builder);
|
|
|
|
array = g_array_new (FALSE, FALSE, sizeof (GskCurve));
|
|
|
|
g_assert_true (gsk_curve_decompose_curve (&c, flags, 0.1, add_curve_to_array, array));
|
|
|
|
g_assert_cmpint (array->len, >=, 1);
|
|
|
|
for (int j = 0; j < array->len; j++)
|
|
{
|
|
GskCurve *c2 = &g_array_index (array, GskCurve, j);
|
|
|
|
switch (c2->op)
|
|
{
|
|
case GSK_PATH_MOVE:
|
|
case GSK_PATH_CLOSE:
|
|
case GSK_PATH_LINE:
|
|
break;
|
|
case GSK_PATH_QUAD:
|
|
g_assert_true (flags & GSK_PATH_FOREACH_ALLOW_QUAD);
|
|
break;
|
|
case GSK_PATH_CUBIC:
|
|
g_assert_true (flags & GSK_PATH_FOREACH_ALLOW_CUBIC);
|
|
break;
|
|
case GSK_PATH_CONIC:
|
|
g_assert_true (flags & GSK_PATH_FOREACH_ALLOW_CONIC);
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
g_array_unref (array);
|
|
gsk_path_unref (path);
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_curve_decompose_into_line (void)
|
|
{
|
|
test_curve_decompose_into (0);
|
|
}
|
|
|
|
static void
|
|
test_curve_decompose_into_quad (void)
|
|
{
|
|
test_curve_decompose_into (GSK_PATH_FOREACH_ALLOW_QUAD);
|
|
}
|
|
|
|
static void
|
|
test_curve_decompose_into_cubic (void)
|
|
{
|
|
test_curve_decompose_into (GSK_PATH_FOREACH_ALLOW_CUBIC);
|
|
}
|
|
|
|
/* Some sanity checks for splitting curves. */
|
|
static void
|
|
test_curve_split (void)
|
|
{
|
|
for (int i = 0; i < 20; i++)
|
|
{
|
|
GskCurve c;
|
|
|
|
init_random_curve (&c);
|
|
|
|
for (int j = 0; j < 20; j++)
|
|
{
|
|
GskCurve c1, c2;
|
|
graphene_point_t p;
|
|
graphene_vec2_t t, t1, t2;
|
|
float split;
|
|
|
|
split = g_test_rand_double_range (0.1, 0.9);
|
|
|
|
gsk_curve_split (&c, split, &c1, &c2);
|
|
|
|
g_assert_true (c1.op == c.op);
|
|
g_assert_true (c2.op == c.op);
|
|
|
|
g_assert_true (graphene_point_near (gsk_curve_get_start_point (&c),
|
|
gsk_curve_get_start_point (&c1), 0.005));
|
|
g_assert_true (graphene_point_near (gsk_curve_get_end_point (&c1),
|
|
gsk_curve_get_start_point (&c2), 0.005));
|
|
g_assert_true (graphene_point_near (gsk_curve_get_end_point (&c),
|
|
gsk_curve_get_end_point (&c2), 0.005));
|
|
gsk_curve_get_point (&c, split, &p);
|
|
gsk_curve_get_tangent (&c, split, &t);
|
|
g_assert_true (graphene_point_near (gsk_curve_get_end_point (&c1), &p, 0.005));
|
|
g_assert_true (graphene_point_near (gsk_curve_get_start_point (&c2), &p, 0.005));
|
|
|
|
gsk_curve_get_start_tangent (&c, &t1);
|
|
gsk_curve_get_start_tangent (&c1, &t2);
|
|
g_assert_true (graphene_vec2_near (&t1, &t2, 0.005));
|
|
gsk_curve_get_end_tangent (&c1, &t1);
|
|
gsk_curve_get_start_tangent (&c2, &t2);
|
|
g_assert_true (graphene_vec2_near (&t1, &t2, 0.005));
|
|
g_assert_true (graphene_vec2_near (&t, &t1, 0.005));
|
|
g_assert_true (graphene_vec2_near (&t, &t2, 0.005));
|
|
gsk_curve_get_end_tangent (&c, &t1);
|
|
gsk_curve_get_end_tangent (&c2, &t2);
|
|
g_assert_true (graphene_vec2_near (&t1, &t2, 0.005));
|
|
|
|
#if 0
|
|
/* hard to guarantee this for totally random random curves */
|
|
g_assert_cmpfloat_with_epsilon (gsk_curve_get_length (&c),
|
|
gsk_curve_get_length (&c1) + gsk_curve_get_length (&c2),
|
|
1);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_curve_derivative (void)
|
|
{
|
|
GskCurve c;
|
|
float t;
|
|
graphene_vec2_t t1, t2;
|
|
graphene_point_t p;
|
|
|
|
for (int i = 0; i < 100; i++)
|
|
{
|
|
init_random_curve (&c);
|
|
|
|
for (int j = 0; j < 100; j++)
|
|
{
|
|
t = g_test_rand_double_range (0, 1);
|
|
gsk_curve_get_derivative_at (&c, t, &p);
|
|
gsk_curve_get_tangent (&c, t, &t1);
|
|
graphene_vec2_init (&t2, p.x, p.y);
|
|
graphene_vec2_normalize (&t2, &t2);
|
|
g_assert_true (graphene_vec2_near (&t1, &t2, 0.1));
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_curve_length (void)
|
|
{
|
|
GskCurve c;
|
|
float l, l0;
|
|
|
|
for (int i = 0; i < 1000; i++)
|
|
{
|
|
init_random_curve (&c);
|
|
|
|
l = gsk_curve_get_length (&c);
|
|
l0 = graphene_point_distance (gsk_curve_get_start_point (&c),
|
|
gsk_curve_get_end_point (&c),
|
|
NULL, NULL);
|
|
g_assert_true (l >= l0 - 0.001);
|
|
if (c.op == GSK_PATH_LINE)
|
|
g_assert_cmpfloat_with_epsilon (l, l0, 0.001);
|
|
}
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
(g_test_init) (&argc, &argv, NULL);
|
|
|
|
g_test_add_func ("/curve/points", test_curve_points);
|
|
g_test_add_func ("/curve/tangents", test_curve_tangents);
|
|
g_test_add_func ("/curve/decompose", test_curve_decompose);
|
|
g_test_add_func ("/curve/decompose-line", test_curve_decompose_into_line);
|
|
g_test_add_func ("/curve/decompose-quad", test_curve_decompose_into_quad);
|
|
g_test_add_func ("/curve/decompose-cubic", test_curve_decompose_into_cubic);
|
|
g_test_add_func ("/curve/split", test_curve_split);
|
|
g_test_add_func ("/curve/derivative", test_curve_derivative);
|
|
g_test_add_func ("/curve/length", test_curve_length);
|
|
|
|
return g_test_run ();
|
|
}
|