[gimp/alxsa-psd-cmyk-export] plug-ins: Export PSD as CMYK file
- From: Alx Sa <sawyeralex src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gimp/alxsa-psd-cmyk-export] plug-ins: Export PSD as CMYK file
- Date: Tue, 27 Sep 2022 11:27:24 +0000 (UTC)
commit 48a283bd35dbd3e4e3fed39fdee029cdd4466455
Author: Alx Sa <cmyk student gmail com>
Date: Wed Aug 24 12:28:58 2022 +0000
plug-ins: Export PSD as CMYK file
Adds an option to export as CMYK (mutually exclusive with the existing
Duotone export option). Header information (mode and number of channels)
are set, and the data itself is converted via Babl.
Layer channels are also hardcoded to 4, since GIMP currently doesn't
support CMYK channels.
plug-ins/file-psd/psd-save.c | 453 +++++++++++++++++++++++++++++--------------
plug-ins/file-psd/psd.c | 45 ++---
2 files changed, 318 insertions(+), 180 deletions(-)
---
diff --git a/plug-ins/file-psd/psd-save.c b/plug-ins/file-psd/psd-save.c
index ef70c9483a..2b942e0b4d 100644
--- a/plug-ins/file-psd/psd-save.c
+++ b/plug-ins/file-psd/psd-save.c
@@ -133,6 +133,7 @@ static void reshuffle_cmap_write (guchar *mapGimp);
static void save_header (GOutputStream *output,
GimpImage *image,
+ gboolean export_cmyk,
gboolean export_duotone);
static void save_color_mode_data (GOutputStream *output,
@@ -141,13 +142,16 @@ static void save_color_mode_data (GOutputStream *output,
static void save_resources (GOutputStream *output,
GimpImage *image,
+ gboolean export_cmyk,
gboolean export_duotone);
static void save_layer_and_mask (GOutputStream *output,
- GimpImage *image);
+ GimpImage *image,
+ gboolean export_cmyk);
static void save_data (GOutputStream *output,
- GimpImage *image);
+ GimpImage *image,
+ gboolean export_cmyk);
static void xfwrite (GOutputStream *output,
gconstpointer buf,
@@ -181,10 +185,12 @@ static void write_datablock_luni (GOutputStream *output,
static void write_pixel_data (GOutputStream *output,
+ GimpImage *image,
GimpDrawable *drawable,
goffset *ChanLenPosition,
goffset rowlenOffset,
- gboolean write_mask);
+ gboolean write_mask,
+ gboolean export_cmyk);
static GimpLayer * create_merged_image (GimpImage *image);
@@ -528,8 +534,11 @@ reshuffle_cmap_write (guchar *mapGimp)
static void
save_header (GOutputStream *output,
GimpImage *image,
+ gboolean export_cmyk,
gboolean export_duotone)
{
+ gint nChannels;
+
IFDBG(1) g_debug ("Function: save_header\n"
"\tRows: %d\n"
"\tColumns: %d\n"
@@ -538,18 +547,29 @@ save_header (GOutputStream *output,
PSDImageData.image_height, PSDImageData.image_width,
PSDImageData.baseType, PSDImageData.nChannels);
+ if (export_cmyk)
+ {
+ nChannels =
+ gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)) ?
+ 5 : 4;
+ }
+ else
+ {
+ nChannels = (PSDImageData.nChannels + nChansLayer (PSDImageData.baseType,
+ gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)), 0));
+ }
+
xfwrite (output, "8BPS", 4, "signature");
write_gint16 (output, 1, "version");
write_gint32 (output, 0, "reserved 1"); /* 6 for the 'reserved' field + 4 bytes for a long */
write_gint16 (output, 0, "reserved 1"); /* and 2 bytes for a short */
- write_gint16 (output, (PSDImageData.nChannels +
- nChansLayer (PSDImageData.baseType,
- gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)), 0)),
- "channels");
+ write_gint16 (output, nChannels, "channels");
write_gint32 (output, PSDImageData.image_height, "rows");
write_gint32 (output, PSDImageData.image_width, "columns");
write_gint16 (output, 8 * get_bpc (image), "depth");
- if (export_duotone)
+ if (export_cmyk)
+ write_gint16 (output, PSD_CMYK, "mode");
+ else if (export_duotone)
write_gint16 (output, PSD_DUOTONE, "mode");
else
write_gint16 (output, gimpBaseTypeToPsdMode (PSDImageData.baseType), "mode");
@@ -634,6 +654,7 @@ save_color_mode_data (GOutputStream *output,
static void
save_resources (GOutputStream *output,
GimpImage *image,
+ gboolean export_cmyk,
gboolean export_duotone)
{
GList *iter;
@@ -668,124 +689,125 @@ save_resources (GOutputStream *output,
/* --------------- Write Channel names --------------- */
-
- if (PSDImageData.nChannels > 0 ||
- gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
+ if (! export_cmyk)
{
- xfwrite (output, "8BIM", 4, "imageresources signature");
- write_gint16 (output, 0x03EE, "0x03EE Id"); /* 1006 */
- /* write_pascalstring (output, Name, "Id name"); */
- write_gint16 (output, 0, "Id name"); /* Set to null string (two zeros) */
+ if (PSDImageData.nChannels > 0 ||
+ gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
+ {
+ xfwrite (output, "8BIM", 4, "imageresources signature");
+ write_gint16 (output, 0x03EE, "0x03EE Id"); /* 1006 */
+ /* write_pascalstring (output, Name, "Id name"); */
+ write_gint16 (output, 0, "Id name"); /* Set to null string (two zeros) */
- /* Mark current position in the file */
+ /* Mark current position in the file */
- name_sec = g_seekable_tell (G_SEEKABLE (output));
- write_gint32 (output, 0, "0x03EE resource size");
+ name_sec = g_seekable_tell (G_SEEKABLE (output));
+ write_gint32 (output, 0, "0x03EE resource size");
- /* Write all strings */
+ /* Write all strings */
- /* if the merged_image contains transparency, write a name for it first */
- if (gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
- write_string (output, "Transparency", "channel name");
+ /* if the merged_image contains transparency, write a name for it first */
+ if (gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
+ write_string (output, "Transparency", "channel name");
- for (iter = PSDImageData.lChannels; iter; iter = g_list_next (iter))
- {
- char *chName = gimp_item_get_name (iter->data);
- write_string (output, chName, "channel name");
- g_free (chName);
- }
- /* Calculate and write actual resource's length */
+ for (iter = PSDImageData.lChannels; iter; iter = g_list_next (iter))
+ {
+ char *chName = gimp_item_get_name (iter->data);
+ write_string (output, chName, "channel name");
+ g_free (chName);
+ }
+ /* Calculate and write actual resource's length */
- eof_pos = g_seekable_tell (G_SEEKABLE (output));
+ eof_pos = g_seekable_tell (G_SEEKABLE (output));
- g_seekable_seek (G_SEEKABLE (output),
- name_sec, G_SEEK_SET,
- NULL, NULL /*FIXME: error*/);
- write_gint32 (output, eof_pos - name_sec - sizeof (gint32), "0x03EE resource size");
- IFDBG(1) g_debug ("\tTotal length of 0x03EE resource: %d",
- (int) (eof_pos - name_sec - sizeof (gint32)));
+ g_seekable_seek (G_SEEKABLE (output),
+ name_sec, G_SEEK_SET,
+ NULL, NULL /*FIXME: error*/);
+ write_gint32 (output, eof_pos - name_sec - sizeof (gint32), "0x03EE resource size");
+ IFDBG(1) g_debug ("\tTotal length of 0x03EE resource: %d",
+ (int) (eof_pos - name_sec - sizeof (gint32)));
- /* Return to EOF to continue writing */
+ /* Return to EOF to continue writing */
- g_seekable_seek (G_SEEKABLE (output),
- eof_pos, G_SEEK_SET,
- NULL, NULL /*FIXME: error*/);
+ g_seekable_seek (G_SEEKABLE (output),
+ eof_pos, G_SEEK_SET,
+ NULL, NULL /*FIXME: error*/);
- /* Pad if length is odd */
+ /* Pad if length is odd */
- if ((eof_pos - name_sec - sizeof (gint32)) & 1)
- write_gchar (output, 0, "pad byte");
- }
+ if ((eof_pos - name_sec - sizeof (gint32)) & 1)
+ write_gchar (output, 0, "pad byte");
+ }
- /* --------------- Write Channel properties --------------- */
+ /* --------------- Write Channel properties --------------- */
- if (PSDImageData.nChannels > 0 ||
- gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
- {
- xfwrite (output, "8BIM", 4, "imageresources signature");
- write_gint16 (output, 0x0435, "0x0435 Id"); /* 1077 */
- /* write_pascalstring (output, Name, "Id name"); */
- write_gint16 (output, 0, "Id name"); /* Set to null string (two zeros) */
- write_gint32 (output,
- 4 +
- 13 * (gimp_drawable_has_alpha (
- GIMP_DRAWABLE (PSDImageData.merged_layer)) +
- PSDImageData.nChannels),
- "0x0435 resource size");
-
- /* The function of the first 4 bytes is unclear. As per
- * load_resource_1077() in psd-image-res-load.c, it seems to be a version
- * number that is always one.
- */
- write_gint32 (output, 1, "0x0435 version");
+ if (PSDImageData.nChannels > 0 ||
+ gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
+ {
+ xfwrite (output, "8BIM", 4, "imageresources signature");
+ write_gint16 (output, 0x0435, "0x0435 Id"); /* 1077 */
+ /* write_pascalstring (output, Name, "Id name"); */
+ write_gint16 (output, 0, "Id name"); /* Set to null string (two zeros) */
+ write_gint32 (output,
+ 4 +
+ 13 * (gimp_drawable_has_alpha (
+ GIMP_DRAWABLE (PSDImageData.merged_layer)) +
+ PSDImageData.nChannels),
+ "0x0435 resource size");
+
+ /* The function of the first 4 bytes is unclear. As per
+ * load_resource_1077() in psd-image-res-load.c, it seems to be a version
+ * number that is always one.
+ */
+ write_gint32 (output, 1, "0x0435 version");
- /* Write all channel properties */
+ /* Write all channel properties */
- #define DOUBLE_TO_INT16(x) ROUND (SAFE_CLAMP (x, 0.0, 1.0) * 0xffff)
+ #define DOUBLE_TO_INT16(x) ROUND (SAFE_CLAMP (x, 0.0, 1.0) * 0xffff)
- /* if the merged_image contains transparency, write its properties first */
- if (gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
- {
- write_gint16 (output, PSD_CS_RGB, "channel color space");
- write_gint16 (output, DOUBLE_TO_INT16 (1.0), "channel color r");
- write_gint16 (output, DOUBLE_TO_INT16 (0.0), "channel color g");
- write_gint16 (output, DOUBLE_TO_INT16 (0.0), "channel color b");
- write_gint16 (output, 0, "channel color padding");
- write_gint16 (output, 100, "channel opacity");
- write_gchar (output, 1, "channel mode");
- }
+ /* if the merged_image contains transparency, write its properties first */
+ if (gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)))
+ {
+ write_gint16 (output, PSD_CS_RGB, "channel color space");
+ write_gint16 (output, DOUBLE_TO_INT16 (1.0), "channel color r");
+ write_gint16 (output, DOUBLE_TO_INT16 (0.0), "channel color g");
+ write_gint16 (output, DOUBLE_TO_INT16 (0.0), "channel color b");
+ write_gint16 (output, 0, "channel color padding");
+ write_gint16 (output, 100, "channel opacity");
+ write_gchar (output, 1, "channel mode");
+ }
- for (iter = PSDImageData.lChannels; iter; iter = g_list_next (iter))
- {
- GimpChannel *channel = iter->data;
- GimpRGB color;
- gdouble opacity;
-
- gimp_channel_get_color (channel, &color);
- opacity = gimp_channel_get_opacity (channel);
-
- write_gint16 (output, PSD_CS_RGB, "channel color space");
- write_gint16 (output, DOUBLE_TO_INT16 (color.r), "channel color r");
- write_gint16 (output, DOUBLE_TO_INT16 (color.g), "channel color g");
- write_gint16 (output, DOUBLE_TO_INT16 (color.b), "channel color b");
- write_gint16 (output, 0, "channel color padding");
- write_gint16 (output, ROUND (opacity), "channel opacity");
- write_gchar (output, 1, "channel mode");
- }
+ for (iter = PSDImageData.lChannels; iter; iter = g_list_next (iter))
+ {
+ GimpChannel *channel = iter->data;
+ GimpRGB color;
+ gdouble opacity;
+
+ gimp_channel_get_color (channel, &color);
+ opacity = gimp_channel_get_opacity (channel);
+
+ write_gint16 (output, PSD_CS_RGB, "channel color space");
+ write_gint16 (output, DOUBLE_TO_INT16 (color.r), "channel color r");
+ write_gint16 (output, DOUBLE_TO_INT16 (color.g), "channel color g");
+ write_gint16 (output, DOUBLE_TO_INT16 (color.b), "channel color b");
+ write_gint16 (output, 0, "channel color padding");
+ write_gint16 (output, ROUND (opacity), "channel opacity");
+ write_gchar (output, 1, "channel mode");
+ }
- #undef DOUBLE_TO_INT16
+ #undef DOUBLE_TO_INT16
- /* Pad if length is odd */
+ /* Pad if length is odd */
- if (g_seekable_tell (G_SEEKABLE (output)) & 1)
- write_gchar (output, 0, "pad byte");
+ if (g_seekable_tell (G_SEEKABLE (output)) & 1)
+ write_gchar (output, 0, "pad byte");
+ }
}
-
/* --------------- Write Guides --------------- */
if (gimp_image_find_next_guide (image, 0))
{
gint n_guides = 0;
- gint guide_id =0;
+ gint guide_id = 0;
/* Count the guides */
while ((guide_id = gimp_image_find_next_guide(image, guide_id)))
@@ -914,6 +936,14 @@ save_resources (GOutputStream *output,
if (! export_duotone)
profile = gimp_image_get_effective_color_profile (image);
+ if (export_cmyk)
+ {
+ profile = gimp_image_get_simulation_profile (image);
+
+ if (! gimp_color_profile_is_cmyk (profile))
+ g_object_unref (profile);
+ }
+
if (profile)
{
const guint8 *icc_data;
@@ -1038,7 +1068,8 @@ get_compress_channel_data (guchar *channel_data,
static void
save_layer_and_mask (GOutputStream *output,
- GimpImage *image)
+ GimpImage *image,
+ gboolean export_cmyk)
{
gint i,j;
gint idChannel;
@@ -1152,6 +1183,15 @@ save_layer_and_mask (GOutputStream *output,
gimp_drawable_has_alpha (GIMP_DRAWABLE (psd_layer->layer)),
hasMask);
+ /* Manually set channels to 4 or 5 when export as CMYK;
+ * Can be removed once CMYK channels are accessible in GIMP
+ */
+ if (export_cmyk)
+ {
+ nChannelsLayer =
+ gimp_drawable_has_alpha (GIMP_DRAWABLE (psd_layer->layer)) ?
+ 5 : 4;
+ }
write_gint16 (output, nChannelsLayer, "Number channels in the layer");
IFDBG(1) g_debug ("\t\tNumber of channels: %d", nChannelsLayer);
@@ -1352,8 +1392,8 @@ save_layer_and_mask (GOutputStream *output,
gimp_progress_update ((PSDImageData.nLayers - i - 1.0) / (PSDImageData.nLayers + 1.0));
IFDBG(1) g_debug ("\t\tWriting pixel data for layer slot %d", i-1);
- write_pixel_data (output, GIMP_DRAWABLE (psd_layer->layer), ChannelLengthPos[i-1], 0,
- psd_layer->type != PSD_LAYER_TYPE_GROUP_END);
+ write_pixel_data (output, image, GIMP_DRAWABLE (psd_layer->layer), ChannelLengthPos[i-1], 0,
+ psd_layer->type != PSD_LAYER_TYPE_GROUP_END, export_cmyk);
g_free (ChannelLengthPos[i-1]);
}
@@ -1387,28 +1427,33 @@ save_layer_and_mask (GOutputStream *output,
static void
write_pixel_data (GOutputStream *output,
+ GimpImage *image,
GimpDrawable *drawable,
goffset *ChanLenPosition,
goffset ltable_offset,
- gboolean write_mask)
+ gboolean write_mask,
+ gboolean export_cmyk)
{
- GeglBuffer *buffer = gimp_drawable_get_buffer (drawable);
- const Babl *format;
- GimpLayerMask *mask;
- gint32 tile_height = gimp_tile_height ();
- gint32 height = gegl_buffer_get_height (buffer);
- gint32 width = gegl_buffer_get_width (buffer);
- gint32 bytes;
- gint32 components;
- gint32 bpc;
- gint32 colors;
- gint32 y;
- gsize len; /* Length of compressed data */
- gint16 *LengthsTable; /* Lengths of every compressed row */
- guchar *rledata; /* Compressed data from a region */
- guchar *data; /* Temporary copy of pixel data */
- goffset length_table_pos; /* position in file of the length table */
- int i, j;
+ GeglBuffer *buffer = gimp_drawable_get_buffer (drawable);
+ const Babl *format;
+ const Babl *space = NULL;
+ const Babl *type;
+ GimpColorProfile *profile;
+ GimpLayerMask *mask;
+ gint32 tile_height = gimp_tile_height ();
+ gint32 height = gegl_buffer_get_height (buffer);
+ gint32 width = gegl_buffer_get_width (buffer);
+ gint32 bytes;
+ gint32 components;
+ gint32 bpc;
+ gint32 colors;
+ gint32 y;
+ gsize len; /* Length of compressed data */
+ gint16 *LengthsTable; /* Lengths of every compressed row */
+ guchar *rledata; /* Compressed data from a region */
+ guchar *data; /* Temporary copy of pixel data */
+ goffset length_table_pos; /* position in file of the length table */
+ int i, j;
IFDBG(1) g_debug ("Function: write_pixel_data, drw %d, lto %" G_GOFFSET_FORMAT,
gimp_item_get_id (GIMP_ITEM (drawable)), ltable_offset);
@@ -1430,6 +1475,43 @@ write_pixel_data (GOutputStream *output,
else
format = get_pixel_format (drawable);
+ if (export_cmyk && ! gimp_item_is_channel (GIMP_ITEM (drawable)))
+ {
+ profile = gimp_image_get_simulation_profile (image);
+ if (profile && gimp_color_profile_is_cmyk (profile))
+ space = gimp_color_profile_get_space (profile,
+ gimp_image_get_simulation_intent (image),
+ NULL);
+ if (profile)
+ g_object_unref (profile);
+
+ if (get_bpc (image) == 1)
+ type = babl_type ("u8");
+ else
+ type = babl_type ("u16");
+
+ if (gimp_drawable_has_alpha (drawable))
+ format = babl_format_new (babl_model ("cmykA"),
+ type,
+ babl_component ("cyan"),
+ babl_component ("magenta"),
+ babl_component ("yellow"),
+ babl_component ("key"),
+ babl_component ("A"),
+ NULL);
+ else
+ format = babl_format_new (babl_model ("cmyk"),
+ type,
+ babl_component ("cyan"),
+ babl_component ("magenta"),
+ babl_component ("yellow"),
+ babl_component ("key"),
+ NULL);
+
+ format = babl_format_with_space (babl_format_get_encoding (format),
+ space);
+ }
+
bytes = babl_format_get_bytes_per_pixel (format);
components = babl_format_get_n_components (format);
bpc = bytes / components;
@@ -1635,7 +1717,8 @@ write_pixel_data (GOutputStream *output,
static void
save_data (GOutputStream *output,
- GimpImage *image)
+ GimpImage *image,
+ gboolean export_cmyk)
{
GList *iter;
gint ChanCount;
@@ -1664,8 +1747,8 @@ save_data (GOutputStream *output,
write_gint16 (output, 0, "junk line lengths");
IFDBG(1) g_debug ("\t\tWriting compressed image data");
- write_pixel_data (output, GIMP_DRAWABLE (PSDImageData.merged_layer),
- NULL, offset, FALSE);
+ write_pixel_data (output, image, GIMP_DRAWABLE (PSDImageData.merged_layer),
+ NULL, offset, FALSE, export_cmyk);
chan = nChansLayer (PSDImageData.baseType,
gimp_drawable_has_alpha (GIMP_DRAWABLE (PSDImageData.merged_layer)), 0);
@@ -1673,8 +1756,9 @@ save_data (GOutputStream *output,
for (iter = PSDImageData.lChannels; iter; iter = g_list_next (iter))
{
IFDBG(1) g_debug ("\t\tWriting compressed channel data for channel %d", i);
- write_pixel_data (output, iter->data, NULL,
- offset + 2*imageHeight*chan, FALSE); //check how imgs are channels here
+ write_pixel_data (output, image, iter->data, NULL,
+ offset + 2*imageHeight*chan, FALSE,
+ export_cmyk); //check how imgs are channels here
chan++;
}
}
@@ -1786,15 +1870,20 @@ save_image (GFile *file,
GeglBuffer *buffer;
GList *iter;
GError *local_error = NULL;
- GimpParasite *parasite = NULL;
+ GimpParasite *parasite = NULL;
+ gboolean config_cmyk;
gboolean config_duotone;
g_object_get (config,
+ "cmyk", &config_cmyk,
"duotone", &config_duotone,
NULL);
IFDBG(1) g_debug ("Function: save_image");
+ if (config_cmyk)
+ config_duotone = FALSE;
+
if (gimp_image_get_width (image) > 30000 ||
gimp_image_get_height (image) > 30000)
{
@@ -1876,20 +1965,20 @@ save_image (GFile *file,
IFDBG(1) g_debug ("\tFile '%s' has been opened",
gimp_file_get_utf8_name (file));
- save_header (output, image, config_duotone);
+ save_header (output, image, config_cmyk, config_duotone);
save_color_mode_data (output, image, config_duotone);
- save_resources (output, image, config_duotone);
+ save_resources (output, image, config_cmyk, config_duotone);
/* PSD format does not support layers in indexed images */
if (PSDImageData.baseType == GIMP_INDEXED)
write_gint32 (output, 0, "layers info section length");
else
- save_layer_and_mask (output, image);
+ save_layer_and_mask (output, image, config_cmyk);
/* If this is an indexed image, write now channel and layer info */
- save_data (output, image);
+ save_data (output, image, config_cmyk);
/* Delete merged image now */
@@ -2079,33 +2168,97 @@ save_dialog (GimpImage *image,
GimpProcedure *procedure,
GObject *config)
{
- GtkWidget *dialog;
- GtkWidget *duotone_notice;
- gboolean run;
+ GtkWidget *dialog;
+ GtkWidget *duotone_notice;
+ GtkWidget *profile_label;
+ GimpColorProfile *cmyk_profile;
+ GimpParasite *parasite = NULL;
+ gboolean has_duotone_data = FALSE;
+ gboolean run;
dialog = gimp_procedure_dialog_new (procedure,
GIMP_PROCEDURE_CONFIG (config),
_("Export Image as PSD"));
- /* Profile label */
- duotone_notice = gimp_procedure_dialog_get_label (GIMP_PROCEDURE_DIALOG (dialog),
- "duotone-notice",
- _("Duotone color space information "
- "from the original\nimported image "
- "will be used."));
- gtk_label_set_xalign (GTK_LABEL (duotone_notice), 0.0);
- gtk_label_set_ellipsize (GTK_LABEL (duotone_notice), PANGO_ELLIPSIZE_END);
- gimp_label_set_attributes (GTK_LABEL (duotone_notice),
+ /* CMYK profile label. */
+ profile_label = gimp_procedure_dialog_get_label (GIMP_PROCEDURE_DIALOG (dialog),
+ "profile-label", _("No soft-proofing profile"));
+ gtk_label_set_xalign (GTK_LABEL (profile_label), 0.0);
+ gtk_label_set_ellipsize (GTK_LABEL (profile_label), PANGO_ELLIPSIZE_END);
+ gimp_label_set_attributes (GTK_LABEL (profile_label),
PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC,
-1);
-
+ gimp_help_set_help_data (profile_label,
+ _("Name of the color profile used for CMYK export."), NULL);
gimp_procedure_dialog_fill_frame (GIMP_PROCEDURE_DIALOG (dialog),
- "duotone-frame", "duotone", FALSE,
- "duotone-notice");
+ "cmyk-frame", "cmyk", FALSE,
+ "profile-label");
+ cmyk_profile = gimp_image_get_simulation_profile (image);
+ if (cmyk_profile)
+ {
+ if (gimp_color_profile_is_cmyk (cmyk_profile))
+ {
+ gchar *label_text;
+
+ label_text = g_strdup_printf (_("Profile: %s"),
+ gimp_color_profile_get_label (cmyk_profile));
+ gtk_label_set_text (GTK_LABEL (profile_label), label_text);
+ gimp_label_set_attributes (GTK_LABEL (profile_label),
+ PANGO_ATTR_STYLE, PANGO_STYLE_NORMAL,
+ -1);
+ g_free (label_text);
+ }
+ g_object_unref (cmyk_profile);
+ }
- gimp_procedure_dialog_fill (GIMP_PROCEDURE_DIALOG (dialog),
- "duotone-frame",
- NULL);
+ /* Only show dialog if image is grayscale and duotone color space
+ * information was attached from the original image imported
+ */
+ parasite = gimp_image_get_parasite (image, PSD_PARASITE_DUOTONE_DATA);
+ if (parasite)
+ {
+ if (gimp_image_get_base_type (image) == GIMP_GRAY)
+ {
+ has_duotone_data = TRUE;
+
+ /* Duotone Option label */
+ duotone_notice = gimp_procedure_dialog_get_label (GIMP_PROCEDURE_DIALOG (dialog),
+ "duotone-notice",
+ _("Duotone color space information "
+ "from the original\nimported image "
+ "will be used."));
+ gtk_label_set_xalign (GTK_LABEL (duotone_notice), 0.0);
+ gtk_label_set_ellipsize (GTK_LABEL (duotone_notice), PANGO_ELLIPSIZE_END);
+ gimp_label_set_attributes (GTK_LABEL (duotone_notice),
+ PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC,
+ -1);
+
+ gimp_procedure_dialog_fill_frame (GIMP_PROCEDURE_DIALOG (dialog),
+ "duotone-frame", "duotone", FALSE,
+ "duotone-notice");
+
+ /* Prevent you from setting both Duotone and CMYK exports */
+ gimp_procedure_dialog_set_sensitive (GIMP_PROCEDURE_DIALOG (dialog),
+ "duotone",
+ TRUE, config, "cmyk", TRUE);
+ gimp_procedure_dialog_set_sensitive (GIMP_PROCEDURE_DIALOG (dialog),
+ "cmyk",
+ TRUE, config, "duotone", TRUE);
+
+ }
+
+ gimp_parasite_free (parasite);
+ }
+
+ if (has_duotone_data)
+ gimp_procedure_dialog_fill (GIMP_PROCEDURE_DIALOG (dialog),
+ "cmyk-frame",
+ "duotone-frame",
+ NULL);
+ else
+ gimp_procedure_dialog_fill (GIMP_PROCEDURE_DIALOG (dialog),
+ "cmyk-frame",
+ NULL);
gtk_widget_show (dialog);
@@ -2114,4 +2267,4 @@ save_dialog (GimpImage *image,
gtk_widget_destroy (dialog);
return run;
-}
+}
\ No newline at end of file
diff --git a/plug-ins/file-psd/psd.c b/plug-ins/file-psd/psd.c
index 14e63872ac..2e408a9748 100644
--- a/plug-ins/file-psd/psd.c
+++ b/plug-ins/file-psd/psd.c
@@ -224,6 +224,12 @@ psd_create_procedure (GimpPlugIn *plug_in,
gimp_file_procedure_set_extensions (GIMP_FILE_PROCEDURE (procedure),
"psd");
+ GIMP_PROC_ARG_BOOLEAN (procedure, "cmyk",
+ "Export as _CMYK",
+ "Export a CMYK PSD image using the soft-proofing color profile",
+ FALSE,
+ G_PARAM_READWRITE);
+
GIMP_PROC_ARG_BOOLEAN (procedure, "duotone",
"Export as _Duotone",
"Export as a Duotone PSD file if Duotone color space information "
@@ -360,17 +366,14 @@ psd_save (GimpProcedure *procedure,
GimpProcedureConfig *config;
GimpPDBStatusType status = GIMP_PDB_SUCCESS;
GimpMetadata *metadata;
- GimpMetadataSaveFlags metadata_flags;
GimpExportReturn export = GIMP_EXPORT_IGNORE;
- GimpParasite *parasite = NULL;
GError *error = NULL;
gegl_init (NULL, NULL);
config = gimp_procedure_create_config (procedure);
- metadata = gimp_image_metadata_save_prepare (image,
- "image/x-psd",
- &metadata_flags);
+ metadata = gimp_procedure_config_begin_export (config, image, run_mode,
+ args, "image/x-psd");
switch (run_mode)
{
@@ -397,47 +400,29 @@ psd_save (GimpProcedure *procedure,
if (run_mode == GIMP_RUN_INTERACTIVE)
{
- /* Only show dialog if image is grayscale and duotone color space
- * information was attached from the original image imported
- */
- parasite = gimp_image_get_parasite (image, PSD_PARASITE_DUOTONE_DATA);
- if (parasite)
- {
- if (gimp_image_get_base_type (image) == GIMP_GRAY)
- {
- if (! save_dialog (image, procedure, G_OBJECT (config)))
- return gimp_procedure_new_return_values (procedure, GIMP_PDB_CANCEL,
- NULL);
- }
- gimp_parasite_free (parasite);
- }
+ if (! save_dialog (image, procedure, G_OBJECT (config)))
+ return gimp_procedure_new_return_values (procedure, GIMP_PDB_CANCEL,
+ NULL);
}
if (save_image (file, image, G_OBJECT (config), &error))
{
if (metadata)
- {
- gimp_metadata_set_bits_per_sample (metadata, 8);
-
- gimp_image_metadata_save_finish (image,
- "image/x-psd",
- metadata, metadata_flags,
- file, NULL);
- }
+ gimp_metadata_set_bits_per_sample (metadata, 8);
}
else
{
status = GIMP_PDB_EXECUTION_ERROR;
}
+ gimp_procedure_config_end_export (config, image, file, status);
+ g_object_unref (config);
+
if (export == GIMP_EXPORT_EXPORT)
{
gimp_image_delete (image);
g_free (drawables);
}
- if (metadata)
- g_object_unref (metadata);
-
return gimp_procedure_new_return_values (procedure, status, error);
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]