/* * Copyright © 2020 Benjamin Otte * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . * * Authors: Benjamin Otte */ #include "config.h" #include "gskpathprivate.h" #include "gskcurveprivate.h" #include "gskpathbuilder.h" #include "gskpathpoint.h" #include "gskcontourprivate.h" /** * GskPath: * * A `GskPath` describes lines and curves that are more complex * than simple rectangles. * * Paths can used for rendering (filling or stroking) and for animations * (e.g. as trajectories). * * `GskPath` is an immutable, opaque, reference-counted struct. * After creation, you cannot change the types it represents. Instead, * new `GskPath` objects have to be created. The [struct@Gsk.PathBuilder] * structure is meant to help in this endeavor. * * Conceptually, a path consists of zero or more contours (continous, connected * curves), each of which may or may not be closed. Contours are typically * constructed from Bézier segments. * * * * A Path * * * Since: 4.14 */ struct _GskPath { /*< private >*/ guint ref_count; GskPathFlags flags; gsize n_contours; GskContour *contours[]; /* followed by the contours data */ }; G_DEFINE_BOXED_TYPE (GskPath, gsk_path, gsk_path_ref, gsk_path_unref) /* {{{ Private API */ GskPath * gsk_path_new_from_contours (const GSList *contours) { GskPath *path; const GSList *l; gsize size; gsize n_contours; guint8 *contour_data; GskPathFlags flags; flags = GSK_PATH_CLOSED | GSK_PATH_FLAT; size = 0; n_contours = 0; for (l = contours; l; l = l->next) { GskContour *contour = l->data; n_contours++; size += sizeof (GskContour *); size += gsk_contour_get_size (contour); flags &= gsk_contour_get_flags (contour); } path = g_malloc0 (sizeof (GskPath) + size); path->ref_count = 1; path->flags = flags; path->n_contours = n_contours; contour_data = (guint8 *) &path->contours[n_contours]; n_contours = 0; for (l = contours; l; l = l->next) { GskContour *contour = l->data; path->contours[n_contours] = (GskContour *) contour_data; gsk_contour_copy ((GskContour *) contour_data, contour); size = gsk_contour_get_size (contour); contour_data += size; n_contours++; } return path; } const GskContour * gsk_path_get_contour (const GskPath *self, gsize i) { if (i < self->n_contours) return self->contours[i]; else return NULL; } GskPathFlags gsk_path_get_flags (const GskPath *self) { return self->flags; } gsize gsk_path_get_n_contours (const GskPath *self) { return self->n_contours; } /* }}} */ /* {{{ Public API */ /** * gsk_path_ref: * @self: a `GskPath` * * Increases the reference count of a `GskPath` by one. * * Returns: the passed in `GskPath`. * * Since: 4.14 */ GskPath * gsk_path_ref (GskPath *self) { g_return_val_if_fail (self != NULL, NULL); self->ref_count++; return self; } /** * gsk_path_unref: * @self: a `GskPath` * * Decreases the reference count of a `GskPath` by one. * * If the resulting reference count is zero, frees the path. * * Since: 4.14 */ void gsk_path_unref (GskPath *self) { g_return_if_fail (self != NULL); g_return_if_fail (self->ref_count > 0); self->ref_count--; if (self->ref_count > 0) return; g_free (self); } /** * gsk_path_print: * @self: a `GskPath` * @string: The string to print into * * Converts @self into a human-readable string representation suitable * for printing. * * The string is compatible with (a superset of) * [SVG path syntax](https://www.w3.org/TR/SVG11/paths.html#PathData), * see [func@Gsk.Path.parse] for a summary of the syntax. * * Since: 4.14 */ void gsk_path_print (GskPath *self, GString *string) { gsize i; g_return_if_fail (self != NULL); g_return_if_fail (string != NULL); for (i = 0; i < self->n_contours; i++) { if (i > 0) g_string_append_c (string, ' '); gsk_contour_print (self->contours[i], string); } } /** * gsk_path_to_string: * @self: a `GskPath` * * Converts the path into a string that is suitable for printing. * * You can use this function in a debugger to get a quick overview * of the path. * * This is a wrapper around [method@Gsk.Path.print], see that function * for details. * * Returns: A new string for @self * * Since: 4.14 */ char * gsk_path_to_string (GskPath *self) { GString *string; g_return_val_if_fail (self != NULL, NULL); string = g_string_new (""); gsk_path_print (self, string); return g_string_free (string, FALSE); } static gboolean gsk_path_to_cairo_add_op (GskPathOperation op, const graphene_point_t *pts, gsize n_pts, float weight, gpointer cr) { switch (op) { case GSK_PATH_MOVE: cairo_move_to (cr, pts[0].x, pts[0].y); break; case GSK_PATH_CLOSE: cairo_close_path (cr); break; case GSK_PATH_LINE: cairo_line_to (cr, pts[1].x, pts[1].y); break; case GSK_PATH_CUBIC: cairo_curve_to (cr, pts[1].x, pts[1].y, pts[2].x, pts[2].y, pts[3].x, pts[3].y); break; case GSK_PATH_QUAD: case GSK_PATH_CONIC: default: g_assert_not_reached (); return FALSE; } return TRUE; } /** * gsk_path_to_cairo: * @self: a `GskPath` * @cr: a cairo context * * Appends the given @path to the given cairo context for drawing * with Cairo. * * This may cause some suboptimal conversions to be performed as * Cairo does not support all features of `GskPath`. * * This function does not clear the existing Cairo path. Call * cairo_new_path() if you want this. * * Since: 4.14 */ void gsk_path_to_cairo (GskPath *self, cairo_t *cr) { g_return_if_fail (self != NULL); g_return_if_fail (cr != NULL); gsk_path_foreach_with_tolerance (self, GSK_PATH_FOREACH_ALLOW_CUBIC, cairo_get_tolerance (cr), gsk_path_to_cairo_add_op, cr); } /** * gsk_path_is_empty: * @self: a `GskPath` * * Checks if the path is empty, i.e. contains no lines or curves. * * Returns: `TRUE` if the path is empty * * Since: 4.14 */ gboolean gsk_path_is_empty (GskPath *self) { g_return_val_if_fail (self != NULL, FALSE); return self->n_contours == 0; } /** * gsk_path_is_closed: * @self: a `GskPath` * * Returns if the path represents a single closed * contour. * * Returns: `TRUE` if the path is closed * * Since: 4.14 */ gboolean gsk_path_is_closed (GskPath *self) { g_return_val_if_fail (self != NULL, FALSE); /* XXX: is the empty path closed? Currently it's not */ if (self->n_contours != 1) return FALSE; return gsk_contour_get_flags (self->contours[0]) & GSK_PATH_CLOSED ? TRUE : FALSE; } /** * gsk_path_get_bounds: * @self: a `GskPath` * @bounds: (out caller-allocates): the bounds of the given path * * Computes the bounds of the given path. * * The returned bounds may be larger than necessary, because this * function aims to be fast, not accurate. The bounds are guaranteed * to contain the path. * * It is possible that the returned rectangle has 0 width and/or height. * This can happen when the path only describes a point or an * axis-aligned line. * * If the path is empty, `FALSE` is returned and @bounds are set to * graphene_rect_zero(). This is different from the case where the path * is a single point at the origin, where the @bounds will also be set to * the zero rectangle but `TRUE` will be returned. * * Returns: `TRUE` if the path has bounds, `FALSE` if the path is known * to be empty and have no bounds. * * Since: 4.14 */ gboolean gsk_path_get_bounds (GskPath *self, graphene_rect_t *bounds) { GskBoundingBox b; g_return_val_if_fail (self != NULL, FALSE); g_return_val_if_fail (bounds != NULL, FALSE); if (self->n_contours == 0) { graphene_rect_init_from_rect (bounds, graphene_rect_zero ()); return FALSE; } gsk_contour_get_bounds (self->contours[0], &b); for (gsize i = 1; i < self->n_contours; i++) { GskBoundingBox tmp; gsk_contour_get_bounds (self->contours[i], &tmp); gsk_bounding_box_union (&b, &tmp, &b); } gsk_bounding_box_to_rect (&b, bounds); return TRUE; } /** * gsk_path_get_stroke_bounds: * @self: a #GtkPath * @stroke: stroke parameters * @bounds: (out caller-allocates): the bounds to fill in * * Computes the bounds for stroking the given path with the * parameters in @stroke. * * The returned bounds may be larger than necessary, because this * function aims to be fast, not accurate. The bounds are guaranteed * to contain the area affected by the stroke, including protrusions * like miters. * * Returns: `TRUE` if the path has bounds, `FALSE` if the path is known * to be empty and have no bounds. * * Since: 4.14 */ gboolean gsk_path_get_stroke_bounds (GskPath *self, const GskStroke *stroke, graphene_rect_t *bounds) { GskBoundingBox b; g_return_val_if_fail (self != NULL, FALSE); g_return_val_if_fail (bounds != NULL, FALSE); if (self->n_contours == 0) { graphene_rect_init_from_rect (bounds, graphene_rect_zero ()); return FALSE; } gsk_contour_get_stroke_bounds (self->contours[0], stroke, &b); for (gsize i = 1; i < self->n_contours; i++) { GskBoundingBox tmp; if (gsk_contour_get_stroke_bounds (self->contours[i], stroke, &tmp)) gsk_bounding_box_union (&b, &tmp, &b); } gsk_bounding_box_to_rect (&b, bounds); return TRUE; } /** * gsk_path_in_fill: * @self: a `GskPath` * @point: the point to test * @fill_rule: the fill rule to follow * * Returns whether the given point is inside the area * that would be affected if the path was filled according * to @fill_rule. * * Note that this function assumes that filling a contour * implicitly closes it. * * Returns: `TRUE` if @point is inside * * Since: 4.14 */ gboolean gsk_path_in_fill (GskPath *self, const graphene_point_t *point, GskFillRule fill_rule) { int winding = 0; for (int i = 0; i < self->n_contours; i++) winding += gsk_contour_get_winding (self->contours[i], point); switch (fill_rule) { case GSK_FILL_RULE_EVEN_ODD: return winding & 1; case GSK_FILL_RULE_WINDING: return winding != 0; default: g_assert_not_reached (); } } /** * gsk_path_get_start_point: * @self: a `GskPath` * @result: (out caller-allocates): return location for point * * Gets the start point of the path. * * An empty path has no points, so `FALSE` * is returned in this case. * * Returns: `TRUE` if @result was filled * * Since: 4.14 */ gboolean gsk_path_get_start_point (GskPath *self, GskPathPoint *result) { g_return_val_if_fail (self != NULL, FALSE); g_return_val_if_fail (result != NULL, FALSE); if (self->n_contours == 0) return FALSE; /* Conceptually, there is always a move at the * beginning, which jumps from where to the start * point of the contour, so we use idx == 1 here. */ result->contour = 0; result->idx = 1; result->t = 0; return TRUE; } /** * gsk_path_get_end_point: * @self: a `GskPath` * @result: (out caller-allocates): return location for point * * Gets the end point of the path. * * An empty path has no points, so `FALSE` * is returned in this case. * * Returns: `TRUE` if @result was filled * * Since: 4.14 */ gboolean gsk_path_get_end_point (GskPath *self, GskPathPoint *result) { g_return_val_if_fail (self != NULL, FALSE); g_return_val_if_fail (result != NULL, FALSE); if (self->n_contours == 0) return FALSE; result->contour = self->n_contours - 1; result->idx = gsk_contour_get_n_ops (self->contours[self->n_contours - 1]) - 1; result->t = 1; return TRUE; } /** * gsk_path_get_closest_point: * @self: a `GskPath` * @point: the point * @threshold: maximum allowed distance * @result: (out caller-allocates): return location for the closest point * @distance: (out) (optional): return location for the distance * * Computes the closest point on the path to the given point * and sets the @result to it. * * If there is no point closer than the given threshold, * `FALSE` is returned. * * Returns: `TRUE` if @point was set to the closest point * on @self, `FALSE` if no point is closer than @threshold * * Since: 4.14 */ gboolean gsk_path_get_closest_point (GskPath *self, const graphene_point_t *point, float threshold, GskPathPoint *result, float *distance) { gboolean found; g_return_val_if_fail (self != NULL, FALSE); g_return_val_if_fail (point != NULL, FALSE); g_return_val_if_fail (threshold >= 0, FALSE); g_return_val_if_fail (result != NULL, FALSE); found = FALSE; for (int i = 0; i < self->n_contours; i++) { float dist; if (gsk_contour_get_closest_point (self->contours[i], point, threshold, result, &dist)) { found = TRUE; g_assert (0 <= result->t && result->t <= 1); result->contour = i; threshold = dist; if (distance) *distance = dist; } } return found; } /* }}} */ /* {{{ Foreach and decomposition */ /** * gsk_path_foreach: * @self: a `GskPath` * @flags: flags to pass to the foreach function. See [flags@Gsk.PathForeachFlags] * for details about flags * @func: (scope call) (closure user_data): the function to call for operations * @user_data: (nullable): user data passed to @func * * Calls @func for every operation of the path. * * Note that this may only approximate @self, because paths can contain * optimizations for various specialized contours, and depending on the * @flags, the path may be decomposed into simpler curves than the ones * that it contained originally. * * This function serves two purposes: * * - When the @flags allow everything, it provides access to the raw, * unmodified data of the path. * - When the @flags disallow certain operations, it provides * an approximation of the path using just the allowed operations. * * Returns: `FALSE` if @func returned FALSE`, `TRUE` otherwise. * * Since: 4.14 */ gboolean gsk_path_foreach (GskPath *self, GskPathForeachFlags flags, GskPathForeachFunc func, gpointer user_data) { g_return_val_if_fail (self != NULL, FALSE); g_return_val_if_fail (func, FALSE); return gsk_path_foreach_with_tolerance (self, flags, GSK_PATH_TOLERANCE_DEFAULT, func, user_data); } typedef struct _GskPathForeachTrampoline GskPathForeachTrampoline; struct _GskPathForeachTrampoline { GskPathForeachFlags flags; double tolerance; GskPathForeachFunc func; gpointer user_data; }; static gboolean gsk_path_foreach_trampoline_add_line (const graphene_point_t *from, const graphene_point_t *to, float from_progress, float to_progress, GskCurveLineReason reason, gpointer data) { GskPathForeachTrampoline *trampoline = data; return trampoline->func (GSK_PATH_LINE, (graphene_point_t[2]) { *from, *to }, 2, 0.f, trampoline->user_data); } static gboolean gsk_path_foreach_trampoline_add_curve (GskPathOperation op, const graphene_point_t *pts, gsize n_pts, float weight, gpointer data) { GskPathForeachTrampoline *trampoline = data; return trampoline->func (op, pts, n_pts, weight, trampoline->user_data); } static gboolean gsk_path_foreach_trampoline (GskPathOperation op, const graphene_point_t *pts, gsize n_pts, float weight, gpointer data) { GskPathForeachTrampoline *trampoline = data; switch (op) { case GSK_PATH_MOVE: case GSK_PATH_CLOSE: case GSK_PATH_LINE: return trampoline->func (op, pts, n_pts, weight, trampoline->user_data); case GSK_PATH_QUAD: { GskCurve curve; if (trampoline->flags & GSK_PATH_FOREACH_ALLOW_QUAD) return trampoline->func (op, pts, n_pts, weight, trampoline->user_data); else if (trampoline->flags & GSK_PATH_FOREACH_ALLOW_CUBIC) { return trampoline->func (GSK_PATH_CUBIC, (graphene_point_t[4]) { pts[0], GRAPHENE_POINT_INIT ((pts[0].x + 2 * pts[1].x) / 3, (pts[0].y + 2 * pts[1].y) / 3), GRAPHENE_POINT_INIT ((pts[2].x + 2 * pts[1].x) / 3, (pts[2].y + 2 * pts[1].y) / 3), pts[2], }, 4, weight, trampoline->user_data); } gsk_curve_init (&curve, gsk_pathop_encode (GSK_PATH_QUAD, pts)); return gsk_curve_decompose (&curve, trampoline->tolerance, gsk_path_foreach_trampoline_add_line, trampoline); } case GSK_PATH_CUBIC: { GskCurve curve; if (trampoline->flags & GSK_PATH_FOREACH_ALLOW_CUBIC) return trampoline->func (op, pts, n_pts, weight, trampoline->user_data); gsk_curve_init (&curve, gsk_pathop_encode (GSK_PATH_CUBIC, pts)); if (trampoline->flags & (GSK_PATH_FOREACH_ALLOW_QUAD|GSK_PATH_FOREACH_ALLOW_CONIC)) return gsk_curve_decompose_curve (&curve, trampoline->flags, trampoline->tolerance, gsk_path_foreach_trampoline_add_curve, trampoline); return gsk_curve_decompose (&curve, trampoline->tolerance, gsk_path_foreach_trampoline_add_line, trampoline); } case GSK_PATH_CONIC: { GskCurve curve; if (trampoline->flags & GSK_PATH_FOREACH_ALLOW_CONIC) return trampoline->func (op, pts, n_pts, weight, trampoline->user_data); gsk_curve_init (&curve, gsk_pathop_encode (GSK_PATH_CONIC, (graphene_point_t[4]) { pts[0], pts[1], { weight, 0.f }, pts[2] } )); if (trampoline->flags & (GSK_PATH_FOREACH_ALLOW_QUAD|GSK_PATH_FOREACH_ALLOW_CUBIC)) return gsk_curve_decompose_curve (&curve, trampoline->flags, trampoline->tolerance, gsk_path_foreach_trampoline_add_curve, trampoline); return gsk_curve_decompose (&curve, trampoline->tolerance, gsk_path_foreach_trampoline_add_line, trampoline); } default: g_assert_not_reached (); return FALSE; } } #define ALLOW_ANY (GSK_PATH_FOREACH_ALLOW_QUAD | \ GSK_PATH_FOREACH_ALLOW_CUBIC | \ GSK_PATH_FOREACH_ALLOW_CONIC) gboolean gsk_path_foreach_with_tolerance (GskPath *self, GskPathForeachFlags flags, double tolerance, GskPathForeachFunc func, gpointer user_data) { GskPathForeachTrampoline trampoline; gsize i; /* If we need to massage the data, set up a trampoline here */ if ((flags & ALLOW_ANY) != ALLOW_ANY) { trampoline = (GskPathForeachTrampoline) { flags, tolerance, func, user_data }; func = gsk_path_foreach_trampoline; user_data = &trampoline; } for (i = 0; i < self->n_contours; i++) { if (!gsk_contour_foreach (self->contours[i], func, user_data)) return FALSE; } return TRUE; } /* }}} */ /* {{{ Parser and utilities */ static void skip_whitespace (const char **p) { while (g_ascii_isspace (**p)) (*p)++; } static void skip_optional_comma (const char **p) { skip_whitespace (p); if (**p == ',') (*p)++; } static gboolean parse_number (const char **p, double *c) { char *e; *c = g_ascii_strtod (*p, &e); if (e == *p) return FALSE; *p = e; skip_optional_comma (p); return TRUE; } static gboolean parse_coordinate (const char **p, double *c) { return parse_number (p, c); } static gboolean parse_coordinate_pair (const char **p, double *x, double *y) { double xx, yy; const char *o = *p; if (!parse_coordinate (p, &xx)) { *p = o; return FALSE; } if (!parse_coordinate (p, &yy)) { *p = o; return FALSE; } *x = xx; *y = yy; return TRUE; } static gboolean parse_nonnegative_number (const char **p, double *x) { const char *o = *p; double n; if (!parse_number (p, &n)) return FALSE; if (n < 0) { *p = o; return FALSE; } *x = n; return TRUE; } /* This fixes a flaw in our use of strchr() below: * * If p already points at the end of the string, * we misinterpret strchr ("xyz", *p) returning * non-NULL to mean that we can increment p. * * But strchr() will return a pointer to the * final NUL byte in this case, and we walk off * the end of the string. Oops */ static inline char * _strchr (const char *str, int c) { if (c == 0) return NULL; else return strchr (str, c); } static gboolean parse_flag (const char **p, gboolean *f) { skip_whitespace (p); if (_strchr ("01", **p)) { *f = **p == '1'; (*p)++; skip_optional_comma (p); return TRUE; } return FALSE; } static gboolean parse_command (const char **p, char *cmd) { char *s; const char *allowed; if (*cmd == 'X') allowed = "mM"; else allowed = "mMhHvVzZlLcCsStTqQaAoO"; skip_whitespace (p); s = _strchr (allowed, **p); if (s) { *cmd = *s; (*p)++; return TRUE; } return FALSE; } static gboolean parse_string (const char **p, const char *s) { int len = strlen (s); if (strncmp (*p, s, len) != 0) return FALSE; (*p) += len; return TRUE; } #define NEAR(x, y) (fabs ((x) - (y)) < 0.001) static gboolean is_rect (double x0, double y0, double x1, double y1, double x2, double y2, double x3, double y3) { return NEAR (x0, x3) && NEAR (x1, x2) && NEAR (y0, y1) && NEAR (y2, y3) && x0 < x1 && y1 < y2; } static gboolean is_line (double x0, double y0, double x1, double y1, double x2, double y2, double x3, double y3) { if (NEAR (y0, y3)) return x0 <= x1 && x1 <= x2 && x2 <= x3 && NEAR (y0, y1) && NEAR (y0, y2) && NEAR (y0, y3); else return y0 <= y1 && y1 <= y2 && y2 <= y3 && NEAR (x0, x1) && NEAR (x0, x2) && NEAR (x0, x3); } static gboolean parse_rectangle (const char **p, double *x, double *y, double *w, double *h) { const char *o = *p; double w2; if (parse_coordinate_pair (p, x, y) && parse_string (p, "h") && parse_coordinate (p, w) && parse_string (p, "v") && parse_coordinate (p, h) && parse_string (p, "h") && parse_coordinate (p, &w2) && parse_string (p, "z") && w2 == -*w && *w >= 0 && *h >= 0) { skip_whitespace (p); return TRUE; } *p = o; return FALSE; } static gboolean parse_circle (const char **p, double *cx, double *cy, double *r) { const char *o = *p; double x0, y0, x1, y1, x2, y2, x3, y3; double x4, y4, x5, y5, x6, y6, x7, y7; double x8, y8, w0, w1, w2, w3; double rr; if (parse_coordinate_pair (p, &x0, &y0) && parse_string (p, "o") && parse_coordinate_pair (p, &x1, &y1) && parse_coordinate_pair (p, &x2, &y2) && parse_nonnegative_number (p, &w0) && parse_string (p, "o") && parse_coordinate_pair (p, &x3, &y3) && parse_coordinate_pair (p, &x4, &y4) && parse_nonnegative_number (p, &w1) && parse_string (p, "o") && parse_coordinate_pair (p, &x5, &y5) && parse_coordinate_pair (p, &x6, &y6) && parse_nonnegative_number (p, &w2) && parse_string (p, "o") && parse_coordinate_pair (p, &x7, &y7) && parse_coordinate_pair (p, &x8, &y8) && parse_nonnegative_number (p, &w3) && parse_string (p, "z")) { rr = y1; if (x1 == 0 && y1 == rr && x2 == -rr && y2 == rr && x3 == -rr && y3 == 0 && x4 == -rr && y4 == -rr && x5 == 0 && y5 == -rr && x6 == rr && y6 == -rr && x7 == rr && y7 == 0 && x8 == rr && y8 == rr && NEAR (w0, M_SQRT1_2) && NEAR (w1, M_SQRT1_2) && NEAR (w2, M_SQRT1_2) && NEAR (w3, M_SQRT1_2)) { *cx = x0 - rr; *cy = y0; *r = rr; skip_whitespace (p); return TRUE; } } *p = o; return FALSE; } static gboolean parse_rounded_rect (const char **p, GskRoundedRect *rr) { const char *o = *p; double x0, y0, x1, y1, x2, y2, x3, y3; double x4, y4, x5, y5, x6, y6, x7, y7; double x8, y8, x9, y9, x10, y10, x11, y11; double x12, y12, w0, w1, w2, w3; if (parse_coordinate_pair (p, &x0, &y0) && parse_string (p, "L") && parse_coordinate_pair (p, &x1, &y1) && parse_string (p, "O") && parse_coordinate_pair (p, &x2, &y2) && parse_coordinate_pair (p, &x3, &y3) && parse_nonnegative_number (p, &w0) && parse_string (p, "L") && parse_coordinate_pair (p, &x4, &y4) && parse_string (p, "O") && parse_coordinate_pair (p, &x5, &y5) && parse_coordinate_pair (p, &x6, &y6) && parse_nonnegative_number (p, &w1) && parse_string (p, "L") && parse_coordinate_pair (p, &x7, &y7) && parse_string (p, "O") && parse_coordinate_pair (p, &x8, &y8) && parse_coordinate_pair (p, &x9, &y9) && parse_nonnegative_number (p, &w2) && parse_string (p, "L") && parse_coordinate_pair (p, &x10, &y10) && parse_string (p, "O") && parse_coordinate_pair (p, &x11, &y11) && parse_coordinate_pair (p, &x12, &y12) && parse_nonnegative_number (p, &w3) && parse_string (p, "Z")) { if (NEAR (x0, x12) && NEAR (y0, y12) && is_rect (x11, y11, x2, y2, x5, y5, x8, y8) && is_line (x11, y11, x0, y0, x1, y1, x2, y2) && is_line (x2, y2, x3, y3, x4, y4, x5, y5) && is_line (x8, y8, x7, y7, x6, y6, x5, y5) && is_line (x11, y11, x10, y10, x9, y9, x8, y8) && NEAR (w0, M_SQRT1_2) && NEAR (w1, M_SQRT1_2) && NEAR (w2, M_SQRT1_2) && NEAR (w3, M_SQRT1_2)) { rr->bounds = GRAPHENE_RECT_INIT (x11, y11, x5 - x11, y5 - y11); rr->corner[GSK_CORNER_TOP_LEFT] = GRAPHENE_SIZE_INIT (x12 - x11, y10 - y11); rr->corner[GSK_CORNER_TOP_RIGHT] = GRAPHENE_SIZE_INIT (x2 - x1, y3 - y2); rr->corner[GSK_CORNER_BOTTOM_RIGHT] = GRAPHENE_SIZE_INIT (x5 - x6, y5 - y4); rr->corner[GSK_CORNER_BOTTOM_LEFT] = GRAPHENE_SIZE_INIT (x7 - x8, y8 - y7); skip_whitespace (p); return TRUE; } } *p = o; return FALSE; } #undef NEAR /** * gsk_path_parse: * @string: a string * * This is a convenience function that constructs a `GskPath` * from a serialized form. * * The string is expected to be in (a superset of) * [SVG path syntax](https://www.w3.org/TR/SVG11/paths.html#PathData), * as e.g. produced by [method@Gsk.Path.to_string]. * * A high-level summary of the syntax: * * - `M x y` Move to `(x, y)` * - `L x y` Add a line from the current point to `(x, y)` * - `Q x1 y1 x2 y2` Add a quadratic Bézier from the current point to `(x2, y2)`, with control point `(x1, y1)` * - `C x1 y1 x2 y2 x3 y3` Add a cubic Bézier from the current point to `(x3, y3)`, with control points `(x1, y1)` and `(x2, y2)` * - `Z` Close the contour by drawing a line back to the start point * - `H x` Add a horizontal line from the current point to the given x value * - `V y` Add a vertical line from the current point to the given y value * - `T x2 y2` Add a quadratic Bézier, using the reflection of the previous segments' control point as control point * - `S x2 y2 x3 y3` Add a cubic Bézier, using the reflection of the previous segments' second control point as first control point * - `A rx ry r l s x y` Add an elliptical arc from the current point to `(x, y)` with radii rx and ry. See the SVG documentation for how the other parameters influence the arc. * - `O x1 y1 x2 y2 w` Add a rational quadratic Bézier from the current point to `(x2, y2)` with control point `(x1, y1)` and weight `w`. * * All the commands have lowercase variants that interpret coordinates * relative to the current point. * * The `O` command is an extension that is not supported in SVG. * * Returns: (nullable): a new `GskPath`, or `NULL` if @string could not be parsed * * Since: 4.14 */ GskPath * gsk_path_parse (const char *string) { GskPathBuilder *builder; double x, y; double prev_x1, prev_y1; double path_x, path_y; const char *p; char cmd; char prev_cmd; gboolean after_comma; gboolean repeat; builder = gsk_path_builder_new (); cmd = 'X'; path_x = path_y = 0; x = y = 0; prev_x1 = prev_y1 = 0; after_comma = FALSE; p = string; while (*p) { prev_cmd = cmd; repeat = !parse_command (&p, &cmd); if (after_comma && !repeat) goto error; switch (cmd) { case 'X': goto error; case 'Z': case 'z': if (repeat) goto error; else { gsk_path_builder_close (builder); x = path_x; y = path_y; } break; case 'M': case 'm': { double x1, y1, w, h, r; GskRoundedRect rr; /* Look for special contours */ if (parse_rectangle (&p, &x1, &y1, &w, &h)) { gsk_path_builder_add_rect (builder, &GRAPHENE_RECT_INIT (x1, y1, w, h)); if (_strchr ("zZX", prev_cmd)) { path_x = x1; path_y = y1; } x = x1; y = y1; } else if (parse_circle (&p, &x1, &y1, &r)) { gsk_path_builder_add_circle (builder, &GRAPHENE_POINT_INIT (x1, y1), r); if (_strchr ("zZX", prev_cmd)) { path_x = x1 + r; path_y = y1; } x = x1 + r; y = y1; } else if (parse_rounded_rect (&p, &rr)) { gsk_path_builder_add_rounded_rect (builder, &rr); if (_strchr ("zZX", prev_cmd)) { path_x = rr.bounds.origin.x + rr.corner[GSK_CORNER_TOP_LEFT].width; path_y = rr.bounds.origin.y; } x = rr.bounds.origin.x + rr.corner[GSK_CORNER_TOP_LEFT].width; y = rr.bounds.origin.y; } else if (parse_coordinate_pair (&p, &x1, &y1)) { if (cmd == 'm') { x1 += x; y1 += y; } if (repeat) gsk_path_builder_line_to (builder, x1, y1); else { gsk_path_builder_move_to (builder, x1, y1); if (_strchr ("zZX", prev_cmd)) { path_x = x1; path_y = y1; } } x = x1; y = y1; } else goto error; } break; case 'L': case 'l': { double x1, y1; if (parse_coordinate_pair (&p, &x1, &y1)) { if (cmd == 'l') { x1 += x; y1 += y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_line_to (builder, x1, y1); x = x1; y = y1; } else goto error; } break; case 'H': case 'h': { double x1; if (parse_coordinate (&p, &x1)) { if (cmd == 'h') x1 += x; if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_line_to (builder, x1, y); x = x1; } else goto error; } break; case 'V': case 'v': { double y1; if (parse_coordinate (&p, &y1)) { if (cmd == 'v') y1 += y; if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_line_to (builder, x, y1); y = y1; } else goto error; } break; case 'C': case 'c': { double x0, y0, x1, y1, x2, y2; if (parse_coordinate_pair (&p, &x0, &y0) && parse_coordinate_pair (&p, &x1, &y1) && parse_coordinate_pair (&p, &x2, &y2)) { if (cmd == 'c') { x0 += x; y0 += y; x1 += x; y1 += y; x2 += x; y2 += y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_cubic_to (builder, x0, y0, x1, y1, x2, y2); prev_x1 = x1; prev_y1 = y1; x = x2; y = y2; } else goto error; } break; case 'S': case 's': { double x0, y0, x1, y1, x2, y2; if (parse_coordinate_pair (&p, &x1, &y1) && parse_coordinate_pair (&p, &x2, &y2)) { if (cmd == 's') { x1 += x; y1 += y; x2 += x; y2 += y; } if (_strchr ("CcSs", prev_cmd)) { x0 = 2 * x - prev_x1; y0 = 2 * y - prev_y1; } else { x0 = x; y0 = y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_cubic_to (builder, x0, y0, x1, y1, x2, y2); prev_x1 = x1; prev_y1 = y1; x = x2; y = y2; } else goto error; } break; case 'Q': case 'q': { double x1, y1, x2, y2; if (parse_coordinate_pair (&p, &x1, &y1) && parse_coordinate_pair (&p, &x2, &y2)) { if (cmd == 'q') { x1 += x; y1 += y; x2 += x; y2 += y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_quad_to (builder, x1, y1, x2, y2); prev_x1 = x1; prev_y1 = y1; x = x2; y = y2; } else goto error; } break; case 'T': case 't': { double x1, y1, x2, y2; if (parse_coordinate_pair (&p, &x2, &y2)) { if (cmd == 't') { x2 += x; y2 += y; } if (_strchr ("QqTt", prev_cmd)) { x1 = 2 * x - prev_x1; y1 = 2 * y - prev_y1; } else { x1 = x; y1 = y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_quad_to (builder, x1, y1, x2, y2); prev_x1 = x1; prev_y1 = y1; x = x2; y = y2; } else goto error; } break; case 'O': case 'o': { double x1, y1, x2, y2, weight; if (parse_coordinate_pair (&p, &x1, &y1) && parse_coordinate_pair (&p, &x2, &y2) && parse_nonnegative_number (&p, &weight)) { if (cmd == 'o') { x1 += x; y1 += y; x2 += x; y2 += y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_conic_to (builder, x1, y1, x2, y2, weight); x = x2; y = y2; } else goto error; } break; case 'A': case 'a': { double rx, ry; double x_axis_rotation; int large_arc, sweep; double x1, y1; if (parse_nonnegative_number (&p, &rx) && parse_nonnegative_number (&p, &ry) && parse_number (&p, &x_axis_rotation) && parse_flag (&p, &large_arc) && parse_flag (&p, &sweep) && parse_coordinate_pair (&p, &x1, &y1)) { if (cmd == 'a') { x1 += x; y1 += y; } if (_strchr ("zZ", prev_cmd)) { gsk_path_builder_move_to (builder, x, y); path_x = x; path_y = y; } gsk_path_builder_svg_arc_to (builder, rx, ry, x_axis_rotation, large_arc, sweep, x1, y1); x = x1; y = y1; } else goto error; } break; default: goto error; } after_comma = (p > string) && p[-1] == ','; } if (after_comma) goto error; return gsk_path_builder_free_to_path (builder); error: //g_warning ("Can't parse string '%s' as GskPath, error at %ld", string, p - string); gsk_path_builder_unref (builder); return NULL; } /* }}} */ /* vim:set foldmethod=marker expandtab: */