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',