402 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			402 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GTK - The GIMP Toolkit
 | |
|  * Copyright (C) 2011 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 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/>.
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| 
 | |
| #include "gtkcsswin32sizevalueprivate.h"
 | |
| 
 | |
| #include "gtkwin32drawprivate.h"
 | |
| #include "gtkwin32themeprivate.h"
 | |
| 
 | |
| typedef enum {
 | |
|   GTK_WIN32_SIZE,
 | |
|   GTK_WIN32_PART_WIDTH,
 | |
|   GTK_WIN32_PART_HEIGHT,
 | |
|   GTK_WIN32_PART_BORDER_TOP,
 | |
|   GTK_WIN32_PART_BORDER_RIGHT,
 | |
|   GTK_WIN32_PART_BORDER_BOTTOM,
 | |
|   GTK_WIN32_PART_BORDER_LEFT
 | |
| } GtkWin32SizeType;
 | |
| 
 | |
| static const char *css_value_names[] = {
 | |
|   "-gtk-win32-size(",
 | |
|   "-gtk-win32-part-width(",
 | |
|   "-gtk-win32-part-height(",
 | |
|   "-gtk-win32-part-border-top(",
 | |
|   "-gtk-win32-part-border-right(",
 | |
|   "-gtk-win32-part-border-bottom(",
 | |
|   "-gtk-win32-part-border-left("
 | |
| };
 | |
| 
 | |
| struct _GtkCssValue {
 | |
|   GTK_CSS_VALUE_BASE
 | |
|   double                 scale;         /* needed for calc() math */
 | |
|   GtkWin32Theme         *theme;
 | |
|   GtkWin32SizeType       type;
 | |
|   union {
 | |
|     struct {
 | |
|       gint               id;
 | |
|     } size;
 | |
|     struct {
 | |
|       gint               part;
 | |
|       gint               state;
 | |
|     } part;
 | |
|   }                      val;
 | |
| };
 | |
| 
 | |
| static GtkCssValue *    gtk_css_win32_size_value_new (double            scale,
 | |
|                                                       GtkWin32Theme    *theme,
 | |
|                                                       GtkWin32SizeType  type);
 | |
| 
 | |
| static void
 | |
| gtk_css_value_win32_size_free (GtkCssValue *value)
 | |
| {
 | |
|   gtk_win32_theme_unref (value->theme);
 | |
|   g_slice_free (GtkCssValue, value);
 | |
| }
 | |
| 
 | |
| static int
 | |
| gtk_css_value_win32_compute_size (const GtkCssValue *value)
 | |
