2024-06-08 18:37:51 +00:00
|
|
|
#include <gdk/gdk.h>
|
2024-07-09 02:03:00 +00:00
|
|
|
#include <math.h>
|
2024-06-08 18:37:51 +00:00
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-08 18:37:51 +00:00
|
|
|
static void
|
2024-07-14 19:13:10 +00:00
|
|
|
test_equal (void)
|
2024-06-08 18:37:51 +00:00
|
|
|
{
|
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-06-08 18:37:51 +00:00
|
|
|
|
2024-07-14 19:13:10 +00:00
|
|
|
g_assert_true (gdk_color_state_equal (csi, csi));
|
2024-06-08 18:37:51 +00:00
|
|
|
|
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? */
|
|
|
|
}
|
|
|
|
}
|
2024-06-08 18:37:51 +00:00
|
|
|
}
|
|
|
|
|
2024-07-09 02:03:00 +00:00
|
|
|
static float
|
|
|
|
image_distance (const guchar *data,
|
2024-07-14 19:13:10 +00:00
|
|
|
gsize stride,
|
2024-07-09 02:03:00 +00:00
|
|
|
const guchar *data2,
|
2024-07-14 19:13:10 +00:00
|
|
|
gsize stride2,
|
2024-07-09 02:03:00 +00:00
|
|
|
gsize width,
|
2024-07-14 19:13:10 +00:00
|
|
|
gsize height)
|
2024-07-09 02:03:00 +00:00
|
|
|
{
|
|
|
|
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);
|
2024-07-09 02:03:00 +00:00
|
|
|
|
|
|
|
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;
|
2024-07-09 02:03:00 +00:00
|
|
|
char *path;
|
2024-07-14 19:13:10 +00:00
|
|
|
GdkTexture *texture, *texture2;
|
2024-07-09 02:03:00 +00:00
|
|
|
GdkTextureDownloader *downloader;
|
2024-07-14 19:13:10 +00:00
|
|
|
GdkMemoryTextureBuilder *membuild;
|
2024-07-09 02:03:00 +00:00
|
|
|
GError *error = NULL;
|
2024-07-14 19:13:10 +00:00
|
|
|
GBytes *bytes, *bytes2;
|
|
|
|
const guchar *data, *data2;
|
|
|
|
gsize stride, stride2;
|
2024-07-09 02:03:00 +00:00
|
|
|
gsize width, height;
|
2024-07-14 19:13:10 +00:00
|
|
|
GdkMemoryFormat test_format;
|
2024-07-09 02:03:00 +00:00
|
|
|
|
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;
|
2024-07-09 02:03:00 +00:00
|
|
|
|
|
|
|
path = g_test_build_filename (G_TEST_DIST, "image-data", "image.png", NULL);
|
|
|
|
|
2024-07-14 19:13:10 +00:00
|
|
|
/* Create a texture */
|
2024-07-09 02:03:00 +00:00
|
|
|
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 ()));
|
2024-07-09 02:03:00 +00:00
|
|
|
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 */
|
2024-07-09 02:03:00 +00:00
|
|
|
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));
|
2024-07-09 02:03:00 +00:00
|
|
|
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);
|
2024-07-09 02:03:00 +00:00
|
|
|
g_bytes_unref (bytes);
|
|
|
|
gdk_texture_downloader_free (downloader);
|
2024-07-14 19:13:10 +00:00
|
|
|
g_object_unref (texture2);
|
2024-07-09 02:03:00 +00:00
|
|
|
g_object_unref (texture);
|
|
|
|
g_free (path);
|
|
|
|
}
|
|
|
|
|
2024-07-16 15:43:27 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2024-06-08 18:37:51 +00:00
|
|
|
int
|
|
|
|
main (int argc, char *argv[])
|
|
|
|
{
|
2024-07-14 19:13:10 +00:00
|
|
|
guint i;
|
|
|
|
|
2024-06-08 18:37:51 +00:00
|
|
|
(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);
|
2024-07-16 15:43:27 +00:00
|
|
|
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
|
|
|
}
|
2024-06-08 18:37:51 +00:00
|
|
|
|
|
|
|
return g_test_run ();
|
|
|
|
}
|