/* 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; }