diff --git a/gtk/gtkcssimage.c b/gtk/gtkcssimage.c index 62839d2cf5..616ac74f0a 100644 --- a/gtk/gtkcssimage.c +++ b/gtk/gtkcssimage.c @@ -26,14 +26,15 @@ #include "gtkprivate.h" /* for the types only */ +#include "gtk/gtkcssimageconicprivate.h" #include "gtk/gtkcssimagecrossfadeprivate.h" +#include "gtk/gtkcssimagefallbackprivate.h" #include "gtk/gtkcssimageiconthemeprivate.h" #include "gtk/gtkcssimagelinearprivate.h" #include "gtk/gtkcssimageradialprivate.h" #include "gtk/gtkcssimageurlprivate.h" #include "gtk/gtkcssimagescaledprivate.h" #include "gtk/gtkcssimagerecolorprivate.h" -#include "gtk/gtkcssimagefallbackprivate.h" G_DEFINE_ABSTRACT_TYPE (GtkCssImage, _gtk_css_image, G_TYPE_OBJECT) @@ -521,6 +522,7 @@ gtk_css_image_get_parser_type (GtkCssParser *parser) { "repeating-linear-gradient", _gtk_css_image_linear_get_type }, { "radial-gradient", _gtk_css_image_radial_get_type }, { "repeating-radial-gradient", _gtk_css_image_radial_get_type }, + { "conic-gradient", gtk_css_image_conic_get_type }, { "cross-fade", gtk_css_image_cross_fade_get_type }, { "image", _gtk_css_image_fallback_get_type } }; diff --git a/gtk/gtkcssimageconic.c b/gtk/gtkcssimageconic.c new file mode 100644 index 0000000000..b046c0df4e --- /dev/null +++ b/gtk/gtkcssimageconic.c @@ -0,0 +1,529 @@ +/* + * Copyright © 2012 Red Hat Inc. + * + * 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 "gtkcssimageconicprivate.h" + +#include + +#include "gtkcsscolorvalueprivate.h" +#include "gtkcssnumbervalueprivate.h" +#include "gtkcsspositionvalueprivate.h" +#include "gtkcssprovider.h" + +G_DEFINE_TYPE (GtkCssImageConic, gtk_css_image_conic, GTK_TYPE_CSS_IMAGE) + +static void +gtk_css_image_conic_snapshot (GtkCssImage *image, + GtkSnapshot *snapshot, + double width, + double height) +{ + GtkCssImageConic *self = GTK_CSS_IMAGE_CONIC (image); + GskColorStop *stops; + int i, last; + double offset; + + stops = g_newa (GskColorStop, self->n_stops); + + last = -1; + offset = 0; + for (i = 0; i < self->n_stops; i++) + { + const GtkCssImageConicColorStop *stop = &self->color_stops[i]; + double pos, step; + + if (stop->offset == NULL) + { + if (i == 0) + pos = 0.0; + else if (i + 1 == self->n_stops) + pos = 1.0; + else + continue; + } + else + { + pos = _gtk_css_number_value_get (stop->offset, 360) / 360; + pos = CLAMP (pos, 0.0, 1.0); + } + + pos = MAX (pos, offset); + step = (pos - offset) / (i - last); + for (last = last + 1; last <= i; last++) + { + stop = &self->color_stops[last]; + + offset += step; + + stops[last].offset = offset; + stops[last].color = *gtk_css_color_value_get_rgba (stop->color); + } + + offset = pos; + last = i; + } + + gtk_snapshot_append_conic_gradient ( + snapshot, + &GRAPHENE_RECT_INIT (0, 0, width, height), + &GRAPHENE_POINT_INIT (_gtk_css_position_value_get_x (self->center, width), + _gtk_css_position_value_get_y (self->center, height)), + _gtk_css_number_value_get (self->rotation, 360), + stops, + self->n_stops); +} + +static gboolean +parse_angles (GtkCssParser *parser, + gpointer option_data, + gpointer unused) +{ + GtkCssValue **angles = option_data; + + angles[0] = _gtk_css_number_value_parse (parser, GTK_CSS_PARSE_ANGLE | GTK_CSS_PARSE_PERCENT); + if (angles[0] == NULL) + return FALSE; + + if (gtk_css_number_value_can_parse (parser)) + { + angles[1] = _gtk_css_number_value_parse (parser, GTK_CSS_PARSE_ANGLE | GTK_CSS_PARSE_PERCENT); + if (angles[1] == NULL) + return FALSE; + } + + return TRUE; +} + +static gboolean +parse_color (GtkCssParser *parser, + gpointer option_data, + gpointer unused) +{ + GtkCssValue **color = option_data; + + *color = _gtk_css_color_value_parse (parser); + if (*color == NULL) + return FALSE; + + return TRUE; +} + +static guint +gtk_css_image_conic_parse_color_stop (GtkCssImageConic *self, + GtkCssParser *parser, + GArray *stop_array) +{ + GtkCssValue *angles[2] = { NULL, NULL }; + GtkCssValue *color = NULL; + GtkCssParseOption options[] = + { + { (void *) gtk_css_number_value_can_parse, parse_angles, &angles }, + { (void *) gtk_css_color_value_can_parse, parse_color, &color }, + }; + + if (!gtk_css_parser_consume_any (parser, options, G_N_ELEMENTS (options), NULL)) + goto fail; + + if (color == NULL) + { + gtk_css_parser_error_syntax (parser, "Expected shadow value to contain a length"); + goto fail; + } + + g_array_append_vals (stop_array, (GtkCssImageConicColorStop[1]) { + { angles[0], color } + }, + 1); + if (angles[1]) + g_array_append_vals (stop_array, (GtkCssImageConicColorStop[1]) { + { angles[1], gtk_css_value_ref (color) } + }, + 1); + + return 1; + +fail: + g_clear_pointer (&angles[0], gtk_css_value_unref); + g_clear_pointer (&angles[1], gtk_css_value_unref); + g_clear_pointer (&color, gtk_css_value_unref); + return 0; +} + +static guint +gtk_css_image_conic_parse_first_arg (GtkCssImageConic *self, + GtkCssParser *parser, + GArray *stop_array) +{ + gboolean nothing_parsed = TRUE; + + if (gtk_css_parser_try_ident (parser, "from")) + { + self->rotation = _gtk_css_number_value_parse (parser, GTK_CSS_PARSE_ANGLE); + if (self->rotation == NULL) + return 0; + nothing_parsed = FALSE; + } + else + { + self->rotation = _gtk_css_number_value_new (0, GTK_CSS_DEG); + } + + if (gtk_css_parser_try_ident (parser, "at")) + { + self->center = _gtk_css_position_value_parse (parser); + if (self->center == NULL) + return 0; + nothing_parsed = FALSE; + } + else + { + self->center = _gtk_css_position_value_new (_gtk_css_number_value_new (50, GTK_CSS_PERCENT), + _gtk_css_number_value_new (50, GTK_CSS_PERCENT)); + } + + if (!nothing_parsed) + return 1; + + return 1 + gtk_css_image_conic_parse_color_stop (self, parser, stop_array); +} + +typedef struct +{ + GtkCssImageConic *self; + GArray *stop_array; +} ParseData; + +static guint +gtk_css_image_conic_parse_arg (GtkCssParser *parser, + guint arg, + gpointer user_data) +{ + ParseData *parse_data = user_data; + GtkCssImageConic *self = parse_data->self; + + if (arg == 0) + return gtk_css_image_conic_parse_first_arg (self, parser, parse_data->stop_array); + else + return gtk_css_image_conic_parse_color_stop (self, parser, parse_data->stop_array); +} + +static gboolean +gtk_css_image_conic_parse (GtkCssImage *image, + GtkCssParser *parser) +{ + GtkCssImageConic *self = GTK_CSS_IMAGE_CONIC (image); + ParseData parse_data; + gboolean success; + + if (!gtk_css_parser_has_function (parser, "conic-gradient")) + { + gtk_css_parser_error_syntax (parser, "Not a conic gradient"); + return FALSE; + } + + parse_data.self = self; + parse_data.stop_array = g_array_new (TRUE, FALSE, sizeof (GtkCssImageConicColorStop)); + + success = gtk_css_parser_consume_function (parser, 3, G_MAXUINT, gtk_css_image_conic_parse_arg, &parse_data); + + if (!success) + { + g_array_free (parse_data.stop_array, TRUE); + } + else + { + self->n_stops = parse_data.stop_array->len; + self->color_stops = (GtkCssImageConicColorStop *)g_array_free (parse_data.stop_array, FALSE); + } + + return success; +} + +static void +gtk_css_image_conic_print (GtkCssImage *image, + GString *string) +{ + GtkCssImageConic *self = GTK_CSS_IMAGE_CONIC (image); + gboolean written = FALSE; + guint i; + + g_string_append (string, "self-gradient("); + + if (self->center) + { + GtkCssValue *compare = _gtk_css_position_value_new (_gtk_css_number_value_new (50, GTK_CSS_PERCENT), + _gtk_css_number_value_new (50, GTK_CSS_PERCENT)); + + if (!_gtk_css_value_equal (self->center, compare)) + { + g_string_append (string, "from "); + _gtk_css_value_print (self->center, string); + written = TRUE; + } + + gtk_css_value_unref (compare); + } + + if (self->rotation && _gtk_css_number_value_get (self->rotation, 360) != 0) + { + if (written) + g_string_append_c (string, ' '); + g_string_append (string, "at "); + _gtk_css_value_print (self->rotation, string); + } + + if (written) + g_string_append (string, ", "); + + for (i = 0; i < self->n_stops; i++) + { + const GtkCssImageConicColorStop *stop = &self->color_stops[i]; + + if (i > 0) + g_string_append (string, ", "); + + _gtk_css_value_print (stop->color, string); + + if (stop->offset) + { + g_string_append (string, " "); + _gtk_css_value_print (stop->offset, string); + } + } + + g_string_append (string, ")"); +} + +static GtkCssImage * +gtk_css_image_conic_compute (GtkCssImage *image, + guint property_id, + GtkStyleProvider *provider, + GtkCssStyle *style, + GtkCssStyle *parent_style) +{ + GtkCssImageConic *self = GTK_CSS_IMAGE_CONIC (image); + GtkCssImageConic *copy; + guint i; + + copy = g_object_new (GTK_TYPE_CSS_IMAGE_CONIC, NULL); + + copy->center = _gtk_css_value_compute (self->center, property_id, provider, style, parent_style); + copy->rotation = _gtk_css_value_compute (self->rotation, property_id, provider, style, parent_style); + + copy->n_stops = self->n_stops; + copy->color_stops = g_malloc (sizeof (GtkCssImageConicColorStop) * copy->n_stops); + for (i = 0; i < self->n_stops; i++) + { + const GtkCssImageConicColorStop *stop = &self->color_stops[i]; + GtkCssImageConicColorStop *scopy = ©->color_stops[i]; + + scopy->color = _gtk_css_value_compute (stop->color, property_id, provider, style, parent_style); + + if (stop->offset) + { + scopy->offset = _gtk_css_value_compute (stop->offset, property_id, provider, style, parent_style); + } + else + { + scopy->offset = NULL; + } + } + + return GTK_CSS_IMAGE (copy); +} + +static GtkCssImage * +gtk_css_image_conic_transition (GtkCssImage *start_image, + GtkCssImage *end_image, + guint property_id, + double progress) +{ + GtkCssImageConic *start, *end, *result; + guint i; + + start = GTK_CSS_IMAGE_CONIC (start_image); + + if (end_image == NULL) + return GTK_CSS_IMAGE_CLASS (gtk_css_image_conic_parent_class)->transition (start_image, end_image, property_id, progress); + + if (!GTK_IS_CSS_IMAGE_CONIC (end_image)) + return GTK_CSS_IMAGE_CLASS (gtk_css_image_conic_parent_class)->transition (start_image, end_image, property_id, progress); + + end = GTK_CSS_IMAGE_CONIC (end_image); + + if (start->n_stops != end->n_stops) + return GTK_CSS_IMAGE_CLASS (gtk_css_image_conic_parent_class)->transition (start_image, end_image, property_id, progress); + + result = g_object_new (GTK_TYPE_CSS_IMAGE_CONIC, NULL); + + result->center = _gtk_css_value_transition (start->center, end->center, property_id, progress); + if (result->center == NULL) + goto fail; + + result->rotation = _gtk_css_value_transition (start->rotation, end->rotation, property_id, progress); + if (result->rotation == NULL) + goto fail; + + result->color_stops = g_malloc (sizeof (GtkCssImageConicColorStop) * start->n_stops); + result->n_stops = 0; + for (i = 0; i < start->n_stops; i++) + { + const GtkCssImageConicColorStop *start_stop = &start->color_stops[i]; + const GtkCssImageConicColorStop *end_stop = &end->color_stops[i]; + GtkCssImageConicColorStop *stop = &result->color_stops[i]; + + if ((start_stop->offset != NULL) != (end_stop->offset != NULL)) + goto fail; + + if (start_stop->offset == NULL) + { + stop->offset = NULL; + } + else + { + stop->offset = _gtk_css_value_transition (start_stop->offset, + end_stop->offset, + property_id, + progress); + if (stop->offset == NULL) + goto fail; + } + + stop->color = _gtk_css_value_transition (start_stop->color, + end_stop->color, + property_id, + progress); + if (stop->color == NULL) + { + if (stop->offset) + _gtk_css_value_unref (stop->offset); + goto fail; + } + + result->n_stops ++; + } + + return GTK_CSS_IMAGE (result); + +fail: + g_object_unref (result); + return GTK_CSS_IMAGE_CLASS (gtk_css_image_conic_parent_class)->transition (start_image, end_image, property_id, progress); +} + +static gboolean +gtk_css_image_conic_equal (GtkCssImage *image1, + GtkCssImage *image2) +{ + GtkCssImageConic *conic1 = (GtkCssImageConic *) image1; + GtkCssImageConic *conic2 = (GtkCssImageConic *) image2; + guint i; + + if (!_gtk_css_value_equal (conic1->center, conic2->center) || + !_gtk_css_value_equal (conic1->rotation, conic2->rotation)) + return FALSE; + + for (i = 0; i < conic1->n_stops; i++) + { + const GtkCssImageConicColorStop *stop1 = &conic1->color_stops[i]; + const GtkCssImageConicColorStop *stop2 = &conic2->color_stops[i]; + + if (!_gtk_css_value_equal0 (stop1->offset, stop2->offset) || + !_gtk_css_value_equal (stop1->color, stop2->color)) + return FALSE; + } + + return TRUE; +} + +static void +gtk_css_image_conic_dispose (GObject *object) +{ + GtkCssImageConic *self = GTK_CSS_IMAGE_CONIC (object); + guint i; + + for (i = 0; i < self->n_stops; i ++) + { + GtkCssImageConicColorStop *stop = &self->color_stops[i]; + + _gtk_css_value_unref (stop->color); + if (stop->offset) + _gtk_css_value_unref (stop->offset); + } + g_free (self->color_stops); + + g_clear_pointer (&self->center, gtk_css_value_unref); + g_clear_pointer (&self->rotation, gtk_css_value_unref); + + G_OBJECT_CLASS (gtk_css_image_conic_parent_class)->dispose (object); +} + +static gboolean +gtk_css_image_conic_is_computed (GtkCssImage *image) +{ + GtkCssImageConic *self = GTK_CSS_IMAGE_CONIC (image); + guint i; + gboolean computed = TRUE; + + computed = !self->center || gtk_css_value_is_computed (self->center); + computed &= !self->rotation || gtk_css_value_is_computed (self->rotation); + + for (i = 0; i < self->n_stops; i ++) + { + const GtkCssImageConicColorStop *stop = &self->color_stops[i]; + + if (stop->offset && !gtk_css_value_is_computed (stop->offset)) + { + computed = FALSE; + break; + } + + if (!gtk_css_value_is_computed (stop->color)) + { + computed = FALSE; + break; + } + } + + return computed; +} + +static void +gtk_css_image_conic_class_init (GtkCssImageConicClass *klass) +{ + GtkCssImageClass *image_class = GTK_CSS_IMAGE_CLASS (klass); + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + image_class->snapshot = gtk_css_image_conic_snapshot; + image_class->parse = gtk_css_image_conic_parse; + image_class->print = gtk_css_image_conic_print; + image_class->compute = gtk_css_image_conic_compute; + image_class->equal = gtk_css_image_conic_equal; + image_class->transition = gtk_css_image_conic_transition; + image_class->is_computed = gtk_css_image_conic_is_computed; + + object_class->dispose = gtk_css_image_conic_dispose; +} + +static void +gtk_css_image_conic_init (GtkCssImageConic *self) +{ +} + diff --git a/gtk/gtkcssimageconicprivate.h b/gtk/gtkcssimageconicprivate.h new file mode 100644 index 0000000000..89d9c26739 --- /dev/null +++ b/gtk/gtkcssimageconicprivate.h @@ -0,0 +1,64 @@ +/* + * Copyright © 2012 Red Hat Inc. + * + * 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 + */ + +#ifndef __GTK_CSS_IMAGE_CONIC_PRIVATE_H__ +#define __GTK_CSS_IMAGE_CONIC_PRIVATE_H__ + +#include "gtk/gtkcssimageprivate.h" +#include "gtk/gtkcssvalueprivate.h" + +G_BEGIN_DECLS + +#define GTK_TYPE_CSS_IMAGE_CONIC (gtk_css_image_conic_get_type ()) +#define GTK_CSS_IMAGE_CONIC(obj) (G_TYPE_CHECK_INSTANCE_CAST (obj, GTK_TYPE_CSS_IMAGE_CONIC, GtkCssImageConic)) +#define GTK_CSS_IMAGE_CONIC_CLASS(cls) (G_TYPE_CHECK_CLASS_CAST (cls, GTK_TYPE_CSS_IMAGE_CONIC, GtkCssImageConicClass)) +#define GTK_IS_CSS_IMAGE_CONIC(obj) (G_TYPE_CHECK_INSTANCE_TYPE (obj, GTK_TYPE_CSS_IMAGE_CONIC)) +#define GTK_IS_CSS_IMAGE_CONIC_CLASS(obj) (G_TYPE_CHECK_CLASS_TYPE (obj, GTK_TYPE_CSS_IMAGE_CONIC)) +#define GTK_CSS_IMAGE_CONIC_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_CSS_IMAGE_CONIC, GtkCssImageConicClass)) + +typedef struct _GtkCssImageConic GtkCssImageConic; +typedef struct _GtkCssImageConicClass GtkCssImageConicClass; +typedef struct _GtkCssImageConicColorStop GtkCssImageConicColorStop; + +struct _GtkCssImageConicColorStop { + GtkCssValue *offset; + GtkCssValue *color; +}; + +struct _GtkCssImageConic +{ + GtkCssImage parent; + + GtkCssValue *center; + GtkCssValue *rotation; + + guint n_stops; + GtkCssImageConicColorStop *color_stops; +}; + +struct _GtkCssImageConicClass +{ + GtkCssImageClass parent_class; +}; + +GType gtk_css_image_conic_get_type (void) G_GNUC_CONST; + +G_END_DECLS + +#endif /* __GTK_CSS_IMAGE_CONIC_PRIVATE_H__ */ diff --git a/gtk/meson.build b/gtk/meson.build index d7f206c231..00f8a500df 100644 --- a/gtk/meson.build +++ b/gtk/meson.build @@ -63,6 +63,7 @@ gtk_private_sources = files([ 'gtkcssfontfeaturesvalue.c', 'gtkcssfontvariationsvalue.c', 'gtkcssimage.c', + 'gtkcssimageconic.c', 'gtkcssimagecrossfade.c', 'gtkcssimagefallback.c', 'gtkcssimageicontheme.c',