/* * Copyright © 2013 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: Alexander Larsson */ #include "config.h" #include "gtkcssimagescaledprivate.h" #include "gtkstyleproviderprivate.h" G_DEFINE_TYPE (GtkCssImageScaled, _gtk_css_image_scaled, GTK_TYPE_CSS_IMAGE) static int gtk_css_image_scaled_get_width (GtkCssImage *image) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (image); return _gtk_css_image_get_width (scaled->images[0])/scaled->scales[0]; } static int gtk_css_image_scaled_get_height (GtkCssImage *image) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (image); return _gtk_css_image_get_height (scaled->images[0])/scaled->scales[0]; } static double gtk_css_image_scaled_get_aspect_ratio (GtkCssImage *image) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (image); return _gtk_css_image_get_aspect_ratio (scaled->images[0]); } static void gtk_css_image_scaled_snapshot (GtkCssImage *image, GtkSnapshot *snapshot, double width, double height) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (image); gtk_css_image_snapshot (scaled->images[0], snapshot, width, height); // FIXME apply scale } static void gtk_css_image_scaled_print (GtkCssImage *image, GString *string) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (image); int i; g_string_append (string, "-gtk-scaled("); for (i = 0; i < scaled->n_images; i++) { _gtk_css_image_print (scaled->images[i], string); g_string_append_printf (string, ",%d", scaled->scales[i]); if (i != scaled->n_images - 1) g_string_append (string, ","); } g_string_append (string, ")"); } static void gtk_css_image_scaled_dispose (GObject *object) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (object); int i; for (i = 0; i < scaled->n_images; i++) g_object_unref (scaled->images[i]); g_free (scaled->images); scaled->images = NULL; g_free (scaled->scales); scaled->scales = NULL; G_OBJECT_CLASS (_gtk_css_image_scaled_parent_class)->dispose (object); } static GtkCssImage * gtk_css_image_scaled_compute (GtkCssImage *image, guint property_id, GtkStyleProvider *provider, GtkCssStyle *style, GtkCssStyle *parent_style) { GtkCssImageScaled *scaled = GTK_CSS_IMAGE_SCALED (image); int scale; GtkCssImageScaled *res; int i; int best; scale = gtk_style_provider_get_scale (provider); scale = MAX(scale, 1); best = 0; for (i = 0; i < scaled->n_images; i++) { if (scaled->scales[i] == scale) { best = i; break; } else if ((scaled->scales[best] < scaled->scales[i] && scaled->scales[i] < scale) || (scale < scaled->scales[i] && scaled->scales[i] < scaled->scales[best]) || (scaled->scales[best] < scale && scaled->scales[i] > scale)) { best = i; } } res = g_object_new (GTK_TYPE_CSS_IMAGE_SCALED, NULL); res->n_images = 1; res->images = g_new (GtkCssImage *, 1); res->scales = g_new (int, 1); res->images[0] = _gtk_css_image_compute (scaled->images[best], property_id, provider, style, parent_style); res->scales[0] = scaled->scales[best]; return GTK_CSS_IMAGE (res); } typedef struct { GPtrArray *images; GArray *scales; } GtkCssImageScaledParseData; static guint gtk_css_image_scaled_parse_arg (GtkCssParser *parser, guint arg, gpointer data_) { GtkCssImageScaledParseData *data = data_; GtkCssImage *child; int scale; child = _gtk_css_image_new_parse (parser); if (child == NULL) return 0; if (!gtk_css_parser_has_integer (parser)) scale = arg > 0 ? g_array_index (data->scales, int, arg - 1) + 1 : 1; else if (!gtk_css_parser_consume_integer (parser, &scale)) return 0; g_ptr_array_add (data->images, child); g_array_append_val (data->scales, scale); return 1; } static gboolean gtk_css_image_scaled_parse (GtkCssImage *image, GtkCssParser *parser) { GtkCssImageScaled *self = GTK_CSS_IMAGE_SCALED (image); GtkCssImageScaledParseData data; if (!gtk_css_parser_has_function (parser, "-gtk-scaled")) { _gtk_css_parser_error (parser, "Expected '-gtk-scaled('"); return FALSE; } data.images = g_ptr_array_new_with_free_func (g_object_unref); data.scales = g_array_new (FALSE, FALSE, sizeof (int)); if (!gtk_css_parser_consume_function (parser, 1, G_MAXUINT, gtk_css_image_scaled_parse_arg, &data)) { g_ptr_array_unref (data.images); g_array_unref (data.scales); return FALSE; } self->n_images = data.images->len; self->images = (GtkCssImage **) g_ptr_array_free (data.images, FALSE); self->scales = (int *) g_array_free (data.scales, FALSE); return TRUE; } static void _gtk_css_image_scaled_class_init (GtkCssImageScaledClass *klass) { GtkCssImageClass *image_class = GTK_CSS_IMAGE_CLASS (klass); GObjectClass *object_class = G_OBJECT_CLASS (klass); image_class->get_width = gtk_css_image_scaled_get_width; image_class->get_height = gtk_css_image_scaled_get_height; image_class->get_aspect_ratio = gtk_css_image_scaled_get_aspect_ratio; image_class->snapshot = gtk_css_image_scaled_snapshot; image_class->parse = gtk_css_image_scaled_parse; image_class->compute = gtk_css_image_scaled_compute; image_class->print = gtk_css_image_scaled_print; object_class->dispose = gtk_css_image_scaled_dispose; } static void _gtk_css_image_scaled_init (GtkCssImageScaled *image_scaled) { }