/* 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 "libpikacolor/pikacolor.h" #include "libpikaconfig/pikaconfig.h" #include "core-types.h" #include "gegl/pika-babl.h" #include "gegl/pika-gegl-loops.h" #include "pikabuffer.h" #include "pikaimage.h" #include "pikaimage-color-profile.h" #include "pikalayer.h" #include "pikalayer-new.h" /* local function prototypes */ static void pika_layer_new_convert_buffer (PikaLayer *layer, GeglBuffer *src_buffer, PikaColorProfile *src_profile, GError **error); /* public functions */ PikaLayer * pika_layer_new (PikaImage *image, gint width, gint height, const Babl *format, const gchar *name, gdouble opacity, PikaLayerMode mode) { PikaLayer *layer; g_return_val_if_fail (PIKA_IS_IMAGE (image), NULL); g_return_val_if_fail (width > 0, NULL); g_return_val_if_fail (height > 0, NULL); g_return_val_if_fail (format != NULL, NULL); layer = PIKA_LAYER (pika_drawable_new (PIKA_TYPE_LAYER, image, name, 0, 0, width, height, format)); pika_layer_set_opacity (layer, opacity, FALSE); pika_layer_set_mode (layer, mode, FALSE); return layer; } /** * pika_layer_new_from_buffer: * @buffer: The buffer to make the new layer from. * @dest_image: The image the new layer will be added to. * @format: The #Babl format of the new layer. * @name: The new layer's name. * @opacity: The new layer's opacity. * @mode: The new layer's mode. * * Copies %buffer to a layer taking into consideration the * possibility of transforming the contents to meet the requirements * of the target image type * * Returns: The new layer. **/ PikaLayer * pika_layer_new_from_buffer (PikaBuffer *buffer, PikaImage *dest_image, const Babl *format, const gchar *name, gdouble opacity, PikaLayerMode mode) { g_return_val_if_fail (PIKA_IS_BUFFER (buffer), NULL); g_return_val_if_fail (PIKA_IS_IMAGE (dest_image), NULL); g_return_val_if_fail (format != NULL, NULL); return pika_layer_new_from_gegl_buffer (pika_buffer_get_buffer (buffer), dest_image, format, name, opacity, mode, pika_buffer_get_color_profile (buffer)); } /** * pika_layer_new_from_gegl_buffer: * @buffer: The buffer to make the new layer from. * @dest_image: The image the new layer will be added to. * @format: The #Babl format of the new layer. * @name: The new layer's name. * @opacity: The new layer's opacity. * @mode: The new layer's mode. * * Copies %buffer to a layer taking into consideration the * possibility of transforming the contents to meet the requirements * of the target image type * * Returns: The new layer. **/ PikaLayer * pika_layer_new_from_gegl_buffer (GeglBuffer *buffer, PikaImage *dest_image, const Babl *format, const gchar *name, gdouble opacity, PikaLayerMode mode, PikaColorProfile *buffer_profile) { PikaLayer *layer; const GeglRectangle *extent; g_return_val_if_fail (GEGL_IS_BUFFER (buffer), NULL); g_return_val_if_fail (PIKA_IS_IMAGE (dest_image), NULL); g_return_val_if_fail (format != NULL, NULL); g_return_val_if_fail (buffer_profile == NULL || PIKA_IS_COLOR_PROFILE (buffer_profile), NULL); extent = gegl_buffer_get_extent (buffer); /* do *not* use the buffer's format because this function gets * buffers of any format passed, and converts them */ layer = pika_layer_new (dest_image, extent->width, extent->height, format, name, opacity, mode); if (extent->x != 0 || extent->y != 0) pika_item_translate (PIKA_ITEM (layer), extent->x, extent->y, FALSE); pika_layer_new_convert_buffer (layer, buffer, buffer_profile, NULL); return layer; } /** * pika_layer_new_from_pixbuf: * @pixbuf: The pixbuf to make the new layer from. * @dest_image: The image the new layer will be added to. * @format: The #Babl format of the new layer. * @name: The new layer's name. * @opacity: The new layer's opacity. * @mode: The new layer's mode. * * Copies %pixbuf to a layer taking into consideration the * possibility of transforming the contents to meet the requirements * of the target image type * * Returns: The new layer. **/ PikaLayer * pika_layer_new_from_pixbuf (GdkPixbuf *pixbuf, PikaImage *dest_image, const Babl *format, const gchar *name, gdouble opacity, PikaLayerMode mode) { PikaLayer *layer; GeglBuffer *buffer; guint8 *icc_data; gsize icc_len; PikaColorProfile *profile = NULL; g_return_val_if_fail (GDK_IS_PIXBUF (pixbuf), NULL); g_return_val_if_fail (PIKA_IS_IMAGE (dest_image), NULL); g_return_val_if_fail (format != NULL, NULL); layer = pika_layer_new (dest_image, gdk_pixbuf_get_width (pixbuf), gdk_pixbuf_get_height (pixbuf), format, name, opacity, mode); buffer = pika_pixbuf_create_buffer (pixbuf); icc_data = pika_pixbuf_get_icc_profile (pixbuf, &icc_len); if (icc_data) { profile = pika_color_profile_new_from_icc_profile (icc_data, icc_len, NULL); g_free (icc_data); } pika_layer_new_convert_buffer (layer, buffer, profile, NULL); if (profile) g_object_unref (profile); g_object_unref (buffer); return layer; } /* private functions */ static void pika_layer_new_convert_buffer (PikaLayer *layer, GeglBuffer *src_buffer, PikaColorProfile *src_profile, GError **error) { PikaDrawable *drawable = PIKA_DRAWABLE (layer); GeglBuffer *dest_buffer = pika_drawable_get_buffer (drawable); PikaColorProfile *dest_profile; if (! src_profile) { const Babl *src_format = gegl_buffer_get_format (src_buffer); src_profile = pika_babl_format_get_color_profile (src_format); } else { g_object_ref (src_profile); } dest_profile = pika_color_managed_get_color_profile (PIKA_COLOR_MANAGED (layer)); pika_gegl_convert_color_profile (src_buffer, NULL, src_profile, dest_buffer, NULL, dest_profile, PIKA_COLOR_RENDERING_INTENT_PERCEPTUAL, TRUE, NULL); g_object_unref (src_profile); }