2013-02-01 22:15:27 +00:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Authors: Benjamin Otte <otte@gnome.org>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "gtkcssimagesurfaceprivate.h"
|
2017-12-29 18:21:52 +00:00
|
|
|
#include <math.h>
|
2013-02-01 22:15:27 +00:00
|
|
|
|
|
|
|
G_DEFINE_TYPE (GtkCssImageSurface, _gtk_css_image_surface, GTK_TYPE_CSS_IMAGE)
|
|
|
|
|
|
|
|
static int
|
|
|
|
gtk_css_image_surface_get_width (GtkCssImage *image)
|
|
|
|
{
|
|
|
|
GtkCssImageSurface *surface = GTK_CSS_IMAGE_SURFACE (image);
|
|
|
|
|
|
|
|
return cairo_image_surface_get_width (surface->surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
gtk_css_image_surface_get_height (GtkCssImage *image)
|
|
|
|
{
|
|
|
|
GtkCssImageSurface *surface = GTK_CSS_IMAGE_SURFACE (image);
|
|
|
|
|
|
|
|
return cairo_image_surface_get_height (surface->surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_css_image_surface_draw (GtkCssImage *image,
|
|
|
|
cairo_t *cr,
|
|
|
|
double width,
|
|
|
|
double height)
|
|
|
|
{
|
|
|
|
GtkCssImageSurface *surface = GTK_CSS_IMAGE_SURFACE (image);
|
2015-12-15 20:23:59 +00:00
|
|
|
int image_width, image_height;
|
|
|
|
|
|
|
|
image_width = cairo_image_surface_get_width (surface->surface);
|
|
|
|
image_height = cairo_image_surface_get_height (surface->surface);
|
|
|
|
|
2017-12-29 18:21:52 +00:00
|
|
|
if (image_width == 0 || image_height == 0 || width <= 0 || height <= 0)
|
2015-12-15 20:23:59 +00:00
|
|
|
return;
|
2013-02-01 22:15:27 +00:00
|
|
|
|
2017-12-29 18:21:52 +00:00
|
|
|
/* Update cache image if size is different */
|
|
|
|
if (surface->cache == NULL ||
|
|
|
|
ABS (width - surface->width) > 0.001 ||
|
|
|
|
ABS (height - surface->height) > 0.001)
|
|
|
|
{
|
2018-03-05 17:47:00 +00:00
|
|
|
double xscale, yscale;
|
2017-12-29 18:21:52 +00:00
|
|
|
cairo_t *cache;
|
|
|
|
|
2018-03-05 17:47:00 +00:00
|
|
|
/* We need the device scale (HiDPI mode) to calculate the proper size in
|
|
|
|
* pixels for the image surface and set the cache device scale
|
|
|
|
*/
|
|
|
|
cairo_surface_get_device_scale (cairo_get_target (cr), &xscale, &yscale);
|
|
|
|
|
2017-12-29 18:21:52 +00:00
|
|
|
/* Save original size to preserve precision */
|
|
|
|
surface->width = width;
|
|
|
|
surface->height = height;
|
|
|
|
|
|
|
|
/* Destroy old cache if any */
|
|
|
|
g_clear_pointer (&surface->cache, cairo_surface_destroy);
|
|
|
|
|
|
|
|
/* Image big enough to contain scaled image with subpixel precision */
|
|
|
|
surface->cache = cairo_surface_create_similar_image (surface->surface,
|
|
|
|
CAIRO_FORMAT_ARGB32,
|
2018-03-05 17:47:00 +00:00
|
|
|
ceil (width*xscale),
|
|
|
|
ceil (height*yscale));
|
|
|
|
cairo_surface_set_device_scale (surface->cache, xscale, yscale);
|
2017-12-29 18:21:52 +00:00
|
|
|
cache = cairo_create (surface->cache);
|
|
|
|
cairo_rectangle (cache, 0, 0, width, height);
|
|
|
|
cairo_scale (cache, width / image_width, height / image_height);
|
|
|
|
cairo_set_source_surface (cache, surface->surface, 0, 0);
|
|
|
|
cairo_fill (cache);
|
|
|
|
|
|
|
|
cairo_destroy (cache);
|
|
|
|
}
|
|
|
|
|
2013-02-01 22:15:27 +00:00
|
|
|
cairo_rectangle (cr, 0, 0, width, height);
|
2017-12-29 18:21:52 +00:00
|
|
|
cairo_set_source_surface (cr, surface->cache ? surface->cache : surface->surface, 0, 0);
|
2013-02-01 22:15:27 +00:00
|
|
|
cairo_fill (cr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static cairo_status_t
|
|
|
|
surface_write (void *closure,
|
|
|
|
const unsigned char *data,
|
|
|
|
unsigned int length)
|
|
|
|
{
|
|
|
|
g_byte_array_append (closure, data, length);
|
|
|
|
|
|
|
|
return CAIRO_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_css_image_surface_print (GtkCssImage *image,
|
|
|
|
GString *string)
|
|
|
|
{
|
|
|
|
#if CAIRO_HAS_PNG_FUNCTIONS
|
|
|
|
GtkCssImageSurface *surface = GTK_CSS_IMAGE_SURFACE (image);
|
|
|
|
GByteArray *array;
|
|
|
|
char *base64;
|
|
|
|
|
|
|
|
array = g_byte_array_new ();
|
|
|
|
cairo_surface_write_to_png_stream (surface->surface, surface_write, array);
|
|
|
|
base64 = g_base64_encode (array->data, array->len);
|
|
|
|
g_byte_array_free (array, TRUE);
|
|
|
|
|
2013-02-04 13:31:13 +00:00
|
|
|
g_string_append (string, "url(\"data:image/png;base64,");
|
2013-02-01 22:15:27 +00:00
|
|
|
g_string_append (string, base64);
|
|
|
|
g_string_append (string, "\")");
|
|
|
|
|
|
|
|
g_free (base64);
|
|
|
|
#else
|
|
|
|
g_string_append (string, "none /* you need cairo png functions enabled to make this work */");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_css_image_surface_dispose (GObject *object)
|
|
|
|
{
|
|
|
|
GtkCssImageSurface *surface = GTK_CSS_IMAGE_SURFACE (object);
|
|
|
|
|
|
|
|
if (surface->surface)
|
|
|
|
{
|
|
|
|
cairo_surface_destroy (surface->surface);
|
|
|
|
surface->surface = NULL;
|
|
|
|
}
|
|
|
|
|
2020-01-13 22:24:47 +00:00
|
|
|
g_clear_pointer (&surface->cache, cairo_surface_destroy);
|
|
|
|
|
2013-02-01 22:15:27 +00:00
|
|
|
G_OBJECT_CLASS (_gtk_css_image_surface_parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_gtk_css_image_surface_class_init (GtkCssImageSurfaceClass *klass)
|
|
|
|
{
|
|
|
|
GtkCssImageClass *image_class = GTK_CSS_IMAGE_CLASS (klass);
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
image_class->get_width = gtk_css_image_surface_get_width;
|
|
|
|
image_class->get_height = gtk_css_image_surface_get_height;
|
|
|
|
image_class->draw = gtk_css_image_surface_draw;
|
|
|
|
image_class->print = gtk_css_image_surface_print;
|
|
|
|
|
|
|
|
object_class->dispose = gtk_css_image_surface_dispose;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_gtk_css_image_surface_init (GtkCssImageSurface *image_surface)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkCssImage *
|
|
|
|
_gtk_css_image_surface_new (cairo_surface_t *surface)
|
|
|
|
{
|
|
|
|
GtkCssImage *image;
|
|
|
|
|
|
|
|
g_return_val_if_fail (surface != NULL, NULL);
|
|
|
|
|
|
|
|
image = g_object_new (GTK_TYPE_CSS_IMAGE_SURFACE, NULL);
|
|
|
|
|
|
|
|
GTK_CSS_IMAGE_SURFACE (image)->surface = cairo_surface_reference (surface);
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkCssImage *
|
|
|
|
_gtk_css_image_surface_new_for_pixbuf (GdkPixbuf *pixbuf)
|
|
|
|
{
|
|
|
|
GtkCssImage *image;
|
|
|
|
cairo_surface_t *surface;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GDK_IS_PIXBUF (pixbuf), NULL);
|
|
|
|
|
2013-06-24 10:53:53 +00:00
|
|
|
surface = gdk_cairo_surface_create_from_pixbuf (pixbuf, 1, NULL);
|
2013-02-01 22:15:27 +00:00
|
|
|
|
|
|
|
image = _gtk_css_image_surface_new (surface);
|
|
|
|
cairo_surface_destroy (surface);
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|