summaryrefslogtreecommitdiff
path: root/gtk/gtkcssbgsizevalue.c
diff options
context:
space:
mode:
authorBenjamin Otte <otte@redhat.com>2012-04-04 12:39:55 +0200
committerBenjamin Otte <otte@redhat.com>2012-04-17 08:59:18 +0200
commitcfc64627304916e1a1848230dbb426e7dc22251f (patch)
treed853cd9ec13cf6d8aee3d42bd7178c2befd1c2fd /gtk/gtkcssbgsizevalue.c
parentf785f7177c455369f1482ff6582cebf69ddddb7a (diff)
downloadgtk+-cfc64627304916e1a1848230dbb426e7dc22251f.tar.gz
cssvalue: Add a cssvalue for background-size
Diffstat (limited to 'gtk/gtkcssbgsizevalue.c')
-rw-r--r--gtk/gtkcssbgsizevalue.c258
1 files changed, 258 insertions, 0 deletions
diff --git a/gtk/gtkcssbgsizevalue.c b/gtk/gtkcssbgsizevalue.c
new file mode 100644
index 0000000000..24e0fae039
--- /dev/null
+++ b/gtk/gtkcssbgsizevalue.c
@@ -0,0 +1,258 @@
+/* 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 "gtkcssbgsizevalueprivate.h"
+
+#include "gtkcssnumbervalueprivate.h"
+
+struct _GtkCssValue {
+ GTK_CSS_VALUE_BASE
+ guint cover :1;
+ guint contain :1;
+ GtkCssValue *x;
+ GtkCssValue *y;
+};
+
+static void
+gtk_css_value_bg_size_free (GtkCssValue *value)
+{
+ if (value->x)
+ _gtk_css_value_unref (value->x);
+ if (value->y)
+ _gtk_css_value_unref (value->y);
+
+ g_slice_free (GtkCssValue, value);
+}
+
+static gboolean
+gtk_css_value_bg_size_equal (const GtkCssValue *value1,
+ const GtkCssValue *value2)
+{
+ return value1->cover == value2->cover &&
+ value2->contain == value2->contain &&
+ (value1->x == value2->x ||
+ (value1->x != NULL && value2->x != NULL &&
+ _gtk_css_value_equal (value1->x, value2->x))) &&
+ (value1->y == value2->y ||
+ (value1->y != NULL && value2->y != NULL &&
+ _gtk_css_value_equal (value1->y, value2->y)));
+}
+
+static GtkCssValue *
+gtk_css_value_bg_size_transition (GtkCssValue *start,
+ GtkCssValue *end,
+ double progress)
+{
+ GtkCssValue *x, *y;
+
+ if (start->cover)
+ return end->cover ? _gtk_css_value_ref (end) : NULL;
+ if (start->contain)
+ return end->contain ? _gtk_css_value_ref (end) : NULL;
+
+ if ((start->x != NULL) ^ (end->x != NULL) ||
+ (start->y != NULL) ^ (end->y != NULL))
+ return NULL;
+
+ if (start->x)
+ {
+ x = _gtk_css_value_transition (start->x, end->x, progress);
+ if (x == NULL)
+ return NULL;
+ }
+ else
+ x = NULL;
+
+ if (start->y)
+ {
+ y = _gtk_css_value_transition (start->y, end->y, progress);
+ if (y == NULL)
+ {
+ _gtk_css_value_unref (x);
+ return NULL;
+ }
+ }
+ else
+ y = NULL;
+
+ return _gtk_css_bg_size_value_new (x, y);
+}
+
+static void
+gtk_css_value_bg_size_print (const GtkCssValue *value,
+ GString *string)
+{
+ if (value->cover)
+ g_string_append (string, "cover");
+ else if (value->contain)
+ g_string_append (string, "contain");
+ else
+ {
+ if (value->x == NULL)
+ g_string_append (string, "auto");
+ else
+ _gtk_css_value_print (value->x, string);
+
+ if (value->y)
+ {
+ g_string_append_c (string, ' ');
+ _gtk_css_value_print (value->y, string);
+ }
+ }
+}
+
+static const GtkCssValueClass GTK_CSS_VALUE_BG_SIZE = {
+ gtk_css_value_bg_size_free,
+ gtk_css_value_bg_size_equal,
+ gtk_css_value_bg_size_transition,
+ gtk_css_value_bg_size_print
+};
+
+static GtkCssValue auto_singleton = { &GTK_CSS_VALUE_BG_SIZE, 1, FALSE, FALSE, NULL, NULL };
+static GtkCssValue cover_singleton = { &GTK_CSS_VALUE_BG_SIZE, 1, TRUE, FALSE, NULL, NULL };
+static GtkCssValue contain_singleton = { &GTK_CSS_VALUE_BG_SIZE, 1, FALSE, TRUE, NULL, NULL };
+
+GtkCssValue *
+_gtk_css_bg_size_value_new (GtkCssValue *x,
+ GtkCssValue *y)
+{
+ GtkCssValue *result;
+
+ if (x == NULL && y == NULL)
+ return _gtk_css_value_ref (&auto_singleton);
+
+ result = _gtk_css_value_new (GtkCssValue, &GTK_CSS_VALUE_BG_SIZE);
+ result->x = x;
+ result->y = y;
+
+ return result;
+}
+
+GtkCssValue *
+_gtk_css_bg_size_value_parse (GtkCssParser *parser)
+{
+ GtkCssValue *x, *y;
+
+ if (_gtk_css_parser_try (parser, "cover", TRUE))
+ return _gtk_css_value_ref (&cover_singleton);
+ else if (_gtk_css_parser_try (parser, "contain", TRUE))
+ return _gtk_css_value_ref (&contain_singleton);
+
+ if (_gtk_css_parser_try (parser, "auto", TRUE))
+ x = NULL;
+ else
+ {
+ x = _gtk_css_number_value_parse (parser,
+ GTK_CSS_POSITIVE_ONLY
+ | GTK_CSS_PARSE_PERCENT
+ | GTK_CSS_PARSE_LENGTH);
+ if (x == NULL)
+ return NULL;
+ }
+
+ if (_gtk_css_parser_try (parser, "auto", TRUE))
+ y = NULL;
+ else if (!_gtk_css_parser_has_number (parser))
+ y = NULL;
+ else
+ {
+ y = _gtk_css_number_value_parse (parser,
+ GTK_CSS_POSITIVE_ONLY
+ | GTK_CSS_PARSE_PERCENT
+ | GTK_CSS_PARSE_LENGTH);
+ if (y == NULL)
+ {
+ _gtk_css_value_unref (x);
+ return NULL;
+ }
+ }
+
+ return _gtk_css_bg_size_value_new (x, y);
+}
+
+static void
+gtk_css_bg_size_compute_size_for_cover_contain (gboolean cover,
+ GtkCssImage *image,
+ double width,
+ double height,
+ double *concrete_width,
+ double *concrete_height)
+{
+ double aspect, image_aspect;
+
+ image_aspect = _gtk_css_image_get_aspect_ratio (image);
+ if (image_aspect == 0.0)
+ {
+ *concrete_width = width;
+ *concrete_height = height;
+ return;
+ }
+
+ aspect = width / height;
+
+ if ((aspect >= image_aspect && cover) ||
+ (aspect < image_aspect && !cover))
+ {
+ *concrete_width = width;
+ *concrete_height = width / image_aspect;
+ }
+ else
+ {
+ *concrete_height = height;
+ *concrete_width = height * image_aspect;
+ }
+}
+
+void
+_gtk_css_bg_size_value_compute_size (const GtkCssValue *value,
+ GtkCssImage *image,
+ double area_width,
+ double area_height,
+ double *out_width,
+ double *out_height)
+{
+ g_return_if_fail (value->class == &GTK_CSS_VALUE_BG_SIZE);
+
+ if (value->contain || value->cover)
+ gtk_css_bg_size_compute_size_for_cover_contain (value->cover,
+ image,
+ area_width, area_height,
+ out_width, out_height);
+ else
+ _gtk_css_image_get_concrete_size (image,
+ /* note: 0 does the right thing here for 'auto' */
+ value->x ? _gtk_css_number_value_get (value->x, area_width) : 0,
+ value->y ? _gtk_css_number_value_get (value->y, area_height) : 0,
+ area_width, area_height,
+ out_width, out_height);
+}
+
+GtkCssValue *
+_gtk_css_bg_size_value_compute (GtkCssValue *value,
+ GtkStyleContext *context)
+{
+ g_return_val_if_fail (value->class == &GTK_CSS_VALUE_BG_SIZE, NULL);
+
+ if (value->x == NULL && value->y == NULL)
+ return _gtk_css_value_ref (value);
+
+ return _gtk_css_bg_size_value_new (value->x ? _gtk_css_number_value_compute (value->x, context) : NULL,
+ value->y ? _gtk_css_number_value_compute (value->y, context) : NULL);
+}
+