mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-13 22:10:08 +00:00
64e5c76323
gdk_texture_save_to_png_bytes() cannot fail, so ensure that it doesn't. Testsuite has been updated to check for this case. Note that we do not load the PNG file that we generate here. Loading is a lot more scary than saving after all. If people want to load oversized PNG files, they should use a real PNG loader.
463 lines
13 KiB
C
463 lines
13 KiB
C
#include <gtk.h>
|
|
|
|
#include "gdk/gdkmemorytextureprivate.h"
|
|
#include "gdk/gdktextureprivate.h"
|
|
|
|
|
|
#define assert_texture_diff_equal(a, b, expected) G_STMT_START { \
|
|
cairo_region_t *_r; \
|
|
\
|
|
_r = cairo_region_create (); \
|
|
gdk_texture_diff (a, b, _r); \
|
|
g_assert_true (cairo_region_equal (_r, expected)); \
|
|
cairo_region_destroy(_r); \
|
|
\
|
|
_r = cairo_region_create (); \
|
|
gdk_texture_diff (b, a, _r); \
|
|
g_assert_true (cairo_region_equal (_r, expected)); \
|
|
cairo_region_destroy(_r); \
|
|
}G_STMT_END
|
|
|
|
static void
|
|
compare_pixels (int width,
|
|
int height,
|
|
guchar *data1,
|
|
gsize stride1,
|
|
guchar *data2,
|
|
gsize stride2)
|
|
{
|
|
int x, y;
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
const guint32 *p1 = (const guint32*) (data1 + y * stride1);
|
|
const guint32 *p2 = (const guint32*) (data2 + y * stride2);
|
|
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
g_assert_cmphex (p1[x], ==, p2[x]);
|
|
}
|
|
}
|
|
}
|
|
|
|
static GdkTexture *
|
|
red_texture_new (int width,
|
|
int height)
|
|
{
|
|
/* We use GDK_MEMORY_R8G8B8A8 here because it's the expected PNG RGBA format */
|
|
const GdkMemoryFormat format = GDK_MEMORY_R8G8B8A8;
|
|
int i;
|
|
guint32 *data;
|
|
GBytes *bytes;
|
|
GdkTexture *texture;
|
|
|
|
data = g_malloc (width * height * 4);
|
|
|
|
for (i = 0; i < width * height; i++)
|
|
data[i] = 0xFF0000FF;
|
|
|
|
bytes = g_bytes_new_take ((guint8 *) data, width * height * 4);
|
|
texture = gdk_memory_texture_new (width, height, format, bytes, width * 4);
|
|
g_bytes_unref (bytes);
|
|
|
|
return texture;
|
|
}
|
|
|
|
static void
|
|
compare_textures (GdkTexture *texture1,
|
|
GdkTexture *texture2)
|
|
{
|
|
int width, height;
|
|
gsize stride;
|
|
guchar *data1;
|
|
guchar *data2;
|
|
|
|
g_assert_true (gdk_texture_get_width (texture1) == gdk_texture_get_width (texture2));
|
|
g_assert_true (gdk_texture_get_height (texture1) == gdk_texture_get_height (texture2));
|
|
|
|
width = gdk_texture_get_width (texture1);
|
|
height = gdk_texture_get_height (texture1);
|
|
stride = 4 * width;
|
|
|
|
data1 = g_new0 (guchar, stride * height);
|
|
gdk_texture_download (texture1, data1, stride);
|
|
|
|
data2 = g_new0 (guchar, stride * height);
|
|
gdk_texture_download (texture2, data2, stride);
|
|
|
|
compare_pixels (width, height,
|
|
data1, stride,
|
|
data2, stride);
|
|
|
|
g_free (data1);
|
|
g_free (data2);
|
|
}
|
|
|
|
static void
|
|
test_texture_from_pixbuf (void)
|
|
{
|
|
GdkPixbuf *pixbuf;
|
|
GdkTexture *texture;
|
|
GError *error = NULL;
|
|
guchar *data;
|
|
int width, height;
|
|
gsize stride;
|
|
cairo_surface_t *surface;
|
|
cairo_t *cr;
|
|
|
|
pixbuf = gdk_pixbuf_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png", &error);
|
|
g_assert_no_error (error);
|
|
g_assert_nonnull (pixbuf);
|
|
g_assert_true (gdk_pixbuf_get_has_alpha (pixbuf));
|
|
|
|
width = gdk_pixbuf_get_width (pixbuf);
|
|
height = gdk_pixbuf_get_height (pixbuf);
|
|
|
|
texture = gdk_texture_new_for_pixbuf (pixbuf);
|
|
|
|
g_assert_nonnull (texture);
|
|
g_assert_cmpint (gdk_texture_get_width (texture), ==, width);
|
|
g_assert_cmpint (gdk_texture_get_height (texture), ==, height);
|
|
|
|
stride = 4 * width;
|
|
data = g_new0 (guchar, stride * height);
|
|
gdk_texture_download (texture, data, stride);
|
|
|
|
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
|
|
cr = cairo_create (surface);
|
|
gdk_cairo_set_source_pixbuf (cr, pixbuf, 0, 0);
|
|
cairo_paint (cr);
|
|
cairo_destroy (cr);
|
|
|
|
compare_pixels (width, height,
|
|
data, stride,
|
|
cairo_image_surface_get_data (surface),
|
|
cairo_image_surface_get_stride (surface));
|
|
|
|
g_free (data);
|
|
|
|
g_object_unref (pixbuf);
|
|
g_object_unref (texture);
|
|
cairo_surface_destroy (surface);
|
|
}
|
|
|
|
static void
|
|
test_texture_from_resource (void)
|
|
{
|
|
GdkTexture *texture;
|
|
int width, height;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
g_assert_nonnull (texture);
|
|
g_object_get (texture,
|
|
"width", &width,
|
|
"height", &height,
|
|
NULL);
|
|
g_assert_cmpint (width, ==, 16);
|
|
g_assert_cmpint (height, ==, 16);
|
|
|
|
g_object_unref (texture);
|
|
}
|
|
|
|
static void
|
|
test_texture_save_to_png (void)
|
|
{
|
|
GdkTexture *texture;
|
|
GError *error = NULL;
|
|
GFile *file;
|
|
GdkTexture *texture2;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
g_assert_true (gdk_texture_save_to_png (texture, "test.png"));
|
|
file = g_file_new_for_path ("test.png");
|
|
texture2 = gdk_texture_new_from_file (file, &error);
|
|
g_object_unref (file);
|
|
g_assert_no_error (error);
|
|
|
|
compare_textures (texture, texture2);
|
|
|
|
g_object_unref (texture);
|
|
g_object_unref (texture2);
|
|
|
|
/* libpng has a builtin user limit of 1M pixels per dimension, so we make it 1px too big. */
|
|
texture = red_texture_new (1 * 1000 * 1000 + 1, 1);
|
|
g_assert_true (gdk_texture_save_to_png (texture, "test.png"));
|
|
}
|
|
|
|
static void
|
|
test_texture_save_to_tiff (void)
|
|
{
|
|
GdkTexture *texture;
|
|
GError *error = NULL;
|
|
GFile *file;
|
|
GdkTexture *texture2;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
gdk_texture_save_to_tiff (texture, "test.tiff");
|
|
file = g_file_new_for_path ("test.tiff");
|
|
texture2 = gdk_texture_new_from_file (file, &error);
|
|
g_object_unref (file);
|
|
g_assert_no_error (error);
|
|
|
|
compare_textures (texture, texture2);
|
|
|
|
g_object_unref (texture);
|
|
g_object_unref (texture2);
|
|
}
|
|
|
|
static void
|
|
test_texture_subtexture (void)
|
|
{
|
|
cairo_t *cr;
|
|
cairo_surface_t *surface;
|
|
cairo_pattern_t *pattern;
|
|
GdkTexture *texture, *subtexture;
|
|
guchar *data, *subdata;
|
|
gsize stride, substride;
|
|
|
|
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, 64, 64);
|
|
cr = cairo_create (surface);
|
|
|
|
pattern = cairo_pattern_create_linear (0, 0, 64, 64);
|
|
cairo_pattern_add_color_stop_rgba (pattern, 0, 1, 0, 0, 0.2);
|
|
cairo_pattern_add_color_stop_rgba (pattern, 1, 1, 0, 1, 1);
|
|
|
|
cairo_set_source (cr, pattern);
|
|
cairo_paint (cr);
|
|
|
|
texture = gdk_texture_new_for_surface (surface);
|
|
|
|
cairo_pattern_destroy (pattern);
|
|
cairo_destroy (cr);
|
|
cairo_surface_destroy (surface);
|
|
|
|
subtexture = gdk_memory_texture_new_subtexture (GDK_MEMORY_TEXTURE (texture), 0, 0, 32, 32);
|
|
|
|
g_assert_cmpint (gdk_texture_get_width (subtexture), ==, 32);
|
|
g_assert_cmpint (gdk_texture_get_height (subtexture), ==, 32);
|
|
|
|
data = g_new0 (guchar, 64 * 64 * 4);
|
|
stride = 64 * 4;
|
|
gdk_texture_download (texture, data, stride);
|
|
|
|
subdata = g_new0 (guchar, 32 * 32 * 4);
|
|
substride = 32 * 4;
|
|
gdk_texture_download (subtexture, subdata, substride);
|
|
|
|
compare_pixels (32, 32,
|
|
data, stride,
|
|
subdata, substride);
|
|
|
|
g_free (data);
|
|
g_free (subdata);
|
|
|
|
g_object_unref (subtexture);
|
|
g_object_unref (texture);
|
|
}
|
|
|
|
static void
|
|
test_texture_icon (void)
|
|
{
|
|
GdkTexture *texture;
|
|
GdkTexture *texture2;
|
|
GInputStream *stream;
|
|
GdkPixbuf *pixbuf;
|
|
GError *error = NULL;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
stream = g_loadable_icon_load (G_LOADABLE_ICON (texture), 16, NULL, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert_nonnull (stream);
|
|
|
|
pixbuf = gdk_pixbuf_new_from_stream (stream, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert_nonnull (pixbuf);
|
|
|
|
texture2 = gdk_texture_new_for_pixbuf (pixbuf);
|
|
|
|
compare_textures (texture, texture2);
|
|
|
|
g_object_unref (texture2);
|
|
g_object_unref (pixbuf);
|
|
g_object_unref (stream);
|
|
g_object_unref (texture);
|
|
}
|
|
|
|
static void
|
|
icon_loaded (GObject *source,
|
|
GAsyncResult *result,
|
|
gpointer data)
|
|
{
|
|
GdkTexture *texture = GDK_TEXTURE (source);
|
|
GError *error = NULL;
|
|
GdkTexture *texture2;
|
|
GdkPixbuf *pixbuf;
|
|
GInputStream *stream;
|
|
gboolean *done = (gboolean *)data;
|
|
|
|
stream = g_loadable_icon_load_finish (G_LOADABLE_ICON (texture), result, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert_nonnull (stream);
|
|
|
|
pixbuf = gdk_pixbuf_new_from_stream (stream, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert_nonnull (pixbuf);
|
|
|
|
texture2 = gdk_texture_new_for_pixbuf (pixbuf);
|
|
|
|
compare_textures (texture, texture2);
|
|
|
|
g_object_unref (texture2);
|
|
g_object_unref (pixbuf);
|
|
g_object_unref (stream);
|
|
g_object_unref (texture);
|
|
|
|
*done = TRUE;
|
|
}
|
|
|
|
static void
|
|
test_texture_icon_async (void)
|
|
{
|
|
GdkTexture *texture;
|
|
gboolean done = FALSE;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
g_loadable_icon_load_async (G_LOADABLE_ICON (texture), 16, NULL, icon_loaded, &done);
|
|
|
|
while (!done)
|
|
g_main_context_iteration (NULL, FALSE);
|
|
}
|
|
|
|
static void
|
|
test_texture_icon_serialize (void)
|
|
{
|
|
GdkTexture *texture;
|
|
GVariant *data;
|
|
GIcon *icon;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
data = g_icon_serialize (G_ICON (texture));
|
|
g_assert_nonnull (data);
|
|
icon = g_icon_deserialize (data);
|
|
g_assert_true (G_IS_BYTES_ICON (icon));
|
|
|
|
g_variant_unref (data);
|
|
g_object_unref (icon);
|
|
g_object_unref (texture);
|
|
}
|
|
|
|
static void
|
|
test_texture_diff (void)
|
|
{
|
|
GdkTexture *texture0;
|
|
GdkTexture *texture;
|
|
GdkTexture *texture2;
|
|
cairo_region_t *empty;
|
|
cairo_region_t *full;
|
|
cairo_region_t *center;
|
|
cairo_region_t *left;
|
|
cairo_region_t *left_center;
|
|
|
|
texture0 = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
texture2 = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
empty = cairo_region_create();
|
|
full = cairo_region_create_rectangle (&(cairo_rectangle_int_t) { 0, 0, 16, 16 });
|
|
center = cairo_region_create_rectangle (&(cairo_rectangle_int_t) { 4, 4, 8 ,8 });
|
|
left = cairo_region_create_rectangle (&(cairo_rectangle_int_t) { 0, 4, 4, 4 });
|
|
left_center = cairo_region_copy (left);
|
|
cairo_region_union (left_center, center);
|
|
|
|
assert_texture_diff_equal (texture, texture, empty);
|
|
|
|
/* No diff set, so we get the full area */
|
|
assert_texture_diff_equal (texture, texture2, full);
|
|
|
|
gdk_texture_set_diff (texture, texture2, cairo_region_copy (center));
|
|
|
|
assert_texture_diff_equal (texture, texture2, center);
|
|
|
|
gdk_texture_set_diff (texture0, texture, cairo_region_copy (left));
|
|
|
|
assert_texture_diff_equal (texture0, texture2, left_center);
|
|
|
|
g_object_unref (texture);
|
|
|
|
assert_texture_diff_equal (texture0, texture2, left_center);
|
|
|
|
cairo_region_destroy (full);
|
|
cairo_region_destroy (center);
|
|
cairo_region_destroy (left);
|
|
cairo_region_destroy (left_center);
|
|
g_object_unref (texture2);
|
|
g_object_unref (texture0);
|
|
}
|
|
|
|
static void
|
|
test_texture_downloader (void)
|
|
{
|
|
GdkTexture *texture;
|
|
GdkTexture *texture2;
|
|
GdkTextureDownloader *downloader;
|
|
GdkTextureDownloader *downloader2;
|
|
gsize stride;
|
|
GBytes *bytes;
|
|
guchar *data;
|
|
|
|
texture = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
texture2 = gdk_texture_new_from_resource ("/org/gtk/libgtk/icons/16x16/places/user-trash.png");
|
|
|
|
downloader = gdk_texture_downloader_new (texture);
|
|
|
|
downloader2 = gdk_texture_downloader_copy (downloader);
|
|
g_assert_true (gdk_texture_downloader_get_texture (downloader2) == texture);
|
|
gdk_texture_downloader_free (downloader2);
|
|
|
|
gdk_texture_downloader_set_texture (downloader, texture2);
|
|
gdk_texture_downloader_set_format (downloader, GDK_MEMORY_R16G16B16A16);
|
|
g_assert_true (gdk_texture_downloader_get_format (downloader) == GDK_MEMORY_R16G16B16A16);
|
|
|
|
bytes = gdk_texture_downloader_download_bytes (downloader, &stride);
|
|
|
|
g_assert_true (stride == 4 * 2 * 16);
|
|
g_assert_true (g_bytes_get_size (bytes) == stride * 16);
|
|
|
|
data = g_malloc (stride * 16);
|
|
gdk_texture_downloader_download_into (downloader, data, stride);
|
|
|
|
g_assert_true (memcmp (data, g_bytes_get_data (bytes, NULL), stride * 16) == 0);
|
|
|
|
g_free (data);
|
|
g_bytes_unref (bytes);
|
|
gdk_texture_downloader_free (downloader);
|
|
|
|
g_object_unref (texture2);
|
|
g_object_unref (texture);
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
/* We want to use resources from libgtk, so we need gtk initialized */
|
|
gtk_test_init (&argc, &argv, NULL);
|
|
|
|
g_test_add_func ("/texture/from-pixbuf", test_texture_from_pixbuf);
|
|
g_test_add_func ("/texture/from-resource", test_texture_from_resource);
|
|
g_test_add_func ("/texture/save-to-png", test_texture_save_to_png);
|
|
g_test_add_func ("/texture/save-to-tiff", test_texture_save_to_tiff);
|
|
g_test_add_func ("/texture/subtexture", test_texture_subtexture);
|
|
g_test_add_func ("/texture/icon/load", test_texture_icon);
|
|
g_test_add_func ("/texture/icon/load-async", test_texture_icon_async);
|
|
g_test_add_func ("/texture/icon/serialize", test_texture_icon_serialize);
|
|
g_test_add_func ("/texture/diff", test_texture_diff);
|
|
g_test_add_func ("/texture/downloader", test_texture_downloader);
|
|
|
|
return g_test_run ();
|
|
}
|