1005 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1005 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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 <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * Authors: Benjamin Otte <otte@gnome.org>
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| 
 | |
| #include "gtkcssimagebuiltinprivate.h"
 | |
| 
 | |
| #include "gtkcssenumvalueprivate.h"
 | |
| #include "gtkcssnumbervalueprivate.h"
 | |
| #include "gtkcssrgbavalueprivate.h"
 | |
| #include "gtkcssstyleprivate.h"
 | |
| #include "gtkhslaprivate.h"
 | |
| #include "gtkrenderprivate.h"
 | |
| 
 | |
| #include <math.h>
 | |
| 
 | |
| #include "fallback-c89.c"
 | |
| 
 | |
| G_DEFINE_TYPE (GtkCssImageBuiltin, gtk_css_image_builtin, GTK_TYPE_CSS_IMAGE)
 | |
| 
 | |
| static GtkCssImage *the_one_true_image;
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_draw_check (GtkCssImage *image,
 | |
|                                   cairo_t     *cr,
 | |
|                                   double       width,
 | |
|                                   double       height,
 | |
|                                   gboolean     checked,
 | |
|                                   gboolean     inconsistent)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   gint x, y, exterior_size, interior_size, thickness, pad;
 | |
| 
 | |
|   exterior_size = MIN (width, height);
 | |
| 
 | |
|   if (exterior_size % 2 == 0) /* Ensure odd */
 | |
|     exterior_size -= 1;
 | |
| 
 | |
|   /* FIXME: thickness */
 | |
|   thickness = 1;
 | |
|   pad = thickness + MAX (1, (exterior_size - 2 * thickness) / 9);
 | |
|   interior_size = MAX (1, exterior_size - 2 * pad);
 | |
| 
 | |
|   if (interior_size < 7)
 | |
|     {
 | |
|       interior_size = 7;
 | |
|       pad = MAX (0, (exterior_size - interior_size) / 2);
 | |
|     }
 | |
| 
 | |
|   x = - (1 + exterior_size - (gint) width) / 2;
 | |
|   y = - (1 + exterior_size - (gint) height) / 2;
 | |
| 
 | |
|   if (builtin->border_width > 0)
 | |
|     {
 | |
|       cairo_set_line_width (cr, builtin->border_width);
 | |
| 
 | |
|       cairo_rectangle (cr, x + 0.5, y + 0.5, exterior_size - 1, exterior_size - 1);
 | |
|       gdk_cairo_set_source_rgba (cr, &builtin->bg_color);
 | |
|       cairo_fill_preserve (cr);
 | |
| 
 | |
|       gdk_cairo_set_source_rgba (cr, &builtin->border_color);
 | |
|       cairo_stroke (cr);
 | |
|     }
 | |
| 
 | |
|   gdk_cairo_set_source_rgba (cr, &builtin->fg_color);
 | |
| 
 | |
|   if (inconsistent)
 | |
