256 lines
8.4 KiB
C
256 lines
8.4 KiB
C
/* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <cairo.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <gegl.h>
|
|
|
|
#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);
|
|
}
|