gtkcssimageurl.c 5.84 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/*
 * Copyright © 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.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
Javier Jardón's avatar
Javier Jardón committed
15
 * License along with this library. If not, see <http://www.gnu.org/licenses/>.
16 17 18 19 20 21
 *
 * Authors: Benjamin Otte <otte@gnome.org>
 */

#include "config.h"

22 23
#include <string.h>

24
#include "gtkcssimageurlprivate.h"
25
#include "gtkcssimagesurfaceprivate.h"
26
#include "gtkstyleproviderprivate.h"
27 28 29

G_DEFINE_TYPE (GtkCssImageUrl, _gtk_css_image_url, GTK_TYPE_CSS_IMAGE)

Benjamin Otte's avatar
Benjamin Otte committed
30
static GtkCssImage *
31 32
gtk_css_image_url_load_image (GtkCssImageUrl  *url,
                              GError         **error)
Benjamin Otte's avatar
Benjamin Otte committed
33
{
34
  GdkPixbuf *pixbuf;
35
  GError *local_error = NULL;
36 37 38 39 40 41 42 43 44 45 46 47 48
  GFileInputStream *input;

  if (url->loaded_image)
    return url->loaded_image;

  /* We special case resources here so we can use
     gdk_pixbuf_new_from_resource, which in turn has some special casing
     for GdkPixdata files to avoid duplicating the memory for the pixbufs */
  if (g_file_has_uri_scheme (url->file, "resource"))
    {
      char *uri = g_file_get_uri (url->file);
      char *resource_path = g_uri_unescape_string (uri + strlen ("resource://"), NULL);

49
      pixbuf = gdk_pixbuf_new_from_resource (resource_path, &local_error);
50 51 52 53 54
      g_free (resource_path);
      g_free (uri);
    }
  else
    {
55
      input = g_file_read (url->file, NULL, &local_error);
56 57
      if (input != NULL)
	{
58
          pixbuf = gdk_pixbuf_new_from_stream (G_INPUT_STREAM (input), NULL, &local_error);
59 60 61 62 63 64 65 66 67 68
          g_object_unref (input);
	}
      else
        {
          pixbuf = NULL;
        }
    }

  if (pixbuf == NULL)
    {
69
      cairo_surface_t *empty;
70

71 72 73 74 75 76 77 78 79 80 81 82 83 84
      if (error)
        {
          char *uri;

          uri = g_file_get_uri (url->file);
          g_set_error (error,
                       GTK_CSS_PROVIDER_ERROR,
                       GTK_CSS_PROVIDER_ERROR_FAILED,
                       "Error loading image '%s': %s", uri, local_error->message);
          g_error_free (local_error);
          g_free (uri);
       }

      empty = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, 0, 0);
85 86
      url->loaded_image = _gtk_css_image_surface_new (empty);
      cairo_surface_destroy (empty);
87
      return url->loaded_image;
88 89 90 91 92
    }

  url->loaded_image = _gtk_css_image_surface_new_for_pixbuf (pixbuf);
  g_object_unref (pixbuf);

Benjamin Otte's avatar
Benjamin Otte committed
93 94 95
  return url->loaded_image;
}

96 97 98 99 100
static int
gtk_css_image_url_get_width (GtkCssImage *image)
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);

101
  return _gtk_css_image_get_width (gtk_css_image_url_load_image (url, NULL));
102 103 104 105 106 107 108
}

static int
gtk_css_image_url_get_height (GtkCssImage *image)
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);

109
  return _gtk_css_image_get_height (gtk_css_image_url_load_image (url, NULL));
110 111 112 113 114 115 116
}

static double
gtk_css_image_url_get_aspect_ratio (GtkCssImage *image)
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);

117
  return _gtk_css_image_get_aspect_ratio (gtk_css_image_url_load_image (url, NULL));
118 119 120 121 122 123 124 125 126 127
}

static void
gtk_css_image_url_draw (GtkCssImage        *image,
                        cairo_t            *cr,
                        double              width,
                        double              height)
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);

128
  _gtk_css_image_draw (gtk_css_image_url_load_image (url, NULL), cr, width, height);
129 130
}

131 132 133 134
static GtkCssImage *
gtk_css_image_url_compute (GtkCssImage             *image,
                           guint                    property_id,
                           GtkStyleProviderPrivate *provider,
135
                           GtkCssStyle             *style,
136
                           GtkCssStyle             *parent_style)
137 138
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);
139 140 141 142 143 144 145 146 147 148
  GtkCssImage *copy;
  GError *error = NULL;

  copy = gtk_css_image_url_load_image (url, &error);
  if (error)
    {
      GtkCssSection *section = gtk_css_style_get_section (style, property_id);
      _gtk_style_provider_private_emit_error (provider, section, error);
      g_error_free (error);
    }
149

150
  return g_object_ref (copy);
151 152
}

153 154
static gboolean
gtk_css_image_url_parse (GtkCssImage  *image,
155
                         GtkCssParser *parser)
156 157 158
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);

159 160
  url->file = _gtk_css_parser_read_url (parser);
  if (url->file == NULL)
161 162 163 164 165 166 167 168 169 170
    return FALSE;

  return TRUE;
}

static void
gtk_css_image_url_print (GtkCssImage *image,
                         GString     *string)
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (image);
171

172
  _gtk_css_image_print (gtk_css_image_url_load_image (url, NULL), string);
173 174 175 176 177 178 179
}

static void
gtk_css_image_url_dispose (GObject *object)
{
  GtkCssImageUrl *url = GTK_CSS_IMAGE_URL (object);

180
  g_clear_object (&url->file);
181
  g_clear_object (&url->loaded_image);
182 183 184 185 186 187 188 189 190 191 192 193

  G_OBJECT_CLASS (_gtk_css_image_url_parent_class)->dispose (object);
}

static void
_gtk_css_image_url_class_init (GtkCssImageUrlClass *klass)
{
  GtkCssImageClass *image_class = GTK_CSS_IMAGE_CLASS (klass);
  GObjectClass *object_class = G_OBJECT_CLASS (klass);

  image_class->get_width = gtk_css_image_url_get_width;
  image_class->get_height = gtk_css_image_url_get_height;
194
  image_class->get_aspect_ratio = gtk_css_image_url_get_aspect_ratio;
195
  image_class->compute = gtk_css_image_url_compute;
196 197 198 199 200 201 202 203 204 205 206 207
  image_class->draw = gtk_css_image_url_draw;
  image_class->parse = gtk_css_image_url_parse;
  image_class->print = gtk_css_image_url_print;

  object_class->dispose = gtk_css_image_url_dispose;
}

static void
_gtk_css_image_url_init (GtkCssImageUrl *image_url)
{
}