gtk/testsuite/gdk/colorstate.c

231 lines
7.3 KiB
C
Raw Normal View History

#include <gdk/gdk.h>
#include <math.h>
2024-07-14 19:13:10 +00:00
static GdkColorState *
get_color_state (guint id,
const char **out_name)
{
const char *unused;
if (out_name == NULL)
out_name = &unused;
switch (id)
{
case 0:
*out_name = "srgb";
return gdk_color_state_ref (gdk_color_state_get_srgb ());
case 1:
*out_name = "srgb-linear";
return gdk_color_state_ref (gdk_color_state_get_srgb_linear ());
case 2:
*out_name = "rec2100-pq";
return gdk_color_state_ref (gdk_color_state_get_rec2100_pq ());
case 3:
*out_name = "rec2100-linear";
return gdk_color_state_ref (gdk_color_state_get_rec2100_linear ());
default:
return NULL;
}
}
static void
2024-07-14 19:13:10 +00:00
test_equal (void)
{
2024-07-14 19:13:10 +00:00
GdkColorState *csi, *csj;
guint i, j;
for (i = 0; ; i++)
{
csi = get_color_state (i, NULL);
if (csi == NULL)
break;
2024-07-14 19:13:10 +00:00
g_assert_true (gdk_color_state_equal (csi, csi));
2024-07-14 19:13:10 +00:00
for (j = 0; ; j++)
{
csj = get_color_state (j, NULL);
if (csj == NULL)
break;
if (i != j)
g_assert_false (gdk_color_state_equal (csi, csj));
else
g_assert_true (gdk_color_state_equal (csi, csj)); /* might break for non-default? */
}
}
}
static float
image_distance (const guchar *data,
2024-07-14 19:13:10 +00:00
gsize stride,
const guchar *data2,
2024-07-14 19:13:10 +00:00
gsize stride2,
gsize width,
2024-07-14 19:13:10 +00:00
gsize height)
{
float dist = 0;
for (gsize i = 0; i < height; i++)
{
const float *p = (const float *) (data + i * stride);
2024-07-14 19:13:10 +00:00
const float *p2 = (const float *) (data2 + i * stride2);
for (gsize j = 0; j < width; j++)
{
float dr, dg, db, da;
dr = p[4 * j + 0] - p2[4 * j + 0];
dg = p[4 * j + 1] - p2[4 * j + 1];
db = p[4 * j + 2] - p2[4 * j + 2];
da = p[4 * j + 3] - p2[4 * j + 3];
dist = MAX (dist, dr * dr + dg * dg + db * db + da * da);
}
}
return sqrt (dist);
}
static void
test_convert (gconstpointer testdata)
{
2024-07-14 19:13:10 +00:00
GdkColorState *cs = (GdkColorState *) testdata;
char *path;
2024-07-14 19:13:10 +00:00
GdkTexture *texture, *texture2;
GdkTextureDownloader *downloader;
2024-07-14 19:13:10 +00:00
GdkMemoryTextureBuilder *membuild;
GError *error = NULL;
2024-07-14 19:13:10 +00:00
GBytes *bytes, *bytes2;
const guchar *data, *data2;
gsize stride, stride2;
gsize width, height;
2024-07-14 19:13:10 +00:00
GdkMemoryFormat test_format;
2024-07-14 19:13:10 +00:00
if (g_test_rand_bit ())
test_format = GDK_MEMORY_R32G32B32A32_FLOAT_PREMULTIPLIED;
else
test_format = GDK_MEMORY_R32G32B32A32_FLOAT;
path = g_test_build_filename (G_TEST_DIST, "image-data", "image.png", NULL);
2024-07-14 19:13:10 +00:00
/* Create a texture */
texture = gdk_texture_new_from_filename (path, &error);
g_assert_no_error (error);
2024-07-14 19:13:10 +00:00
g_assert_true (gdk_color_state_equal (gdk_texture_get_color_state (texture),
gdk_color_state_get_srgb ()));
width = gdk_texture_get_width (texture);
height = gdk_texture_get_height (texture);
2024-07-14 19:13:10 +00:00
/* download the texture as float for later comparison */
downloader = gdk_texture_downloader_new (texture);
2024-07-14 19:13:10 +00:00
gdk_texture_downloader_set_format (downloader, test_format);
gdk_texture_downloader_set_color_state (downloader, gdk_texture_get_color_state (texture));
bytes = gdk_texture_downloader_download_bytes (downloader, &stride);
2024-07-14 19:13:10 +00:00
data = g_bytes_get_data (bytes, NULL);
/* Download the texture into the test colorstate, this does a conversion */
gdk_texture_downloader_set_color_state (downloader, cs);
bytes2 = gdk_texture_downloader_download_bytes (downloader, &stride2);
/* Create a new texture in the test colorstate with the just downloaded data */
membuild = gdk_memory_texture_builder_new ();
gdk_memory_texture_builder_set_format (membuild, test_format);
gdk_memory_texture_builder_set_color_state (membuild, cs);
gdk_memory_texture_builder_set_width (membuild, width);
gdk_memory_texture_builder_set_height (membuild, height);
gdk_memory_texture_builder_set_bytes (membuild, bytes2);
gdk_memory_texture_builder_set_stride (membuild, stride2);
texture2 = gdk_memory_texture_builder_build (membuild);
g_object_unref (membuild);
g_bytes_unref (bytes2);
/* Download the data of the new texture in the original texture's colorstate.
* This does the reverse conversion. */
gdk_texture_downloader_set_texture (downloader, texture2);
gdk_texture_downloader_set_color_state (downloader, gdk_texture_get_color_state (texture));
bytes2 = gdk_texture_downloader_download_bytes (downloader, &stride2);
data2 = g_bytes_get_data (bytes2, NULL);
/* Check that the conversions produce pixels that are close enough */
g_assert_cmpfloat (image_distance (data, stride, data2, stride2, width, height), <, 0.001);
g_test_message ("%g\n", image_distance (data, stride, data2, stride2, width, height));
g_bytes_unref (bytes2);
g_bytes_unref (bytes);
gdk_texture_downloader_free (downloader);
2024-07-14 19:13:10 +00:00
g_object_unref (texture2);
g_object_unref (texture);
g_free (path);
}
static void
test_png (gconstpointer testdata)
{
GdkColorState *cs = (GdkColorState *) testdata;
static const float texture_data[] = { 0,0,0,1, 0,0,1,1, 0,1,0,1, 0,1,1,1,
1,0,0,1, 1,0,1,1, 1,1,0,1, 1,1,1,1 };
static gsize width = 4, height = 2, stride = 4 * 4 * sizeof (float);
GdkMemoryTextureBuilder *membuild;
GdkTexture *saved, *loaded;
GBytes *bytes;
GError *error = NULL;
membuild = gdk_memory_texture_builder_new ();
gdk_memory_texture_builder_set_format (membuild, GDK_MEMORY_R32G32B32A32_FLOAT);
gdk_memory_texture_builder_set_color_state (membuild, cs);
gdk_memory_texture_builder_set_width (membuild, width);
gdk_memory_texture_builder_set_height (membuild, height);
bytes = g_bytes_new_static (texture_data, sizeof (texture_data));
gdk_memory_texture_builder_set_bytes (membuild, bytes);
g_bytes_unref (bytes);
gdk_memory_texture_builder_set_stride (membuild, stride);
saved = gdk_memory_texture_builder_build (membuild);
g_object_unref (membuild);
bytes = gdk_texture_save_to_png_bytes (saved);
loaded = gdk_texture_new_from_bytes (bytes, &error);
g_assert_nonnull (loaded);
g_assert_no_error (error);
g_bytes_unref (bytes);
g_assert_cmpint (gdk_texture_get_width (saved), ==, gdk_texture_get_width (loaded));
g_assert_cmpint (gdk_texture_get_height (saved), ==, gdk_texture_get_height (loaded));
g_assert_true (gdk_color_state_equal (gdk_texture_get_color_state (saved), gdk_texture_get_color_state (loaded)));
g_object_unref (saved);
g_object_unref (loaded);
}
int
main (int argc, char *argv[])
{
2024-07-14 19:13:10 +00:00
guint i;
(g_test_init) (&argc, &argv, NULL);
2024-07-14 19:13:10 +00:00
g_test_add_func ("/colorstate/equal", test_equal);
for (i = 0; ; i++)
{
GdkColorState *csi;
const char *cs_name;
char *test_name;
csi = get_color_state (i, &cs_name);
if (csi == NULL)
break;
test_name = g_strdup_printf ("/colorstate/convert/srgb/%s", cs_name);
g_test_add_data_func_full (test_name, csi, test_convert, (GDestroyNotify) gdk_color_state_unref);
g_free (test_name);
test_name = g_strdup_printf ("/colorstate/png/%s", cs_name);
g_test_add_data_func_full (test_name, csi, test_png, (GDestroyNotify) gdk_color_state_unref);
g_free (test_name);
2024-07-14 19:13:10 +00:00
}
return g_test_run ();
}