|     {
 | |
|       int line_thickness = MAX (1, (3 + interior_size * 2) / 7);
 | |
| 
 | |
|       cairo_rectangle (cr,
 | |
| 		       x + pad,
 | |
| 		       y + pad + (1 + interior_size - line_thickness) / 2,
 | |
| 		       interior_size,
 | |
| 		       line_thickness);
 | |
|       cairo_fill (cr);
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       if (checked)
 | |
|         {
 | |
|           cairo_translate (cr,
 | |
|                            x + pad, y + pad);
 | |
| 
 | |
|           cairo_scale (cr, interior_size / 7., interior_size / 7.);
 | |
| 
 | |
|           cairo_rectangle (cr, 0, 0, 7, 7);
 | |
|           cairo_clip (cr);
 | |
| 
 | |
|           cairo_move_to  (cr, 7.0, 0.0);
 | |
|           cairo_line_to  (cr, 7.5, 1.0);
 | |
|           cairo_curve_to (cr, 5.3, 2.0,
 | |
|                           4.3, 4.0,
 | |
|                           3.5, 7.0);
 | |
|           cairo_curve_to (cr, 3.0, 5.7,
 | |
|                           1.3, 4.7,
 | |
|                           0.0, 4.7);
 | |
|           cairo_line_to  (cr, 0.2, 3.5);
 | |
|           cairo_curve_to (cr, 1.1, 3.5,
 | |
|                           2.3, 4.3,
 | |
|                           3.0, 5.0);
 | |
|           cairo_curve_to (cr, 1.0, 3.9,
 | |
|                           2.4, 4.1,
 | |
|                           3.2, 4.9);
 | |
|           cairo_curve_to (cr, 3.5, 3.1,
 | |
|                           5.2, 2.0,
 | |
|                           7.0, 0.0);
 | |
| 
 | |
|           cairo_fill (cr);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_draw_option (GtkCssImage *image,
 | |
|                                    cairo_t     *cr,
 | |
|                                    double       width,
 | |
|                                    double       height,
 | |
|                                    gboolean     checked,
 | |
|                                    gboolean     inconsistent)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   gint x, y, exterior_size, interior_size, thickness, pad;
 | |
| 
 | |
|   exterior_size = MIN (width, height);
 | |
| 
 | |
|   if (exterior_size % 2 == 0) /* Ensure odd */
 | |
|     exterior_size -= 1;
 | |
| 
 | |
|   x = - (1 + exterior_size - width) / 2;
 | |
|   y = - (1 + exterior_size - height) / 2;
 | |
| 
 | |
|   if (builtin->border_width > 0)
 | |
|     {
 | |
|       cairo_set_line_width (cr, builtin->border_width);
 | |
| 
 | |
|       cairo_new_sub_path (cr);
 | |
|       cairo_arc (cr,
 | |
|                  x + exterior_size / 2.,
 | |
|                  y + exterior_size / 2.,
 | |
|                  (exterior_size - 1) / 2.,
 | |
|                  0, 2 * G_PI);
 | |
| 
 | |
|       gdk_cairo_set_source_rgba (cr, &builtin->bg_color);
 | |
|       cairo_fill_preserve (cr);
 | |
| 
 | |
|       gdk_cairo_set_source_rgba (cr, &builtin->border_color);
 | |
|       cairo_stroke (cr);
 | |
|     }
 | |
| 
 | |
|   gdk_cairo_set_source_rgba (cr, &builtin->fg_color);
 | |
| 
 | |
|   /* FIXME: thickness */
 | |
|   thickness = 1;
 | |
| 
 | |
|   if (inconsistent)
 | |
|     {
 | |
|       gint line_thickness;
 | |
| 
 | |
|       pad = thickness + MAX (1, (exterior_size - 2 * thickness) / 9);
 | |
|       interior_size = MAX (1, exterior_size - 2 * pad);
 | |
| 
 | |
|       if (interior_size < 7)
 | |
|         {
 | |
|           interior_size = 7;
 | |
|           pad = MAX (0, (exterior_size - interior_size) / 2);
 | |
|         }
 | |
| 
 | |
|       line_thickness = MAX (1, (3 + interior_size * 2) / 7);
 | |
| 
 | |
|       cairo_rectangle (cr,
 | |
|                        x + pad,
 | |
|                        y + pad + (interior_size - line_thickness) / 2.,
 | |
|                        interior_size,
 | |
|                        line_thickness);
 | |
|       cairo_fill (cr);
 | |
|     }
 | |
|   if (checked)
 | |
|     {
 | |
|       pad = thickness + MAX (1, 2 * (exterior_size - 2 * thickness) / 9);
 | |
|       interior_size = MAX (1, exterior_size - 2 * pad);
 | |
| 
 | |
|       if (interior_size < 5)
 | |
|         {
 | |
|           interior_size = 7;
 | |
|           pad = MAX (0, (exterior_size - interior_size) / 2);
 | |
|         }
 | |
| 
 | |
|       cairo_new_sub_path (cr);
 | |
|       cairo_arc (cr,
 | |
|                  x + pad + interior_size / 2.,
 | |
|                  y + pad + interior_size / 2.,
 | |
|                  interior_size / 2.,
 | |
|                  0, 2 * G_PI);
 | |
|       cairo_fill (cr);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_draw_arrow (GtkCssImage            *image,
 | |
|                                   cairo_t                *cr,
 | |
|                                   double                  width,
 | |
|                                   double                  height,
 | |
|                                   GtkCssImageBuiltinType  image_type)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   double line_width;
 | |
|   double size;
 | |
| 
 | |
|   size = MIN (width, height);
 | |
| 
 | |
|   cairo_translate (cr, width / 2.0, height / 2.0);
 | |
|   switch (image_type)
 | |
|   {
 | |
|     case GTK_CSS_IMAGE_BUILTIN_ARROW_UP:
 | |
|       break;
 | |
|     case GTK_CSS_IMAGE_BUILTIN_ARROW_DOWN:
 | |
|       cairo_rotate (cr, G_PI);
 | |
|       break;
 | |
|     case GTK_CSS_IMAGE_BUILTIN_ARROW_LEFT:
 | |
|       cairo_rotate (cr, 3 * G_PI / 2);
 | |
|       break;
 | |
|     case GTK_CSS_IMAGE_BUILTIN_ARROW_RIGHT:
 | |
|       cairo_rotate (cr, G_PI / 2);
 | |
|       break;
 | |
|     default:
 | |
|       g_assert_not_reached ();
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   line_width = size / 3.0 / sqrt (2);
 | |
|   cairo_set_line_width (cr, line_width);
 | |
|   cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
 | |
|   cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
 | |
| 
 | |
|   cairo_scale (cr,
 | |
|                (size / (size + line_width)),
 | |
|                (size / (size + line_width)));
 | |
| 
 | |
|   cairo_move_to (cr, -size / 2.0, size / 4.0);
 | |
|   cairo_rel_line_to (cr, size / 2.0, -size / 2.0);
 | |
|   cairo_rel_line_to (cr, size / 2.0, size / 2.0);
 | |
| 
 | |
|   gdk_cairo_set_source_rgba (cr, &builtin->fg_color);
 | |
|   cairo_stroke (cr);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_draw_expander (GtkCssImage *image,
 | |
|                                      cairo_t     *cr,
 | |
|                                      double       width,
 | |
|                                      double       height,
 | |
|                                      gboolean     horizontal,
 | |
|                                      gboolean     is_rtl,
 | |
|                                      gboolean     expanded)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   double vertical_overshoot;
 | |
|   int diameter;
 | |
|   double radius;
 | |
|   double interp;		/* interpolation factor for center position */
 | |
|   double x_double_horz, y_double_horz;
 | |
|   double x_double_vert, y_double_vert;
 | |
|   double x_double, y_double;
 | |
|   gdouble angle;
 | |
|   gint line_width;
 | |
|   gdouble progress;
 | |
| 
 | |
|   line_width = 1;
 | |
|   progress = expanded ? 1 : 0;
 | |
| 
 | |
|   if (!horizontal)
 | |
|     {
 | |
|       if (is_rtl)
 | |
|         angle = (G_PI) - ((G_PI / 2) * progress);
 | |
|       else
 | |
|         angle = (G_PI / 2) * progress;
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       if (is_rtl)
 | |
|         angle = (G_PI / 2) + ((G_PI / 2) * progress);
 | |
|       else
 | |
|         angle = (G_PI / 2) - ((G_PI / 2) * progress);
 | |
|     }
 | |
| 
 | |
|   interp = progress;
 | |
| 
 | |
|   /* Compute distance that the stroke extends beyonds the end
 | |
|    * of the triangle we draw.
 | |
|    */
 | |
|   vertical_overshoot = line_width / 2.0 * (1. / tan (G_PI / 8));
 | |
| 
 | |
|   /* For odd line widths, we end the vertical line of the triangle
 | |
|    * at a half pixel, so we round differently.
 | |
|    */
 | |
|   if (line_width % 2 == 1)
 | |
|     vertical_overshoot = ceil (0.5 + vertical_overshoot) - 0.5;
 | |
|   else
 | |
|     vertical_overshoot = ceil (vertical_overshoot);
 | |
| 
 | |
|   /* Adjust the size of the triangle we draw so that the entire stroke fits
 | |
|    */
 | |
|   diameter = (gint) MAX (3, width - 2 * vertical_overshoot);
 | |
| 
 | |
|   /* If the line width is odd, we want the diameter to be even,
 | |
|    * and vice versa, so force the sum to be odd. This relationship
 | |
|    * makes the point of the triangle look right.
 | |
|    */
 | |
|   diameter -= (1 - (diameter + line_width) % 2);
 | |
| 
 | |
|   radius = diameter / 2.;
 | |
| 
 | |
|   /* Adjust the center so that the stroke is properly aligned with
 | |
|    * the pixel grid. The center adjustment is different for the
 | |
|    * horizontal and vertical orientations. For intermediate positions
 | |
|    * we interpolate between the two.
 | |
|    */
 | |
|   x_double_vert = floor ((width / 2) - (radius + line_width) / 2.) + (radius + line_width) / 2.;
 | |
|   y_double_vert = (height / 2) - 0.5;
 | |
| 
 | |
|   x_double_horz = (width / 2) - 0.5;
 | |
|   y_double_horz = floor ((height / 2) - (radius + line_width) / 2.) + (radius + line_width) / 2.;
 | |
| 
 | |
|   x_double = x_double_vert * (1 - interp) + x_double_horz * interp;
 | |
|   y_double = y_double_vert * (1 - interp) + y_double_horz * interp;
 | |
| 
 | |
|   cairo_translate (cr, x_double, y_double);
 | |
|   cairo_rotate (cr, angle);
 | |
| 
 | |
|   cairo_move_to (cr, - radius / 2., - radius);
 | |
|   cairo_line_to (cr,   radius / 2.,   0);
 | |
|   cairo_line_to (cr, - radius / 2.,   radius);
 | |
|   cairo_close_path (cr);
 | |
| 
 | |
|   cairo_set_line_width (cr, line_width);
 | |
| 
 | |
|   gdk_cairo_set_source_rgba (cr, &builtin->fg_color);
 | |
| 
 | |
|   cairo_fill_preserve (cr);
 | |
| 
 | |
|   gdk_cairo_set_source_rgba (cr, &builtin->border_color);
 | |
|   cairo_stroke (cr);
 | |
| }
 | |
| 
 | |
| static void
 | |
| color_shade (const GdkRGBA *color,
 | |
|              gdouble        factor,
 | |
|              GdkRGBA       *color_return)
 | |
| {
 | |
|   GtkHSLA hsla;
 | |
| 
 | |
|   _gtk_hsla_init_from_rgba (&hsla, color);
 | |
|   _gtk_hsla_shade (&hsla, &hsla, factor);
 | |
|   _gdk_rgba_init_from_hsla (color_return, &hsla);
 | |
| }
 | |
| 
 | |
| static void
 | |
| render_dot (cairo_t       *cr,
 | |
|             const GdkRGBA *lighter,
 | |
|             const GdkRGBA *darker,
 | |
|             gdouble        x,
 | |
|             gdouble        y,
 | |
|             gdouble        size)
 | |
| {
 | |
|   size = CLAMP ((gint) size, 2, 3);
 | |
| 
 | |
|   if (size == 2)
 | |
|     {
 | |
|       gdk_cairo_set_source_rgba (cr, lighter);
 | |
|       cairo_rectangle (cr, x, y, 1, 1);
 | |
|       cairo_rectangle (cr, x + 1, y + 1, 1, 1);
 | |
|       cairo_fill (cr);
 | |
|     }
 | |
|   else if (size == 3)
 | |
|     {
 | |
|       gdk_cairo_set_source_rgba (cr, lighter);
 | |
|       cairo_rectangle (cr, x, y, 2, 1);
 | |
|       cairo_rectangle (cr, x, y, 1, 2);
 | |
|       cairo_fill (cr);
 | |
| 
 | |
|       gdk_cairo_set_source_rgba (cr, darker);
 | |
|       cairo_rectangle (cr, x + 1, y + 1, 2, 1);
 | |
|       cairo_rectangle (cr, x + 2, y, 1, 2);
 | |
|       cairo_fill (cr);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| add_path_line (cairo_t        *cr,
 | |
|                gdouble         x1,
 | |
|                gdouble         y1,
 | |
|                gdouble         x2,
 | |
|                gdouble         y2)
 | |
| {
 | |
|   /* Adjust endpoints */
 | |
|   if (y1 == y2)
 | |
|     {
 | |
|       y1 += 0.5;
 | |
|       y2 += 0.5;
 | |
|       x2 += 1;
 | |
|     }
 | |
|   else if (x1 == x2)
 | |
|     {
 | |
|       x1 += 0.5;
 | |
|       x2 += 0.5;
 | |
|       y2 += 1;
 | |
|     }
 | |
| 
 | |
|   cairo_move_to (cr, x1, y1);
 | |
|   cairo_line_to (cr, x2, y2);
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_css_image_builtin_draw_grip (GtkCssImage            *image,
 | |
|                                  cairo_t                *cr,
 | |
|                                  double                  width,
 | |
|                                  double                  height,
 | |
|                                  GtkCssImageBuiltinType  image_type)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   GdkRGBA lighter, darker;
 | |
| 
 | |
|   cairo_set_line_width (cr, 1.0);
 | |
| 
 | |
|   color_shade (&builtin->bg_color, 0.7, &darker);
 | |
|   color_shade (&builtin->bg_color, 1.3, &lighter);
 | |
| 
 | |
|   /* align drawing area to the connected side */
 | |
|   if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_LEFT)
 | |
|     {
 | |
|       if (height < width)
 | |
|         width = height;
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_TOPLEFT)
 | |
|     {
 | |
|       if (width < height)
 | |
|         height = width;
 | |
|       else if (height < width)
 | |
|         width = height;
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOMLEFT)
 | |
|     {
 | |
|       /* make it square, aligning to bottom left */
 | |
|       if (width < height)
 | |
|         {
 | |
|           cairo_translate (cr, 0, height - width);
 | |
|           height = width;
 | |
|         }
 | |
|       else if (height < width)
 | |
|         width = height;
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_RIGHT)
 | |
|     {
 | |
|       /* aligning to right */
 | |
|       if (height < width)
 | |
|         {
 | |
|           cairo_translate (cr, width - height, 0);
 | |
|           width = height;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_TOPRIGHT)
 | |
|     {
 | |
|       if (width < height)
 | |
|         height = width;
 | |
|       else if (height < width)
 | |
|         {
 | |
|           cairo_translate (cr, width - height, 0);
 | |
|           width = height;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOMRIGHT)
 | |
|     {
 | |
|       /* make it square, aligning to bottom right */
 | |
|       if (width < height)
 | |
|         {
 | |
|           cairo_translate (cr, 0, height - width);
 | |
|           height = width;
 | |
|         }
 | |
|       else if (height < width)
 | |
|         {
 | |
|           cairo_translate (cr, width - height, 0);
 | |
|           width = height;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_TOP)
 | |
|     {
 | |
|       if (width < height)
 | |
|         height = width;
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOM)
 | |
|     {
 | |
|       /* align to bottom */
 | |
|       if (width < height)
 | |
|         {
 | |
|           cairo_translate (cr, 0, height - width);
 | |
|           height = width;
 | |
|         }
 | |
|     }
 | |
|   else
 | |
|     g_assert_not_reached ();
 | |
| 
 | |
|   if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_LEFT ||
 | |
|       image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_RIGHT)
 | |
|     {
 | |
|       gint xi;
 | |
| 
 | |
|       xi = 0;
 | |
| 
 | |
|       while (xi < width)
 | |
|         {
 | |
|           gdk_cairo_set_source_rgba (cr, &lighter);
 | |
|           add_path_line (cr, 0, 0, 0, height);
 | |
|           cairo_stroke (cr);
 | |
|           xi++;
 | |
| 
 | |
|           gdk_cairo_set_source_rgba (cr, &darker);
 | |
|           add_path_line (cr, xi, 0, xi, height);
 | |
|           cairo_stroke (cr);
 | |
|           xi += 2;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_TOP ||
 | |
|            image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOM)
 | |
|     {
 | |
|       gint yi;
 | |
| 
 | |
|       yi = 0;
 | |
| 
 | |
|       while (yi < height)
 | |
|         {
 | |
|           gdk_cairo_set_source_rgba (cr, &lighter);
 | |
|           add_path_line (cr, 0, yi, width, yi);
 | |
|           cairo_stroke (cr);
 | |
|           yi++;
 | |
| 
 | |
|           gdk_cairo_set_source_rgba (cr, &darker);
 | |
|           add_path_line (cr, 0, yi, width, yi);
 | |
|           cairo_stroke (cr);
 | |
|           yi += 2;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_TOPLEFT)
 | |
|     {
 | |
|       gint xi, yi;
 | |
| 
 | |
|       xi = width;
 | |
|       yi = height;
 | |
| 
 | |
|       while (xi > 3)
 | |
|         {
 | |
|           gdk_cairo_set_source_rgba (cr, &darker);
 | |
|           add_path_line (cr, xi, 0, 0, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           --xi;
 | |
|           --yi;
 | |
| 
 | |
|           add_path_line (cr, xi, 0, 0, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           --xi;
 | |
|           --yi;
 | |
| 
 | |
|           gdk_cairo_set_source_rgba (cr, &lighter);
 | |
|           add_path_line (cr, xi, 0, 0, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           xi -= 3;
 | |
|           yi -= 3;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_TOPRIGHT)
 | |
|     {
 | |
|       gint xi, yi;
 | |
| 
 | |
|       xi = 0;
 | |
|       yi = height;
 | |
| 
 | |
|       while (xi < (width - 3))
 | |
|         {
 | |
|           gdk_cairo_set_source_rgba (cr, &lighter);
 | |
|           add_path_line (cr, xi, 0, width, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           ++xi;
 | |
|           --yi;
 | |
| 
 | |
|           gdk_cairo_set_source_rgba (cr, &darker);
 | |
|           add_path_line (cr, xi, 0, width, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           ++xi;
 | |
|           --yi;
 | |
| 
 | |
|           add_path_line (cr, xi, 0, width, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           xi += 3;
 | |
|           yi -= 3;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOMLEFT)
 | |
|     {
 | |
|       gint xi, yi;
 | |
| 
 | |
|       xi = width;
 | |
|       yi = 0;
 | |
| 
 | |
|       while (xi > 3)
 | |
|         {
 | |
|           gdk_cairo_set_source_rgba (cr, &darker);
 | |
|           add_path_line (cr, 0, yi, xi, height);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           --xi;
 | |
|           ++yi;
 | |
| 
 | |
|           add_path_line (cr, 0, yi, xi, height);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           --xi;
 | |
|           ++yi;
 | |
| 
 | |
|           gdk_cairo_set_source_rgba (cr, &lighter);
 | |
|           add_path_line (cr, 0, yi, xi, height);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           xi -= 3;
 | |
|           yi += 3;
 | |
|         }
 | |
|     }
 | |
|   else if (image_type == GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOMRIGHT)
 | |
|     {
 | |
|       gint xi, yi;
 | |
| 
 | |
|       xi = 0;
 | |
|       yi = 0;
 | |
| 
 | |
|       while (xi < (width - 3))
 | |
|         {
 | |
|           gdk_cairo_set_source_rgba (cr, &lighter);
 | |
|           add_path_line (cr, xi, height, width, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           ++xi;
 | |
|           ++yi;
 | |
| 
 | |
|           gdk_cairo_set_source_rgba (cr, &darker);
 | |
|           add_path_line (cr, xi, height, width, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           ++xi;
 | |
|           ++yi;
 | |
| 
 | |
|           add_path_line (cr, xi, height, width, yi);
 | |
|           cairo_stroke (cr);
 | |
| 
 | |
|           xi += 3;
 | |
|           yi += 3;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_css_image_builtin_draw_pane_separator (GtkCssImage *image,
 | |
|                                            cairo_t     *cr,
 | |
|                                            double       width,
 | |
|                                            double       height)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   GdkRGBA lighter, darker;
 | |
|   gint xx, yy;
 | |
| 
 | |
|   cairo_set_line_width (cr, 1.0);
 | |
| 
 | |
|   color_shade (&builtin->bg_color, 0.7, &darker);
 | |
|   color_shade (&builtin->bg_color, 1.3, &lighter);
 | |
| 
 | |
|   if (width > height)
 | |
|     for (xx = width / 2 - 15; xx <= width / 2 + 15; xx += 5)
 | |
|       render_dot (cr, &lighter, &darker, xx, height / 2 - 1, 3);
 | |
|   else
 | |
|     for (yy = height / 2 - 15; yy <= height / 2 + 15; yy += 5)
 | |
|       render_dot (cr, &lighter, &darker, width / 2 - 1, yy, 3);
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_css_image_builtin_draw_handle (GtkCssImage *image,
 | |
|                                    cairo_t     *cr,
 | |
|                                    double       width,
 | |
|                                    double       height)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   GdkRGBA lighter, darker;
 | |
|   gint xx, yy;
 | |
| 
 | |
|   cairo_set_line_width (cr, 1.0);
 | |
| 
 | |
|   color_shade (&builtin->bg_color, 0.7, &darker);
 | |
|   color_shade (&builtin->bg_color, 1.3, &lighter);
 | |
| 
 | |
|   for (yy = 0; yy < height; yy += 3)
 | |
|     for (xx = 0; xx < width; xx += 6)
 | |
|       {
 | |
|         render_dot (cr, &lighter, &darker, xx, yy, 2);
 | |
|         render_dot (cr, &lighter, &darker, xx + 3, yy + 1, 2);
 | |
|       }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_draw_spinner (GtkCssImage *image,
 | |
|                                     cairo_t     *cr,
 | |
|                                     double       width,
 | |
|                                     double       height)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin = GTK_CSS_IMAGE_BUILTIN (image);
 | |
|   guint num_steps;
 | |
|   gdouble radius;
 | |
|   gdouble half;
 | |
|   gint i;
 | |
| 
 | |
|   radius = MIN (width / 2, height / 2);
 | |
| 
 | |
|   cairo_translate (cr, width / 2, height / 2);
 | |
| 
 | |
|   num_steps = 12;
 | |
| 
 | |
|   cairo_set_line_width (cr, 2.0);
 | |
| 
 | |
|   half = num_steps / 2;
 | |
| 
 | |
|   for (i = 0; i < num_steps; i++)
 | |
|     {
 | |
|       gint inset = 0.7 * radius;
 | |
|       /* transparency is a function of time and intial value */
 | |
|       gdouble t = 1.0 - (gdouble) i / num_steps;
 | |
|       gdouble xscale = - sin (i * G_PI / half);
 | |
|       gdouble yscale = - cos (i * G_PI / half);
 | |
| 
 | |
|       cairo_move_to (cr,
 | |
|                      (radius - inset) * xscale,
 | |
|                      (radius - inset) * yscale);
 | |
|       cairo_line_to (cr,
 | |
|                      radius * xscale,
 | |
|                      radius * yscale);
 | |
| 
 | |
|       cairo_set_source_rgba (cr,
 | |
|                              builtin->fg_color.red,
 | |
|                              builtin->fg_color.green,
 | |
|                              builtin->fg_color.blue,
 | |
|                              builtin->fg_color.alpha * t);
 | |
| 
 | |
|       cairo_stroke (cr);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_real_draw (GtkCssImage        *image,
 | |
|                                  cairo_t            *cr,
 | |
|                                  double              width,
 | |
|                                  double              height)
 | |
| {
 | |
|   /* It's a builtin image, other code will draw things */
 | |
| }
 | |
| 
 | |
| 
 | |
| static gboolean
 | |
| gtk_css_image_builtin_parse (GtkCssImage  *image,
 | |
|                              GtkCssParser *parser)
 | |
| {
 | |
|   if (!_gtk_css_parser_try (parser, "builtin", TRUE))
 | |
|     {
 | |
|       _gtk_css_parser_error (parser, "Expected 'builtin'");
 | |
|       return FALSE;
 | |
|     }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_print (GtkCssImage *image,
 | |
|                              GString     *string)
 | |
| {
 | |
|   g_string_append (string, "builtin");
 | |
| }
 | |
| 
 | |
| static GtkCssImage *
 | |
| gtk_css_image_builtin_compute (GtkCssImage             *image,
 | |
|                                guint                    property_id,
 | |
|                                GtkStyleProviderPrivate *provider,
 | |
|                                GtkCssStyle             *style,
 | |
|                                GtkCssStyle             *parent_style)
 | |
| {
 | |
|   GtkCssImageBuiltin *result;
 | |
|   GtkBorderStyle border_style;
 | |
| 
 | |
|   result = g_object_new (GTK_TYPE_CSS_IMAGE_BUILTIN, NULL);
 | |
| 
 | |
|   border_style = _gtk_css_border_style_value_get (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BORDER_TOP_STYLE));
 | |
|   if (border_style == GTK_BORDER_STYLE_SOLID)
 | |
|     {
 | |
|       GtkBorder border;
 | |
| 
 | |
|       border.top = _gtk_css_number_value_get (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BORDER_TOP_WIDTH), 100);
 | |
|       border.right = _gtk_css_number_value_get (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BORDER_RIGHT_WIDTH), 100);
 | |
|       border.bottom = _gtk_css_number_value_get (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BORDER_BOTTOM_WIDTH), 100);
 | |
|       border.left = _gtk_css_number_value_get (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BORDER_LEFT_WIDTH), 100);
 | |
| 
 | |
|       result->border_width = MIN (MIN (border.top, border.bottom),
 | |
|                                   MIN (border.left, border.right));
 | |
|     }
 | |
| 
 | |
|   result->fg_color = *_gtk_css_rgba_value_get_rgba (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_COLOR));
 | |
|   result->bg_color = *_gtk_css_rgba_value_get_rgba (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BACKGROUND_COLOR));
 | |
|   result->border_color = *_gtk_css_rgba_value_get_rgba (gtk_css_style_get_value (style, GTK_CSS_PROPERTY_BORDER_TOP_COLOR));
 | |
| 
 | |
|   return GTK_CSS_IMAGE (result);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| gtk_css_image_builtin_equal (GtkCssImage *image1,
 | |
|                              GtkCssImage *image2)
 | |
| {
 | |
|   GtkCssImageBuiltin *builtin1 = GTK_CSS_IMAGE_BUILTIN (image1);
 | |
|   GtkCssImageBuiltin *builtin2 = GTK_CSS_IMAGE_BUILTIN (image2);
 | |
| 
 | |
|   return gdk_rgba_equal (&builtin1->fg_color, &builtin2->fg_color)
 | |
|       && gdk_rgba_equal (&builtin1->bg_color, &builtin2->bg_color)
 | |
|       && gdk_rgba_equal (&builtin1->border_color, &builtin2->border_color)
 | |
|       && builtin1->border_width == builtin2->border_width;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_dispose (GObject *object)
 | |
| {
 | |
|   if (the_one_true_image == GTK_CSS_IMAGE (object))
 | |
|     the_one_true_image = NULL;
 | |
| 
 | |
|   G_OBJECT_CLASS (gtk_css_image_builtin_parent_class)->dispose (object);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_class_init (GtkCssImageBuiltinClass *klass)
 | |
| {
 | |
|   GtkCssImageClass *image_class = GTK_CSS_IMAGE_CLASS (klass);
 | |
|   GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | |
| 
 | |
|   image_class->draw = gtk_css_image_builtin_real_draw;
 | |
|   image_class->parse = gtk_css_image_builtin_parse;
 | |
|   image_class->print = gtk_css_image_builtin_print;
 | |
|   image_class->compute = gtk_css_image_builtin_compute;
 | |
|   image_class->equal = gtk_css_image_builtin_equal;
 | |
| 
 | |
|   object_class->dispose = gtk_css_image_builtin_dispose;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_image_builtin_init (GtkCssImageBuiltin *builtin)
 | |
| {
 | |
|   /* white background */
 | |
|   builtin->bg_color.red = builtin->bg_color.green = builtin->bg_color.blue = builtin->bg_color.alpha = 1.0;
 | |
|   /* black foreground */
 | |
|   builtin->fg_color.alpha = 1.0;
 | |
| }
 | |
| 
 | |
| GtkCssImage *
 | |
| gtk_css_image_builtin_new (void)
 | |
| {
 | |
|   if (the_one_true_image == NULL)
 | |
|     the_one_true_image = g_object_new (GTK_TYPE_CSS_IMAGE_BUILTIN, NULL);
 | |
|   else
 | |
|     g_object_ref (the_one_true_image);
 | |
| 
 | |
|   return the_one_true_image;
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_css_image_builtin_draw (GtkCssImage            *image,
 | |
|                             cairo_t                *cr,
 | |
|                             double                  width,
 | |
|                             double                  height,
 | |
|                             GtkCssImageBuiltinType  image_type)
 | |
| {
 | |
|   if (!GTK_IS_CSS_IMAGE_BUILTIN (image))
 | |
|     {
 | |
|       _gtk_css_image_draw (image, cr, width, height);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|   switch (image_type)
 | |
|   {
 | |
|   default:
 | |
|     g_assert_not_reached ();
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_NONE:
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_CHECK:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_CHECK_CHECKED:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_CHECK_INCONSISTENT:
 | |
|     gtk_css_image_builtin_draw_check (image, cr,
 | |
|                                       width, height,
 | |
|                                       image_type == GTK_CSS_IMAGE_BUILTIN_CHECK_CHECKED,
 | |
|                                       image_type == GTK_CSS_IMAGE_BUILTIN_CHECK_INCONSISTENT);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_OPTION:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_OPTION_CHECKED:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_OPTION_INCONSISTENT:
 | |
|     gtk_css_image_builtin_draw_option (image, cr,
 | |
|                                        width, height,
 | |
|                                        image_type == GTK_CSS_IMAGE_BUILTIN_OPTION_CHECKED,
 | |
|                                        image_type == GTK_CSS_IMAGE_BUILTIN_OPTION_INCONSISTENT);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_ARROW_UP:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_ARROW_DOWN:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_ARROW_LEFT:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_ARROW_RIGHT:
 | |
|     gtk_css_image_builtin_draw_arrow (image, cr,
 | |
|                                       width, height,
 | |
|                                       image_type);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_HORIZONTAL_LEFT:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          TRUE, FALSE, FALSE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_VERTICAL_LEFT:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          FALSE, FALSE, FALSE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_HORIZONTAL_RIGHT:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          TRUE, TRUE, FALSE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_VERTICAL_RIGHT:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          FALSE, TRUE, FALSE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_HORIZONTAL_LEFT_EXPANDED:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          TRUE, FALSE, TRUE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_VERTICAL_LEFT_EXPANDED:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          FALSE, FALSE, TRUE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_HORIZONTAL_RIGHT_EXPANDED:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          TRUE, TRUE, TRUE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_EXPANDER_VERTICAL_RIGHT_EXPANDED:
 | |
|     gtk_css_image_builtin_draw_expander (image, cr,
 | |
|                                          width, height,
 | |
|                                          FALSE, TRUE, TRUE);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_TOPLEFT:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_TOP:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_TOPRIGHT:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_RIGHT:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOMRIGHT:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOM:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_BOTTOMLEFT:
 | |
|   case GTK_CSS_IMAGE_BUILTIN_GRIP_LEFT:
 | |
|     gtk_css_image_builtin_draw_grip (image, cr,
 | |
|                                      width, height,
 | |
|                                      image_type);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_PANE_SEPARATOR:
 | |
|     gtk_css_image_builtin_draw_pane_separator (image, cr,
 | |
|                                                width, height);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_HANDLE:
 | |
|     gtk_css_image_builtin_draw_handle (image, cr,
 | |
|                                        width, height);
 | |
|     break;
 | |
|   case GTK_CSS_IMAGE_BUILTIN_SPINNER:
 | |
|     gtk_css_image_builtin_draw_spinner (image, cr,
 | |
|                                         width, height);
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | 
