GTK <= 3.20 allowed that but the CSS spec and borwsers do not. We might want to just remove that feature, but for now, we just warn.
		
			
				
	
	
		
			698 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			698 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright © 2015 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 <http://www.gnu.org/licenses/>.
 | 
						|
 *
 | 
						|
 * Authors: Matthias Clasen <mclasen@redhat.com>
 | 
						|
 */
 | 
						|
 | 
						|
#include "config.h"
 | 
						|
 | 
						|
#include "gtkcssimageradialprivate.h"
 | 
						|
 | 
						|
#include <math.h>
 | 
						|
 | 
						|
#include "gtkcsscolorvalueprivate.h"
 | 
						|
#include "gtkcssnumbervalueprivate.h"
 | 
						|
#include "gtkcsspositionvalueprivate.h"
 | 
						|
#include "gtkcssrgbavalueprivate.h"
 | 
						|
#include "gtkcssprovider.h"
 | 
						|
 | 
						|
G_DEFINE_TYPE (GtkCssImageRadial, _gtk_css_image_radial, GTK_TYPE_CSS_IMAGE)
 | 
						|
 | 
						|
static void
 | 
						|
gtk_css_image_radial_get_start_end (GtkCssImageRadial *radial,
 | 
						|
                                    double             radius,
 | 
						|
                                    double            *start,
 | 
						|
                                    double            *end)
 | 
						|
{
 | 
						|
  GtkCssImageRadialColorStop *stop;
 | 
						|
  double pos;
 | 
						|
  guint i;
 | 
						|
 | 
						|
  if (radial->repeating)
 | 
						|
    {
 | 
						|
      stop = &g_array_index (radial->stops, GtkCssImageRadialColorStop, 0);
 | 
						|
      if (stop->offset == NULL)
 | 
						|
        *start = 0;
 | 
						|
      else
 | 
						|
        *start = _gtk_css_number_value_get (stop->offset, radius) / radius;
 | 
						|
 | 
						|
      *end = *start;
 | 
						|
 | 
						|
      for (i = 0; i < radial->stops->len; i++)
 | 
						|
        {
 | 
						|
          stop = &g_array_index (radial->stops, GtkCssImageRadialColorStop, i);
 | 
						|
 | 
						|
          if (stop->offset == NULL)
 | 
						|
            continue;
 | 
						|
 | 
						|
          pos = _gtk_css_number_value_get (stop->offset, radius) / radius;
 | 
						|
 | 
						|
          *end = MAX (pos, *end);
 | 
						|
        }
 | 
						|
 | 
						|
      if (stop->offset == NULL)
 | 
						|
        *end = MAX (*end, 1.0);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      *start = 0;
 | 
						|
      *end = 1;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_css_image_radial_draw (GtkCssImage *image,
 | 
						|
                           cairo_t     *cr,
 | 
						|
                           double       width,
 | 
						|
                           double       height)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *radial = GTK_CSS_IMAGE_RADIAL (image);
 | 
						|
  cairo_pattern_t *pattern;
 | 
						|
  cairo_matrix_t matrix;
 | 
						|
  double x, y;
 | 
						|
  double radius, yscale;
 | 
						|
  double start, end;
 | 
						|
  double r1, r2, r3, r4, r;
 | 
						|
  double offset;
 | 
						|
  int i, last;
 | 
						|
 | 
						|
  x = _gtk_css_position_value_get_x (radial->position, width);
 | 
						|
  y = _gtk_css_position_value_get_y (radial->position, height);
 | 
						|
 | 
						|
  if (radial->circle)
 | 
						|
    {
 | 
						|
      switch (radial->size)
 | 
						|
        {
 | 
						|
        case GTK_CSS_EXPLICIT_SIZE:
 | 
						|
          radius = _gtk_css_number_value_get (radial->sizes[0], width);
 | 
						|
          break;
 | 
						|
        case GTK_CSS_CLOSEST_SIDE:
 | 
						|
          radius = MIN (MIN (x, width - x), MIN (y, height - y));
 | 
						|
          break;
 | 
						|
        case GTK_CSS_FARTHEST_SIDE:
 | 
						|
          radius = MAX (MAX (x, width - x), MAX (y, height - y));
 | 
						|
          break;
 | 
						|
        case GTK_CSS_CLOSEST_CORNER:
 | 
						|
        case GTK_CSS_FARTHEST_CORNER:
 | 
						|
          r1 = x*x + y*y;
 | 
						|
          r2 = x*x + (height - y)*(height - y);
 | 
						|
          r3 = (width - x)*(width - x) + y*y;
 | 
						|
          r4 = (width - x)*(width - x) + (height - y)*(height - y);
 | 
						|
          if (radial->size == GTK_CSS_CLOSEST_CORNER)
 | 
						|
            r = MIN ( MIN (r1, r2), MIN (r3, r4));
 | 
						|
          else
 | 
						|
            r = MAX ( MAX (r1, r2), MAX (r3, r4));
 | 
						|
          radius = sqrt (r);
 | 
						|
          break;
 | 
						|
        default:
 | 
						|
          g_assert_not_reached ();
 | 
						|
        }
 | 
						|
 | 
						|
      radius = MAX (1.0, radius);
 | 
						|
      yscale = 1.0;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      double hradius, vradius;
 | 
						|
 | 
						|
      switch (radial->size)
 | 
						|
        {
 | 
						|
        case GTK_CSS_EXPLICIT_SIZE:
 | 
						|
          hradius = _gtk_css_number_value_get (radial->sizes[0], width);
 | 
						|
          vradius = _gtk_css_number_value_get (radial->sizes[1], height);
 | 
						|
          break;
 | 
						|
        case GTK_CSS_CLOSEST_SIDE:
 | 
						|
          hradius = MIN (x, width - x);
 | 
						|
          vradius = MIN (y, height - y);
 | 
						|
          break;
 | 
						|
        case GTK_CSS_FARTHEST_SIDE:
 | 
						|
          hradius = MAX (x, width - x);
 | 
						|
          vradius = MAX (y, height - y);
 | 
						|
          break;
 | 
						|
        case GTK_CSS_CLOSEST_CORNER:
 | 
						|
          hradius = M_SQRT2 * MIN (x, width - x);
 | 
						|
          vradius = M_SQRT2 * MIN (y, height - y);
 | 
						|
          break;
 | 
						|
        case GTK_CSS_FARTHEST_CORNER:
 | 
						|
          hradius = M_SQRT2 * MAX (x, width - x);
 | 
						|
          vradius = M_SQRT2 * MAX (y, height - y);
 | 
						|
          break;
 | 
						|
        default:
 | 
						|
          g_assert_not_reached ();
 | 
						|
        }
 | 
						|
 | 
						|
      hradius = MAX (1.0, hradius);
 | 
						|
      vradius = MAX (1.0, vradius);
 | 
						|
 | 
						|
      radius = hradius;
 | 
						|
      yscale = vradius / hradius;
 | 
						|
    }
 | 
						|
 | 
						|
  gtk_css_image_radial_get_start_end (radial, radius, &start, &end);
 | 
						|
 | 
						|
  pattern = cairo_pattern_create_radial (0, 0, 0, 0, 0, radius);
 | 
						|
  if (yscale != 1.0)
 | 
						|
    {
 | 
						|
      cairo_matrix_init_scale (&matrix, 1.0, 1.0 / yscale);
 | 
						|
      cairo_pattern_set_matrix (pattern, &matrix);
 | 
						|
    }
 | 
						|
 | 
						|
 if (radial->repeating)
 | 
						|
    cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
 | 
						|
  else
 | 
						|
    cairo_pattern_set_extend (pattern, CAIRO_EXTEND_PAD);
 | 
						|
 | 
						|
  offset = start;
 | 
						|
  last = -1;
 | 
						|
  for (i = 0; i < radial->stops->len; i++)
 | 
						|
    {
 | 
						|
      GtkCssImageRadialColorStop *stop;
 | 
						|
      double pos, step;
 | 
						|
 | 
						|
      stop = &g_array_index (radial->stops, GtkCssImageRadialColorStop, i);
 | 
						|
 | 
						|
      if (stop->offset == NULL)
 | 
						|
        {
 | 
						|
          if (i == 0)
 | 
						|
            pos = 0.0;
 | 
						|
          else if (i + 1 == radial->stops->len)
 | 
						|
            pos = 1.0;
 | 
						|
          else
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
      else
 | 
						|
        pos = _gtk_css_number_value_get (stop->offset, radius) / radius;
 | 
						|
 | 
						|
      pos = MAX (pos, 0);
 | 
						|
      step = pos / (i - last);
 | 
						|
      for (last = last + 1; last <= i; last++)
 | 
						|
        {
 | 
						|
          const GdkRGBA *rgba;
 | 
						|
 | 
						|
          stop = &g_array_index (radial->stops, GtkCssImageRadialColorStop, last);
 | 
						|
 | 
						|
          rgba = _gtk_css_rgba_value_get_rgba (stop->color);
 | 
						|
          offset += step;
 | 
						|
 | 
						|
          cairo_pattern_add_color_stop_rgba (pattern,
 | 
						|
                                             (offset - start) / (end - start),
 | 
						|
                                             rgba->red,
 | 
						|
                                             rgba->green,
 | 
						|
                                             rgba->blue,
 | 
						|
                                             rgba->alpha);
 | 
						|
        }
 | 
						|
 | 
						|
      offset = pos;
 | 
						|
      last = i;
 | 
						|
    }
 | 
						|
 | 
						|
  cairo_rectangle (cr, 0, 0, width, height);
 | 
						|
  cairo_translate (cr, x, y);
 | 
						|
  cairo_set_source (cr, pattern);
 | 
						|
  cairo_fill (cr);
 | 
						|
 | 
						|
  cairo_pattern_destroy (pattern);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_css_image_radial_parse (GtkCssImage  *image,
 | 
						|
                            GtkCssParser *parser)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *radial = GTK_CSS_IMAGE_RADIAL (image);
 | 
						|
  gboolean has_shape = FALSE;
 | 
						|
  gboolean has_size = FALSE;
 | 
						|
  gboolean found_one = FALSE;
 | 
						|
  guint i;
 | 
						|
  static struct {
 | 
						|
    const char *name;
 | 
						|
    guint       value;
 | 
						|
  } names[] = {
 | 
						|
    { "closest-side", GTK_CSS_CLOSEST_SIDE },
 | 
						|
    { "farthest-side", GTK_CSS_FARTHEST_SIDE },
 | 
						|
    { "closest-corner", GTK_CSS_CLOSEST_CORNER },
 | 
						|
    { "farthest-corner", GTK_CSS_FARTHEST_CORNER }
 | 
						|
  };
 | 
						|
 | 
						|
  if (_gtk_css_parser_try (parser, "repeating-radial-gradient(", TRUE))
 | 
						|
    radial->repeating = TRUE;
 | 
						|
  else if (_gtk_css_parser_try (parser, "radial-gradient(", TRUE))
 | 
						|
    radial->repeating = FALSE;
 | 
						|
  else
 | 
						|
    {
 | 
						|
      _gtk_css_parser_error (parser, "Not a radial gradient");
 | 
						|
      return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
  do {
 | 
						|
    found_one = FALSE;
 | 
						|
    if (!has_shape && _gtk_css_parser_try (parser, "circle", TRUE))
 | 
						|
      {
 | 
						|
        radial->circle = TRUE;
 | 
						|
        found_one = has_shape = TRUE;
 | 
						|
      }
 | 
						|
    else if (!has_shape && _gtk_css_parser_try (parser, "ellipse", TRUE))
 | 
						|
      {
 | 
						|
        radial->circle = FALSE;
 | 
						|
        found_one = has_shape = TRUE;
 | 
						|
      }
 | 
						|
    else if (!has_size)
 | 
						|
      {
 | 
						|
        for (i = 0; i < G_N_ELEMENTS (names); i++)
 | 
						|
          {
 | 
						|
            if (_gtk_css_parser_try (parser, names[i].name, TRUE))
 | 
						|
              {
 | 
						|
                found_one = has_size = TRUE;
 | 
						|
                radial->size = names[i].value;
 | 
						|
                break;
 | 
						|
              }
 | 
						|
          }
 | 
						|
 | 
						|
        if (!has_size)
 | 
						|
          {
 | 
						|
            if (gtk_css_number_value_can_parse (parser))
 | 
						|
              radial->sizes[0] = _gtk_css_number_value_parse (parser, GTK_CSS_PARSE_LENGTH | GTK_CSS_PARSE_PERCENT);
 | 
						|
            if (gtk_css_number_value_can_parse (parser))
 | 
						|
              radial->sizes[1] = _gtk_css_number_value_parse (parser, GTK_CSS_PARSE_LENGTH | GTK_CSS_PARSE_PERCENT);
 | 
						|
            found_one = has_size = radial->sizes[0] != NULL;
 | 
						|
          }
 | 
						|
      }
 | 
						|
 | 
						|
  } while (found_one && !(has_shape && has_size));
 | 
						|
 | 
						|
  if (_gtk_css_parser_try (parser, "at", TRUE))
 | 
						|
    {
 | 
						|
      radial->position = _gtk_css_position_value_parse (parser);
 | 
						|
      if (!radial->position)
 | 
						|
        return FALSE;
 | 
						|
      if (!_gtk_css_parser_try (parser, ",", TRUE))
 | 
						|
        {
 | 
						|
          _gtk_css_parser_error (parser, "Expected a comma here");
 | 
						|
          return FALSE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      radial->position = _gtk_css_position_value_new (_gtk_css_number_value_new (50, GTK_CSS_PERCENT),
 | 
						|
                                                      _gtk_css_number_value_new (50, GTK_CSS_PERCENT));
 | 
						|
 | 
						|
      if ((has_shape || has_size) &&
 | 
						|
          !_gtk_css_parser_try (parser, ",", TRUE))
 | 
						|
        {
 | 
						|
          _gtk_css_parser_error (parser, "Expected a comma here");
 | 
						|
          return FALSE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
  if (!has_size)
 | 
						|
    {
 | 
						|
      radial->size = GTK_CSS_FARTHEST_CORNER;
 | 
						|
    }
 | 
						|
 | 
						|
  if (!has_shape)
 | 
						|
    {
 | 
						|
      if (radial->sizes[0] && radial->sizes[1])
 | 
						|
        radial->circle = FALSE;
 | 
						|
      else
 | 
						|
        radial->circle = TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
  if (has_shape && radial->circle)
 | 
						|
    {
 | 
						|
      if (radial->sizes[0] && radial->sizes[1])
 | 
						|
        {
 | 
						|
          _gtk_css_parser_error (parser, "Circular gradient can only have one size");
 | 
						|
          return FALSE;
 | 
						|
        }
 | 
						|
 | 
						|
      if (radial->sizes[0] && gtk_css_number_value_has_percent (radial->sizes[0]))
 | 
						|
        {
 | 
						|
          _gtk_css_parser_error (parser, "Circular gradient cannot have percentage as size");
 | 
						|
          return FALSE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
  if (has_size && !radial->circle)
 | 
						|
    {
 | 
						|
      if (!radial->sizes[1])
 | 
						|
        radial->sizes[1] = _gtk_css_value_ref (radial->sizes[0]);
 | 
						|
    }
 | 
						|
 | 
						|
  do {
 | 
						|
    GtkCssImageRadialColorStop stop;
 | 
						|
 | 
						|
    stop.color = _gtk_css_color_value_parse (parser);
 | 
						|
    if (stop.color == NULL)
 | 
						|
      return FALSE;
 | 
						|
 | 
						|
    if (gtk_css_number_value_can_parse (parser))
 | 
						|
      {
 | 
						|
        stop.offset = _gtk_css_number_value_parse (parser,
 | 
						|
                                                   GTK_CSS_PARSE_PERCENT
 | 
						|
                                                   | GTK_CSS_PARSE_LENGTH);
 | 
						|
        if (stop.offset == NULL)
 | 
						|
          {
 | 
						|
            _gtk_css_value_unref (stop.color);
 | 
						|
            return FALSE;
 | 
						|
          }
 | 
						|
      }
 | 
						|
    else
 | 
						|
      {
 | 
						|
        stop.offset = NULL;
 | 
						|
      }
 | 
						|
 | 
						|
    g_array_append_val (radial->stops, stop);
 | 
						|
 | 
						|
  } while (_gtk_css_parser_try (parser, ",", TRUE));
 | 
						|
 | 
						|
  if (radial->stops->len < 2)
 | 
						|
    {
 | 
						|
      _gtk_css_parser_error_full (parser,
 | 
						|
                                  GTK_CSS_PROVIDER_ERROR_DEPRECATED,
 | 
						|
                                  "Using one color stop with %s() is deprecated.",
 | 
						|
                                  radial->repeating ? "repeating-radial-gradient" : "radial-gradient");
 | 
						|
    }
 | 
						|
 | 
						|
  if (!_gtk_css_parser_try (parser, ")", TRUE))
 | 
						|
    {
 | 
						|
      _gtk_css_parser_error (parser, "Missing closing bracket at end of radial gradient");
 | 
						|
      return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_css_image_radial_print (GtkCssImage *image,
 | 
						|
                            GString     *string)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *radial = GTK_CSS_IMAGE_RADIAL (image);
 | 
						|
  guint i;
 | 
						|
  const gchar *names[] = {
 | 
						|
    NULL,
 | 
						|
    "closest-side",
 | 
						|
    "farthest-side",
 | 
						|
    "closest-corner",
 | 
						|
    "farthest-corner"
 | 
						|
  };
 | 
						|
 | 
						|
  if (radial->repeating)
 | 
						|
    g_string_append (string, "repeating-radial-gradient(");
 | 
						|
  else
 | 
						|
    g_string_append (string, "radial-gradient(");
 | 
						|
 | 
						|
  if (radial->circle)
 | 
						|
    g_string_append (string, "circle ");
 | 
						|
  else
 | 
						|
    g_string_append (string, "ellipse ");
 | 
						|
 | 
						|
  if (radial->size != 0)
 | 
						|
    g_string_append (string, names[radial->size]);
 | 
						|
  else
 | 
						|
    {
 | 
						|
      if (radial->sizes[0])
 | 
						|
        _gtk_css_value_print (radial->sizes[0], string);
 | 
						|
      if (radial->sizes[1])
 | 
						|
        {
 | 
						|
          g_string_append (string, " ");
 | 
						|
          _gtk_css_value_print (radial->sizes[1], string);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
  g_string_append (string, " at ");
 | 
						|
  _gtk_css_value_print (radial->position, string);
 | 
						|
 | 
						|
  g_string_append (string, ", ");
 | 
						|
 | 
						|
  for (i = 0; i < radial->stops->len; i++)
 | 
						|
    {
 | 
						|
      GtkCssImageRadialColorStop *stop;
 | 
						|
 | 
						|
      if (i > 0)
 | 
						|
        g_string_append (string, ", ");
 | 
						|
 | 
						|
      stop = &g_array_index (radial->stops, GtkCssImageRadialColorStop, i);
 | 
						|
 | 
						|
      _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_radial_compute (GtkCssImage             *image,
 | 
						|
                              guint                    property_id,
 | 
						|
                              GtkStyleProviderPrivate *provider,
 | 
						|
                              GtkCssStyle             *style,
 | 
						|
                              GtkCssStyle             *parent_style)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *radial = GTK_CSS_IMAGE_RADIAL (image);
 | 
						|
  GtkCssImageRadial *copy;
 | 
						|
  guint i;
 | 
						|
 | 
						|
  copy = g_object_new (GTK_TYPE_CSS_IMAGE_RADIAL, NULL);
 | 
						|
  copy->repeating = radial->repeating;
 | 
						|
  copy->circle = radial->circle;
 | 
						|
  copy->size = radial->size;
 | 
						|
 | 
						|
  copy->position = _gtk_css_value_compute (radial->position, property_id, provider, style, parent_style);
 | 
						|
 | 
						|
  if (radial->sizes[0])
 | 
						|
    copy->sizes[0] = _gtk_css_value_compute (radial->sizes[0], property_id, provider, style, parent_style);
 | 
						|
 | 
						|
  if (radial->sizes[1])
 | 
						|
    copy->sizes[1] = _gtk_css_value_compute (radial->sizes[1], property_id, provider, style, parent_style);
 | 
						|
 | 
						|
  g_array_set_size (copy->stops, radial->stops->len);
 | 
						|
  for (i = 0; i < radial->stops->len; i++)
 | 
						|
    {
 | 
						|
      GtkCssImageRadialColorStop *stop, *scopy;
 | 
						|
 | 
						|
      stop = &g_array_index (radial->stops, GtkCssImageRadialColorStop, i);
 | 
						|
      scopy = &g_array_index (copy->stops, GtkCssImageRadialColorStop, 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_radial_transition (GtkCssImage *start_image,
 | 
						|
                                 GtkCssImage *end_image,
 | 
						|
                                 guint        property_id,
 | 
						|
                                 double       progress)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *start, *end, *result;
 | 
						|
  guint i;
 | 
						|
 | 
						|
  start = GTK_CSS_IMAGE_RADIAL (start_image);
 | 
						|
 | 
						|
  if (end_image == NULL)
 | 
						|
    return GTK_CSS_IMAGE_CLASS (_gtk_css_image_radial_parent_class)->transition (start_image, end_image, property_id, progress);
 | 
						|
 | 
						|
  if (!GTK_IS_CSS_IMAGE_RADIAL (end_image))
 | 
						|
    return GTK_CSS_IMAGE_CLASS (_gtk_css_image_radial_parent_class)->transition (start_image, end_image, property_id, progress);
 | 
						|
 | 
						|
  end = GTK_CSS_IMAGE_RADIAL (end_image);
 | 
						|
 | 
						|
  if (start->repeating != end->repeating ||
 | 
						|
      start->stops->len != end->stops->len ||
 | 
						|
      start->size != end->size ||
 | 
						|
      start->circle != end->circle)
 | 
						|
    return GTK_CSS_IMAGE_CLASS (_gtk_css_image_radial_parent_class)->transition (start_image, end_image, property_id, progress);
 | 
						|
 | 
						|
  result = g_object_new (GTK_TYPE_CSS_IMAGE_RADIAL, NULL);
 | 
						|
  result->repeating = start->repeating;
 | 
						|
  result->circle = start->circle;
 | 
						|
  result->size = start->size;
 | 
						|
 | 
						|
  result->position = _gtk_css_value_transition (start->position, end->position, property_id, progress);
 | 
						|
  if (result->position == NULL)
 | 
						|
    goto fail;
 | 
						|
 | 
						|
  if (start->sizes[0] && end->sizes[0])
 | 
						|
    {
 | 
						|
      result->sizes[0] = _gtk_css_value_transition (start->sizes[0], end->sizes[0], property_id, progress);
 | 
						|
      if (result->sizes[0] == NULL)
 | 
						|
        goto fail;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    result->sizes[0] = 0;
 | 
						|
 | 
						|
  if (start->sizes[1] && end->sizes[1])
 | 
						|
    {
 | 
						|
      result->sizes[1] = _gtk_css_value_transition (start->sizes[1], end->sizes[1], property_id, progress);
 | 
						|
      if (result->sizes[1] == NULL)
 | 
						|
        goto fail;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    result->sizes[1] = 0;
 | 
						|
 | 
						|
  for (i = 0; i < start->stops->len; i++)
 | 
						|
    {
 | 
						|
      GtkCssImageRadialColorStop stop, *start_stop, *end_stop;
 | 
						|
 | 
						|
      start_stop = &g_array_index (start->stops, GtkCssImageRadialColorStop, i);
 | 
						|
      end_stop = &g_array_index (end->stops, GtkCssImageRadialColorStop, 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;
 | 
						|
        }
 | 
						|
 | 
						|
      g_array_append_val (result->stops, stop);
 | 
						|
    }
 | 
						|
 | 
						|
  return GTK_CSS_IMAGE (result);
 | 
						|
 | 
						|
fail:
 | 
						|
  g_object_unref (result);
 | 
						|
  return GTK_CSS_IMAGE_CLASS (_gtk_css_image_radial_parent_class)->transition (start_image, end_image, property_id, progress);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_css_image_radial_equal (GtkCssImage *image1,
 | 
						|
                            GtkCssImage *image2)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *radial1 = GTK_CSS_IMAGE_RADIAL (image1);
 | 
						|
  GtkCssImageRadial *radial2 = GTK_CSS_IMAGE_RADIAL (image2);
 | 
						|
  guint i;
 | 
						|
 | 
						|
  if (radial1->repeating != radial2->repeating ||
 | 
						|
      radial1->size != radial2->size ||
 | 
						|
      !_gtk_css_value_equal (radial1->position, radial2->position) ||
 | 
						|
      ((radial1->sizes[0] == NULL) != (radial2->sizes[0] == NULL)) ||
 | 
						|
      (radial1->sizes[0] && radial2->sizes[0] && !_gtk_css_value_equal (radial1->sizes[0], radial2->sizes[0])) ||
 | 
						|
      ((radial1->sizes[1] == NULL) != (radial2->sizes[1] == NULL)) ||
 | 
						|
      (radial1->sizes[1] && radial2->sizes[1] && !_gtk_css_value_equal (radial1->sizes[1], radial2->sizes[1])) ||
 | 
						|
      radial1->stops->len != radial2->stops->len)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  for (i = 0; i < radial1->stops->len; i++)
 | 
						|
    {
 | 
						|
      GtkCssImageRadialColorStop *stop1, *stop2;
 | 
						|
 | 
						|
      stop1 = &g_array_index (radial1->stops, GtkCssImageRadialColorStop, i);
 | 
						|
      stop2 = &g_array_index (radial2->stops, GtkCssImageRadialColorStop, 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_radial_dispose (GObject *object)
 | 
						|
{
 | 
						|
  GtkCssImageRadial *radial = GTK_CSS_IMAGE_RADIAL (object);
 | 
						|
  int i;
 | 
						|
 | 
						|
  if (radial->stops)
 | 
						|
    {
 | 
						|
      g_array_free (radial->stops, TRUE);
 | 
						|
      radial->stops = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
  if (radial->position)
 | 
						|
    {
 | 
						|
      _gtk_css_value_unref (radial->position);
 | 
						|
      radial->position = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
  for (i = 0; i < 2; i++)
 | 
						|
    if (radial->sizes[i])
 | 
						|
      {
 | 
						|
        _gtk_css_value_unref (radial->sizes[i]);
 | 
						|
        radial->sizes[i] = NULL;
 | 
						|
      }
 | 
						|
 | 
						|
  G_OBJECT_CLASS (_gtk_css_image_radial_parent_class)->dispose (object);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
_gtk_css_image_radial_class_init (GtkCssImageRadialClass *klass)
 | 
						|
{
 | 
						|
  GtkCssImageClass *image_class = GTK_CSS_IMAGE_CLASS (klass);
 | 
						|
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | 
						|
 | 
						|
  image_class->draw = gtk_css_image_radial_draw;
 | 
						|
  image_class->parse = gtk_css_image_radial_parse;
 | 
						|
  image_class->print = gtk_css_image_radial_print;
 | 
						|
  image_class->compute = gtk_css_image_radial_compute;
 | 
						|
  image_class->transition = gtk_css_image_radial_transition;
 | 
						|
  image_class->equal = gtk_css_image_radial_equal;
 | 
						|
 | 
						|
  object_class->dispose = gtk_css_image_radial_dispose;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_css_image_clear_color_stop (gpointer color_stop)
 | 
						|
{
 | 
						|
  GtkCssImageRadialColorStop *stop = color_stop;
 | 
						|
 | 
						|
  _gtk_css_value_unref (stop->color);
 | 
						|
  if (stop->offset)
 | 
						|
    _gtk_css_value_unref (stop->offset);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
_gtk_css_image_radial_init (GtkCssImageRadial *radial)
 | 
						|
{
 | 
						|
  radial->stops = g_array_new (FALSE, FALSE, sizeof (GtkCssImageRadialColorStop));
 | 
						|
  g_array_set_clear_func (radial->stops, gtk_css_image_clear_color_stop);
 | 
						|
}
 | 
						|
 |