/* PIKA - Photo and Image Kooker Application
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
*
* Original copyright, applying to most contents (license remains unchanged):
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include "config.h"
#include
#include
#include
#include
#include "libpikabase/pikabase.h"
#include "libpikacolor/pikacolor.h"
#include "core-types.h"
#include "pika.h"
#include "pikaimage.h"
#include "pikaimage-color-profile.h"
#include "pikaimage-metadata.h"
#include "pikaimage-private.h"
#include "pikaimage-rotate.h"
#include "pikaimage-undo.h"
#include "pikaimage-undo-push.h"
#include "pikalayer-new.h"
/* public functions */
PikaMetadata *
pika_image_get_metadata (PikaImage *image)
{
PikaImagePrivate *private;
g_return_val_if_fail (PIKA_IS_IMAGE (image), NULL);
private = PIKA_IMAGE_GET_PRIVATE (image);
return private->metadata;
}
void
pika_image_set_metadata (PikaImage *image,
PikaMetadata *metadata,
gboolean push_undo)
{
PikaImagePrivate *private;
g_return_if_fail (PIKA_IS_IMAGE (image));
private = PIKA_IMAGE_GET_PRIVATE (image);
if (metadata != private->metadata)
{
if (push_undo)
pika_image_undo_push_image_metadata (image, NULL);
g_set_object (&private->metadata, metadata);
if (private->metadata)
{
pika_image_metadata_update_pixel_size (image);
pika_image_metadata_update_bits_per_sample (image);
pika_image_metadata_update_resolution (image);
pika_image_metadata_update_colorspace (image);
}
g_object_notify (G_OBJECT (image), "metadata");
}
}
void
pika_image_metadata_update_pixel_size (PikaImage *image)
{
PikaMetadata *metadata;
g_return_if_fail (PIKA_IS_IMAGE (image));
metadata = pika_image_get_metadata (image);
if (metadata)
{
pika_metadata_set_pixel_size (metadata,
pika_image_get_width (image),
pika_image_get_height (image));
}
}
void
pika_image_metadata_update_bits_per_sample (PikaImage *image)
{
PikaMetadata *metadata;
g_return_if_fail (PIKA_IS_IMAGE (image));
metadata = pika_image_get_metadata (image);
if (metadata)
{
switch (pika_image_get_component_type (image))
{
case PIKA_COMPONENT_TYPE_U8:
pika_metadata_set_bits_per_sample (metadata, 8);
break;
case PIKA_COMPONENT_TYPE_U16:
case PIKA_COMPONENT_TYPE_HALF:
pika_metadata_set_bits_per_sample (metadata, 16);
break;
case PIKA_COMPONENT_TYPE_U32:
case PIKA_COMPONENT_TYPE_FLOAT:
pika_metadata_set_bits_per_sample (metadata, 32);
break;
case PIKA_COMPONENT_TYPE_DOUBLE:
pika_metadata_set_bits_per_sample (metadata, 64);
break;
}
}
}
void
pika_image_metadata_update_resolution (PikaImage *image)
{
PikaMetadata *metadata;
g_return_if_fail (PIKA_IS_IMAGE (image));
metadata = pika_image_get_metadata (image);
if (metadata)
{
gdouble xres, yres;
pika_image_get_resolution (image, &xres, &yres);
pika_metadata_set_resolution (metadata, xres, yres,
pika_image_get_unit (image));
}
}
void
pika_image_metadata_update_colorspace (PikaImage *image)
{
PikaMetadata *metadata;
g_return_if_fail (PIKA_IS_IMAGE (image));
metadata = pika_image_get_metadata (image);
if (metadata)
{
/* See the discussions in issue #3532 and issue #301 */
PikaColorProfile *profile = pika_image_get_color_profile (image);
PikaMetadataColorspace space = PIKA_METADATA_COLORSPACE_UNSPECIFIED;
if (profile)
{
static PikaColorProfile *adobe = NULL;
if (! adobe)
adobe = pika_color_profile_new_rgb_adobe ();
if (pika_color_profile_is_equal (profile, adobe))
space = PIKA_METADATA_COLORSPACE_ADOBERGB;
}
else
{
space = PIKA_METADATA_COLORSPACE_SRGB;
}
pika_metadata_set_colorspace (metadata, space);
}
}
/**
* pika_image_metadata_load_thumbnail:
* @pika: The #Pika object.
* @file: A #GFile image.
* @full_image_width: the width of the full image (not the thumbnail).
* @full_image_height: the height of the full image (not the thumbnail).
* @error: Return location for error message
*
* Retrieves a thumbnail from metadata in @file if present.
* It does not need to actually load the full image, only the metadata through
* GExiv2, which makes it fast.
*
* Returns: (transfer none) (nullable): a #PikaImage of the @file thumbnail.
*/
PikaImage *
pika_image_metadata_load_thumbnail (Pika *pika,
GFile *file,
gint *full_image_width,
gint *full_image_height,
const Babl **format,
GError **error)
{
PikaMetadata *metadata;
GInputStream *input_stream;
GdkPixbuf *pixbuf;
guint8 *thumbnail_buffer;
gint thumbnail_size;
PikaImage *image = NULL;
g_return_val_if_fail (G_IS_FILE (file), NULL);
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
metadata = pika_metadata_load_from_file (file, error);
if (! metadata)
return NULL;
if (! gexiv2_metadata_get_exif_thumbnail (GEXIV2_METADATA (metadata),
&thumbnail_buffer,
&thumbnail_size))
{
g_object_unref (metadata);
return NULL;
}
input_stream = g_memory_input_stream_new_from_data (thumbnail_buffer,
thumbnail_size,
(GDestroyNotify) g_free);
pixbuf = gdk_pixbuf_new_from_stream (input_stream, NULL, error);
g_object_unref (input_stream);
if (pixbuf)
{
PikaLayer *layer;
image = pika_image_new (pika,
gdk_pixbuf_get_width (pixbuf),
gdk_pixbuf_get_height (pixbuf),
PIKA_RGB, PIKA_PRECISION_U8_NON_LINEAR);
pika_image_undo_disable (image);
/* XXX This is possibly wrong, because an image of a given color space may
* have a thumbnail stored in a different colorspace. This is even more
* true with PIKA which always exports RGB thumbnails (see code in
* pika_image_metadata_save_filter()), even for say grayscale images.
* Nevertheless other software may store thumbnail using the same
* colorspace.
*/
*format = pika_pixbuf_get_format (pixbuf);
layer = pika_layer_new_from_pixbuf (pixbuf, image,
pika_image_get_layer_format (image, FALSE),
/* No need to localize; this image is short-lived. */
"Background",
PIKA_OPACITY_OPAQUE,
pika_image_get_default_new_layer_mode (image));
g_object_unref (pixbuf);
pika_image_add_layer (image, layer, NULL, 0, FALSE);
pika_image_apply_metadata_orientation (image, pika_get_user_context (pika), metadata, NULL);
}
/* This is the unoriented dimensions. Should we switch when there is a 90 or
* 270 degree rotation? We don't actually know if the metadata orientation is
* correct.
*/
*full_image_width = gexiv2_metadata_get_pixel_width (GEXIV2_METADATA (metadata));
*full_image_height = gexiv2_metadata_get_pixel_height (GEXIV2_METADATA (metadata));
if (*full_image_width < 1 || *full_image_height < 1)
{
/* Dimensions stored in metadata might be less accurate, yet it's still
* informational.
*/
*full_image_width = gexiv2_metadata_try_get_metadata_pixel_width (GEXIV2_METADATA (metadata), NULL);
*full_image_height = gexiv2_metadata_try_get_metadata_pixel_width (GEXIV2_METADATA (metadata), NULL);
}
if (*full_image_width < 1 || *full_image_height < 1)
{
*full_image_width = 0;
*full_image_height = 0;
}
g_object_unref (metadata);
return image;
}