/* 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 "core-types.h"
#include "config/pikadialogconfig.h"
#include "vectors/pikavectors.h"
#include "pika.h"
#include "pikacontainer.h"
#include "pikacontext.h"
#include "pikaguide.h"
#include "pikaimage.h"
#include "pikaimage-flip.h"
#include "pikaimage-metadata.h"
#include "pikaimage-rotate.h"
#include "pikaimage-guides.h"
#include "pikaimage-sample-points.h"
#include "pikaimage-undo.h"
#include "pikaimage-undo-push.h"
#include "pikaitem.h"
#include "pikalayer.h"
#include "pikaobjectqueue.h"
#include "pikaprogress.h"
#include "pikasamplepoint.h"
static void pika_image_rotate_item_offset (PikaImage *image,
PikaRotationType rotate_type,
PikaItem *item,
gint off_x,
gint off_y);
static void pika_image_rotate_guides (PikaImage *image,
PikaRotationType rotate_type);
static void pika_image_rotate_sample_points (PikaImage *image,
PikaRotationType rotate_type);
static void pika_image_metadata_rotate (PikaImage *image,
PikaContext *context,
GExiv2Orientation orientation,
PikaProgress *progress);
/* Public Functions */
void
pika_image_rotate (PikaImage *image,
PikaContext *context,
PikaRotationType rotate_type,
PikaProgress *progress)
{
PikaObjectQueue *queue;
PikaItem *item;
GList *list;
gdouble center_x;
gdouble center_y;
gint new_image_width;
gint new_image_height;
gint previous_image_width;
gint previous_image_height;
gint offset_x;
gint offset_y;
gboolean size_changed;
g_return_if_fail (PIKA_IS_IMAGE (image));
g_return_if_fail (PIKA_IS_CONTEXT (context));
g_return_if_fail (progress == NULL || PIKA_IS_PROGRESS (progress));
previous_image_width = pika_image_get_width (image);
previous_image_height = pika_image_get_height (image);
center_x = previous_image_width / 2.0;
center_y = previous_image_height / 2.0;
/* Resize the image (if needed) */
switch (rotate_type)
{
case PIKA_ROTATE_90:
case PIKA_ROTATE_270:
new_image_width = pika_image_get_height (image);
new_image_height = pika_image_get_width (image);
size_changed = TRUE;
offset_x = (pika_image_get_width (image) - new_image_width) / 2;
offset_y = (pika_image_get_height (image) - new_image_height) / 2;
break;
case PIKA_ROTATE_180:
new_image_width = pika_image_get_width (image);
new_image_height = pika_image_get_height (image);
size_changed = FALSE;
offset_x = 0;
offset_y = 0;
break;
default:
g_return_if_reached ();
return;
}
pika_set_busy (image->pika);
queue = pika_object_queue_new (progress);
progress = PIKA_PROGRESS (queue);
pika_object_queue_push_container (queue, pika_image_get_layers (image));
pika_object_queue_push (queue, pika_image_get_mask (image));
pika_object_queue_push_container (queue, pika_image_get_channels (image));
pika_object_queue_push_container (queue, pika_image_get_vectors (image));
g_object_freeze_notify (G_OBJECT (image));
pika_image_undo_group_start (image, PIKA_UNDO_GROUP_IMAGE_ROTATE, NULL);
/* Rotate all layers, channels (including selection mask), and vectors */
while ((item = pika_object_queue_pop (queue)))
{
gint off_x;
gint off_y;
pika_item_get_offset (item, &off_x, &off_y);
pika_item_rotate (item, context, rotate_type, center_x, center_y, FALSE);
if (PIKA_IS_LAYER (item))
{
pika_image_rotate_item_offset (image, rotate_type, item, off_x, off_y);
}
else
{
pika_item_set_offset (item, 0, 0);
if (PIKA_IS_VECTORS (item))
{
pika_item_set_size (item, new_image_width, new_image_height);
pika_item_translate (item,
(new_image_width - pika_image_get_width (image)) / 2,
(new_image_height - pika_image_get_height (image)) / 2,
FALSE);
}
}
pika_progress_set_value (progress, 1.0);
}
/* Rotate all Guides */
pika_image_rotate_guides (image, rotate_type);
/* Rotate all sample points */
pika_image_rotate_sample_points (image, rotate_type);
/* Resize the image (if needed) */
if (size_changed)
{
gdouble xres;
gdouble yres;
pika_image_undo_push_image_size (image,
NULL,
offset_x,
offset_y,
new_image_width,
new_image_height);
g_object_set (image,
"width", new_image_width,
"height", new_image_height,
NULL);
pika_image_get_resolution (image, &xres, &yres);
if (xres != yres)
pika_image_set_resolution (image, yres, xres);
}
/* Notify guide movements */
for (list = pika_image_get_guides (image);
list;
list = g_list_next (list))
{
pika_image_guide_moved (image, list->data);
}
/* Notify sample point movements */
for (list = pika_image_get_sample_points (image);
list;
list = g_list_next (list))
{
pika_image_sample_point_moved (image, list->data);
}
pika_image_undo_group_end (image);
g_object_unref (queue);
if (size_changed)
pika_image_size_changed_detailed (image,
-offset_x,
-offset_y,
previous_image_width,
previous_image_height);
g_object_thaw_notify (G_OBJECT (image));
pika_unset_busy (image->pika);
}
void
pika_image_import_rotation_metadata (PikaImage *image,
PikaContext *context,
PikaProgress *progress,
gboolean interactive)
{
PikaMetadata *metadata;
g_return_if_fail (PIKA_IS_IMAGE (image));
g_return_if_fail (PIKA_IS_CONTEXT (context));
g_return_if_fail (progress == NULL || PIKA_IS_PROGRESS (progress));
metadata = pika_image_get_metadata (image);
if (metadata)
{
PikaMetadataRotationPolicy policy;
policy = PIKA_DIALOG_CONFIG (image->pika->config)->metadata_rotation_policy;
if (policy == PIKA_METADATA_ROTATION_POLICY_ASK)
{
if (interactive)
{
gboolean dont_ask = FALSE;
policy = pika_query_rotation_policy (image->pika, image,
context, &dont_ask);
if (dont_ask)
{
g_object_set (G_OBJECT (image->pika->config),
"metadata-rotation-policy", policy,
NULL);
}
}
else
{
policy = PIKA_METADATA_ROTATION_POLICY_ROTATE;
}
}
if (policy == PIKA_METADATA_ROTATION_POLICY_ROTATE)
pika_image_metadata_rotate (image, context,
gexiv2_metadata_try_get_orientation (GEXIV2_METADATA (metadata), NULL),
progress);
gexiv2_metadata_try_set_orientation (GEXIV2_METADATA (metadata),
GEXIV2_ORIENTATION_NORMAL,
NULL);
}
}
void
pika_image_apply_metadata_orientation (PikaImage *image,
PikaContext *context,
PikaMetadata *metadata,
PikaProgress *progress)
{
pika_image_metadata_rotate (image, context,
gexiv2_metadata_try_get_orientation (GEXIV2_METADATA (metadata), NULL),
progress);
}
/* Private Functions */
static void
pika_image_rotate_item_offset (PikaImage *image,
PikaRotationType rotate_type,
PikaItem *item,
gint off_x,
gint off_y)
{
gint x = 0;
gint y = 0;
switch (rotate_type)
{
case PIKA_ROTATE_90:
x = pika_image_get_height (image) - off_y - pika_item_get_width (item);
y = off_x;
break;
case PIKA_ROTATE_270:
x = off_y;
y = pika_image_get_width (image) - off_x - pika_item_get_height (item);
break;
case PIKA_ROTATE_180:
return;
default:
g_return_if_reached ();
}
pika_item_get_offset (item, &off_x, &off_y);
x -= off_x;
y -= off_y;
if (x || y)
pika_item_translate (item, x, y, FALSE);
}
static void
pika_image_rotate_guides (PikaImage *image,
PikaRotationType rotate_type)
{
GList *list;
/* Rotate all Guides */
for (list = pika_image_get_guides (image);
list;
list = g_list_next (list))
{
PikaGuide *guide = list->data;
PikaOrientationType orientation = pika_guide_get_orientation (guide);
gint position = pika_guide_get_position (guide);
switch (rotate_type)
{
case PIKA_ROTATE_90:
switch (orientation)
{
case PIKA_ORIENTATION_HORIZONTAL:
pika_image_undo_push_guide (image, NULL, guide);
pika_guide_set_orientation (guide, PIKA_ORIENTATION_VERTICAL);
pika_guide_set_position (guide,
pika_image_get_height (image) - position);
break;
case PIKA_ORIENTATION_VERTICAL:
pika_image_undo_push_guide (image, NULL, guide);
pika_guide_set_orientation (guide, PIKA_ORIENTATION_HORIZONTAL);
break;
default:
break;
}
break;
case PIKA_ROTATE_180:
switch (orientation)
{
case PIKA_ORIENTATION_HORIZONTAL:
pika_image_move_guide (image, guide,
pika_image_get_height (image) - position,
TRUE);
break;
case PIKA_ORIENTATION_VERTICAL:
pika_image_move_guide (image, guide,
pika_image_get_width (image) - position,
TRUE);
break;
default:
break;
}
break;
case PIKA_ROTATE_270:
switch (orientation)
{
case PIKA_ORIENTATION_HORIZONTAL:
pika_image_undo_push_guide (image, NULL, guide);
pika_guide_set_orientation (guide, PIKA_ORIENTATION_VERTICAL);
break;
case PIKA_ORIENTATION_VERTICAL:
pika_image_undo_push_guide (image, NULL, guide);
pika_guide_set_orientation (guide, PIKA_ORIENTATION_HORIZONTAL);
pika_guide_set_position (guide,
pika_image_get_width (image) - position);
break;
default:
break;
}
break;
}
}
}
static void
pika_image_rotate_sample_points (PikaImage *image,
PikaRotationType rotate_type)
{
GList *list;
/* Rotate all sample points */
for (list = pika_image_get_sample_points (image);
list;
list = g_list_next (list))
{
PikaSamplePoint *sample_point = list->data;
gint old_x;
gint old_y;
pika_image_undo_push_sample_point (image, NULL, sample_point);
pika_sample_point_get_position (sample_point, &old_x, &old_y);
switch (rotate_type)
{
case PIKA_ROTATE_90:
pika_sample_point_set_position (sample_point,
pika_image_get_height (image) - old_y,
old_x);
break;
case PIKA_ROTATE_180:
pika_sample_point_set_position (sample_point,
pika_image_get_width (image) - old_x,
pika_image_get_height (image) - old_y);
break;
case PIKA_ROTATE_270:
pika_sample_point_set_position (sample_point,
old_y,
pika_image_get_width (image) - old_x);
break;
}
}
}
static void
pika_image_metadata_rotate (PikaImage *image,
PikaContext *context,
GExiv2Orientation orientation,
PikaProgress *progress)
{
switch (orientation)
{
case GEXIV2_ORIENTATION_UNSPECIFIED:
case GEXIV2_ORIENTATION_NORMAL: /* standard orientation, do nothing */
break;
case GEXIV2_ORIENTATION_HFLIP:
pika_image_flip (image, context, PIKA_ORIENTATION_HORIZONTAL, progress);
break;
case GEXIV2_ORIENTATION_ROT_180:
pika_image_rotate (image, context, PIKA_ROTATE_180, progress);
break;
case GEXIV2_ORIENTATION_VFLIP:
pika_image_flip (image, context, PIKA_ORIENTATION_VERTICAL, progress);
break;
case GEXIV2_ORIENTATION_ROT_90_HFLIP: /* flipped diagonally around '\' */
pika_image_rotate (image, context, PIKA_ROTATE_90, progress);
pika_image_flip (image, context, PIKA_ORIENTATION_HORIZONTAL, progress);
break;
case GEXIV2_ORIENTATION_ROT_90: /* 90 CW */
pika_image_rotate (image, context, PIKA_ROTATE_90, progress);
break;
case GEXIV2_ORIENTATION_ROT_90_VFLIP: /* flipped diagonally around '/' */
pika_image_rotate (image, context, PIKA_ROTATE_90, progress);
pika_image_flip (image, context, PIKA_ORIENTATION_VERTICAL, progress);
break;
case GEXIV2_ORIENTATION_ROT_270: /* 90 CCW */
pika_image_rotate (image, context, PIKA_ROTATE_270, progress);
break;
default: /* shouldn't happen */
break;
}
}