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;
|
2024-07-17 00:19:39 +00:00
|
|
|
gsize imax = 0, jmax = 0;
|
|
|
|
float r1 = 0, g1 = 0, b1 = 0, a1 = 0;
|
|
|
|
float r2 = 0, g2 = 0, b2 = 0, a2 = 0;
|
2024-07-09 02:03:00 +00:00
|
|
|
|
|
|
|
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];
|
|
|
|
|
2024-07-17 00:19:39 +00:00
|
|
|
float d = dr * dr + dg * dg + db * db + da * da;
|
|
|
|
if (d > dist)
|
|
|
|
{
|
|
|
|
dist = d;
|
|
|
|
imax = i;
|
|
|
|
jmax = j;
|
|
|
|
r1 = p[4+j+0];
|
|
|
|
g1 = p[4+j+1];
|
|
|
|
b1 = p[4+j+2];
|
|
|
|
a1 = p[4+j+3];
|
|
|
|
r2 = p2[4+j+0];
|
|
|
|
g2 = p2[4+j+1];
|
|
|
|
b2 = p2[4+j+2];
|
|
|
|
a2 = p2[4+j+3];
|
|
|
|
}
|
2024-07-09 02:03:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-17 00:19:39 +00:00
|
|
|
if (g_test_verbose() &&
|
|
|
|
dist > 0)
|
|
|
|
{
|
|
|
|
g_print ("worst pixel %lu %lu: %f %f %f %f vs %f %f %f %f\n",
|
|
|
|
imax, jmax, r1, g1, b1, a1, r2, g2, b2, a2);
|
|
|
|
}
|
|
|
|
|
2024-07-09 02:03:00 +00:00
|
|
|
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.
|
2024-07-17 00:19:39 +00:00
|
|
|
* This does the reverse conversion.
|
|
|
|
*/
|
2024-07-14 19:13:10 +00:00
|
|
|
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);
|
|
|
|
|
2024-07-17 00:19:39 +00:00
|
|
|
if (g_test_verbose ())
|
|
|
|
g_test_message ("%g\n", image_distance (data, stride, data2, stride2, width, height));
|
2024-07-14 19:13:10 +00:00
|
|
|
|
|
|
|
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-07-17 00:19:39 +00:00
|
|
|
static void
|
|
|
|
test_cicp (void)
|
|
|
|
{
|
|
|
|
GdkCicpParams *params;
|
|
|
|
GdkColorState *cs;
|
|
|
|
GError *error = NULL;
|
|
|
|
GdkCicpParams *params2;
|
|
|
|
|
|
|
|
params = gdk_cicp_params_new ();
|
|
|
|
|
|
|
|
g_assert_true (gdk_cicp_params_get_color_primaries (params) == 2);
|
|
|
|
g_assert_true (gdk_cicp_params_get_transfer_function (params) == 2);
|
|
|
|
g_assert_true (gdk_cicp_params_get_matrix_coefficients (params) == 2);
|
|
|
|
g_assert_true (gdk_cicp_params_get_range (params) == GDK_CICP_RANGE_NARROW);
|
|
|
|
|
|
|
|
cs = gdk_cicp_params_build_color_state (params, &error);
|
|
|
|
|
|
|
|
g_assert_null (cs);
|
|
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
|
|
|
|
|
|
|
|
g_clear_error (&error);
|
|
|
|
|
|
|
|
gdk_cicp_params_set_color_primaries (params, 5);
|
|
|
|
gdk_cicp_params_set_transfer_function (params, 1);
|
|
|
|
gdk_cicp_params_set_matrix_coefficients (params, 0);
|
|
|
|
gdk_cicp_params_set_range (params, GDK_CICP_RANGE_FULL);
|
|
|
|
|
|
|
|
cs = gdk_cicp_params_build_color_state (params, &error);
|
|
|
|
|
|
|
|
g_assert_nonnull (cs);
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
params2 = gdk_color_state_create_cicp_params (cs);
|
|
|
|
|
|
|
|
g_assert_true (gdk_cicp_params_get_color_primaries (params) == gdk_cicp_params_get_color_primaries (params2));
|
|
|
|
g_object_unref (params2);
|
|
|
|
|
|
|
|
gdk_color_state_unref (cs);
|
|
|
|
g_object_unref (params);
|
|
|
|
}
|
|
|
|
|
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);
|
2024-07-17 00:19:39 +00:00
|
|
|
g_test_add_func ("/colorstate/cicp", test_cicp);
|
2024-07-14 19:13:10 +00:00
|
|
|
|
|
|
|
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);
|
2024-07-17 00:19:39 +00:00
|
|
|
g_test_add_data_func_full (test_name, gdk_color_state_ref (csi), test_convert, (GDestroyNotify) gdk_color_state_unref);
|
2024-07-14 19:13:10 +00:00
|
|
|
g_free (test_name);
|
2024-07-16 15:43:27 +00:00
|
|
|
test_name = g_strdup_printf ("/colorstate/png/%s", cs_name);
|
2024-07-17 00:19:39 +00:00
|
|
|
g_test_add_data_func_full (test_name, gdk_color_state_ref (csi), test_png, (GDestroyNotify) gdk_color_state_unref);
|
2024-07-16 15:43:27 +00:00
|
|
|
g_free (test_name);
|
2024-07-17 00:19:39 +00:00
|
|
|
|
|
|
|
gdk_color_state_unref (csi);
|
|
|
|
}
|
|
|
|
|
|
|
|
GdkCicpParams *params = gdk_cicp_params_new ();
|
|
|
|
for (guint primaries = 0; primaries < 32; primaries++)
|
|
|
|
{
|
|
|
|
gdk_cicp_params_set_color_primaries (params, primaries);
|
|
|
|
for (guint tf = 0; tf < 32; tf++)
|
|
|
|
{
|
|
|
|
gdk_cicp_params_set_transfer_function (params, tf);
|
|
|
|
for (guint matrix = 0; matrix < 32; matrix++)
|
|
|
|
{
|
|
|
|
GdkColorState *color_state;
|
|
|
|
|
|
|
|
gdk_cicp_params_set_matrix_coefficients (params, matrix);
|
|
|
|
gdk_cicp_params_set_range (params, GDK_CICP_RANGE_NARROW);
|
|
|
|
color_state = gdk_cicp_params_build_color_state (params, NULL);
|
|
|
|
|
|
|
|
if (color_state)
|
|
|
|
{
|
|
|
|
char *test_name = g_strdup_printf ("/colorstate/convert/srgb/cicp/%u/%u/%u/0", primaries, tf, matrix);
|
|
|
|
g_test_add_data_func_full (test_name, gdk_color_state_ref (color_state), test_convert, (GDestroyNotify) gdk_color_state_unref);
|
|
|
|
g_free (test_name);
|
|
|
|
|
|
|
|
gdk_color_state_unref (color_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
gdk_cicp_params_set_range (params, GDK_CICP_RANGE_FULL);
|
|
|
|
color_state = gdk_cicp_params_build_color_state (params, NULL);
|
|
|
|
if (color_state)
|
|
|
|
{
|
|
|
|
char *test_name = g_strdup_printf ("/colorstate/convert/srgb/cicp/%u/%u/%u/1", primaries, tf, matrix);
|
|
|
|
g_test_add_data_func_full (test_name, gdk_color_state_ref (color_state), test_convert, (GDestroyNotify) gdk_color_state_unref);
|
|
|
|
g_free (test_name);
|
|
|
|
|
|
|
|
gdk_color_state_unref (color_state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-07-14 19:13:10 +00:00
|
|
|
}
|
2024-06-08 18:37:51 +00:00
|
|
|
|
|
|
|
return g_test_run ();
|
|
|
|
}
|