| {
 | |
|   GtkBorder border;
 | |
|   int size;
 | |
| 
 | |
|   switch (value->type)
 | |
|     {
 | |
|     case GTK_WIN32_SIZE:
 | |
|       size = gtk_win32_theme_get_size (value->theme, value->val.size.id);
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_WIDTH:
 | |
|       gtk_win32_theme_get_part_size (value->theme, value->val.part.part, value->val.part.state, &size, NULL);
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_HEIGHT:
 | |
|       gtk_win32_theme_get_part_size (value->theme, value->val.part.part, value->val.part.state, NULL, &size);
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_BORDER_TOP:
 | |
|       gtk_win32_theme_get_part_border (value->theme, value->val.part.part, value->val.part.state, &border);
 | |
|       size = border.top;
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_BORDER_RIGHT:
 | |
|       gtk_win32_theme_get_part_border (value->theme, value->val.part.part, value->val.part.state, &border);
 | |
|       size = border.right;
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_BORDER_BOTTOM:
 | |
|       gtk_win32_theme_get_part_border (value->theme, value->val.part.part, value->val.part.state, &border);
 | |
|       size = border.bottom;
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_BORDER_LEFT:
 | |
|       gtk_win32_theme_get_part_border (value->theme, value->val.part.part, value->val.part.state, &border);
 | |
|       size = border.left;
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       g_assert_not_reached ();
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|   return size;
 | |
| }
 | |
| 
 | |
| static GtkCssValue *
 | |
| gtk_css_value_win32_size_compute (GtkCssValue             *value,
 | |
|                                   guint                    property_id,
 | |
|                                   GtkStyleProviderPrivate *provider,
 | |
|                                   GtkCssStyle             *style,
 | |
|                                   GtkCssStyle             *parent_style)
 | |
| {
 | |
|   return _gtk_css_number_value_new (value->scale * gtk_css_value_win32_compute_size (value), GTK_CSS_PX);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| gtk_css_value_win32_size_equal (const GtkCssValue *value1,
 | |
|                                 const GtkCssValue *value2)
 | |
| {
 | |
|   if (value1->type != value2->type ||
 | |
|       !gtk_win32_theme_equal (value1->theme, value2->theme) )
 | |
|     return FALSE;
 | |
| 
 | |
|   switch (value1->type)
 | |
|     {
 | |
|     case GTK_WIN32_SIZE:
 | |
|       return value1->val.size.id == value2->val.size.id;
 | |
| 
 | |
|     case GTK_WIN32_PART_WIDTH:
 | |
|     case GTK_WIN32_PART_HEIGHT:
 | |
|       return value1->val.part.part == value2->val.part.part
 | |
|           && value1->val.part.state == value2->val.part.state;
 | |
| 
 | |
|     default:
 | |
|       g_assert_not_reached ();
 | |
|       return FALSE;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_css_value_win32_size_print (const GtkCssValue *value,
 | |
|                                 GString           *string)
 | |
| {
 | |
|   if (value->scale != 1.0)
 | |
|     {
 | |
|       g_string_append_printf (string, "%g * ", value->scale);
 | |
|     }
 | |
|   g_string_append (string, css_value_names[value->type]);
 | |
|   gtk_win32_theme_print (value->theme, string);
 | |
| 
 | |
|   switch (value->type)
 | |
|     {
 | |
|     case GTK_WIN32_SIZE:
 | |
|       {
 | |
|         const char *name = gtk_win32_get_sys_metric_name_for_id (value->val.size.id);
 | |
|         if (name)
 | |
|           g_string_append (string, name);
 | |
|         else
 | |
|           g_string_append_printf (string, ", %d", value->val.size.id);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_WIDTH:
 | |
|     case GTK_WIN32_PART_HEIGHT:
 | |
|     case GTK_WIN32_PART_BORDER_TOP:
 | |
|     case GTK_WIN32_PART_BORDER_RIGHT:
 | |
|     case GTK_WIN32_PART_BORDER_BOTTOM:
 | |
|     case GTK_WIN32_PART_BORDER_LEFT:
 | |
|       g_string_append_printf (string, ", %d, %d", value->val.part.part, value->val.part.state);
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       g_assert_not_reached ();
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|   g_string_append (string, ")");
 | |
| }
 | |
| 
 | |
| static double
 | |
| gtk_css_value_win32_size_get (const GtkCssValue *value,
 | |
|                               double             one_hundred_percent)
 | |
| {
 | |
|   return value->scale * gtk_css_value_win32_compute_size (value);
 | |
| }
 | |
| 
 | |
| static GtkCssDimension
 | |
| gtk_css_value_win32_size_get_dimension (const GtkCssValue *value)
 | |
| {
 | |
|   return GTK_CSS_DIMENSION_LENGTH;
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| gtk_css_value_win32_size_has_percent (const GtkCssValue *value)
 | |
| {
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| static GtkCssValue *
 | |
| gtk_css_value_win32_size_multiply (const GtkCssValue *value,
 | |
|                                    double             factor)
 | |
| {
 | |
|   GtkCssValue *result;
 | |
| 
 | |
|   result = gtk_css_win32_size_value_new (value->scale * factor, value->theme, value->type);
 | |
|   result->val = value->val;
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| static GtkCssValue *
 | |
| gtk_css_value_win32_size_try_add (const GtkCssValue *value1,
 | |
|                                   const GtkCssValue *value2)
 | |
| {
 | |
|   GtkCssValue *result;
 | |
| 
 | |
|   if (!gtk_css_value_win32_size_equal (value1, value2))
 | |
|     return NULL;
 | |
| 
 | |
|   result = gtk_css_win32_size_value_new (value1->scale + value2->scale, value1->theme, value1->type);
 | |
|   result->val = value1->val;
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| static gint
 | |
| gtk_css_value_win32_size_get_calc_term_order (const GtkCssValue *value)
 | |
| {
 | |
|   return 2000 + 100 * value->type;
 | |
| }
 | |
| 
 | |
| static const GtkCssNumberValueClass GTK_CSS_VALUE_WIN32_SIZE = {
 | |
|   {
 | |
|     gtk_css_value_win32_size_free,
 | |
|     gtk_css_value_win32_size_compute,
 | |
|     gtk_css_value_win32_size_equal,
 | |
|     gtk_css_number_value_transition,
 | |
|     gtk_css_value_win32_size_print
 | |
|   },
 | |
|   gtk_css_value_win32_size_get,
 | |
|   gtk_css_value_win32_size_get_dimension,
 | |
|   gtk_css_value_win32_size_has_percent,
 | |
|   gtk_css_value_win32_size_multiply,
 | |
|   gtk_css_value_win32_size_try_add,
 | |
|   gtk_css_value_win32_size_get_calc_term_order
 | |
| };
 | |
| 
 | |
| static GtkCssValue *
 | |
| gtk_css_win32_size_value_new (double            scale,
 | |
|                               GtkWin32Theme    *theme,
 | |
|                               GtkWin32SizeType  type)
 | |
| {
 | |
|   GtkCssValue *result;
 | |
| 
 | |
|   result = _gtk_css_value_new (GtkCssValue, >K_CSS_VALUE_WIN32_SIZE.value_class);
 | |
|   result->scale = scale;
 | |
|   result->theme = gtk_win32_theme_ref (theme);
 | |
|   result->type = type;
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| static GtkCssValue *
 | |
| gtk_css_win32_size_value_parse_size (GtkCssValue *value,
 | |
|                                      GtkCssParser *parser)
 | |
| {
 | |
|   char *name;
 | |
| 
 | |
|   name = _gtk_css_parser_try_ident (parser, TRUE);
 | |
|   if (name)
 | |
|     {
 | |
|       value->val.size.id = gtk_win32_get_sys_metric_id_for_name (name);
 | |
|       if (value->val.size.id == -1)
 | |
|         {
 | |
|           _gtk_css_parser_error (parser, "'%s' is not a name for a win32 metric.", name);
 | |
|           _gtk_css_value_unref (value);
 | |
|           g_free (name);
 | |
|           return NULL;
 | |
|         }
 | |
|       g_free (name);
 | |
|     }
 | |
|   else if (!_gtk_css_parser_try_int (parser, &value->val.size.id))
 | |
|     {
 | |
|       _gtk_css_value_unref (value);
 | |
|       _gtk_css_parser_error (parser, "Expected an integer ID");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   return value;
 | |
| }
 | |
| 
 | |
| static GtkCssValue *
 | |
| gtk_css_win32_size_value_parse_part_size (GtkCssValue *value,
 | |
|                                           GtkCssParser *parser)
 | |
| {
 | |
|   if (!_gtk_css_parser_try_int (parser, &value->val.part.part))
 | |
|     {
 | |
|       _gtk_css_value_unref (value);
 | |
|       _gtk_css_parser_error (parser, "Expected an integer part ID");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   if (! _gtk_css_parser_try (parser, ",", TRUE))
 | |
|     {
 | |
|       _gtk_css_value_unref (value);
 | |
|       _gtk_css_parser_error (parser, "Expected ','");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   if (!_gtk_css_parser_try_int (parser, &value->val.part.state))
 | |
|     {
 | |
|       _gtk_css_value_unref (value);
 | |
|       _gtk_css_parser_error (parser, "Expected an integer state ID");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   return value;
 | |
| }
 | |
| 
 | |
| GtkCssValue *
 | |
| gtk_css_win32_size_value_parse (GtkCssParser           *parser,
 | |
|                                 GtkCssNumberParseFlags  flags)
 | |
| {
 | |
|   GtkWin32Theme *theme;
 | |
|   GtkCssValue *result;
 | |
|   guint type;
 | |
| 
 | |
|   for (type = 0; type < G_N_ELEMENTS(css_value_names); type++)
 | |
|     {
 | |
|       if (_gtk_css_parser_try (parser, css_value_names[type], TRUE))
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|   if (type >= G_N_ELEMENTS(css_value_names))
 | |
|     {
 | |
|       _gtk_css_parser_error (parser, "Not a win32 size value");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   theme = gtk_win32_theme_parse (parser);
 | |
|   if (theme == NULL)
 | |
|     return NULL;
 | |
| 
 | |
|   result = gtk_css_win32_size_value_new (1.0, theme, type);
 | |
|   gtk_win32_theme_unref (theme);
 | |
| 
 | |
|   if (! _gtk_css_parser_try (parser, ",", TRUE))
 | |
|     {
 | |
|       _gtk_css_value_unref (result);
 | |
|       _gtk_css_parser_error (parser, "Expected ','");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   switch (result->type)
 | |
|     {
 | |
|     case GTK_WIN32_SIZE:
 | |
|       result = gtk_css_win32_size_value_parse_size (result, parser);
 | |
|       break;
 | |
| 
 | |
|     case GTK_WIN32_PART_WIDTH:
 | |
|     case GTK_WIN32_PART_HEIGHT:
 | |
|     case GTK_WIN32_PART_BORDER_TOP:
 | |
|     case GTK_WIN32_PART_BORDER_RIGHT:
 | |
|     case GTK_WIN32_PART_BORDER_BOTTOM:
 | |
|     case GTK_WIN32_PART_BORDER_LEFT:
 | |
|       result = gtk_css_win32_size_value_parse_part_size (result, parser);
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       g_assert_not_reached ();
 | |
|       _gtk_css_value_unref (result);
 | |
|       result = NULL;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|   if (result == NULL)
 | |
|     return NULL;
 | |
| 
 | |
|   if (!_gtk_css_parser_try (parser, ")", TRUE))
 | |
|     {
 | |
|       _gtk_css_value_unref (result);
 | |
|       _gtk_css_parser_error (parser, "Expected ')'");
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|   return result;
 | |
| }
 | 
