mirror of https://github.com/GNOME/gimp.git
367 lines
10 KiB
C
367 lines
10 KiB
C
/* The GIMP -- an image manipulation program
|
|
* 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 2 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "appenv.h"
|
|
#include "colormaps.h"
|
|
#include "cursorutil.h"
|
|
#include "fileops.h"
|
|
#include "gdisplay_ops.h"
|
|
#include "gimage.h"
|
|
#include "gimpui.h"
|
|
#include "gximage.h"
|
|
#include "menus.h"
|
|
#include "scale.h"
|
|
#include "gimprc.h"
|
|
|
|
#include "config.h"
|
|
#include "libgimp/gimpintl.h"
|
|
|
|
static void gdisplay_close_warning_callback (GtkWidget *widget,
|
|
gboolean close,
|
|
gpointer data);
|
|
static void gdisplay_close_warning_dialog (gchar *image_name,
|
|
GDisplay *gdisp);
|
|
|
|
static GtkWidget *warning_dialog = NULL;
|
|
|
|
/*
|
|
* This file is for operations on the gdisplay object
|
|
*/
|
|
|
|
gulong
|
|
gdisplay_white_pixel (GDisplay *gdisp)
|
|
{
|
|
return g_white_pixel;
|
|
}
|
|
|
|
gulong
|
|
gdisplay_gray_pixel (GDisplay *gdisp)
|
|
{
|
|
return g_gray_pixel;
|
|
}
|
|
|
|
gulong
|
|
gdisplay_black_pixel (GDisplay *gdisp)
|
|
{
|
|
return g_black_pixel;
|
|
}
|
|
|
|
gulong
|
|
gdisplay_color_pixel (GDisplay *gdisp)
|
|
{
|
|
return g_color_pixel;
|
|
}
|
|
|
|
|
|
void
|
|
gdisplay_xserver_resolution (gdouble *xres,
|
|
gdouble *yres)
|
|
{
|
|
gint width, height;
|
|
gint widthMM, heightMM;
|
|
|
|
width = gdk_screen_width ();
|
|
height = gdk_screen_height ();
|
|
|
|
widthMM = gdk_screen_width_mm ();
|
|
heightMM = gdk_screen_height_mm ();
|
|
|
|
/*
|
|
* From xdpyinfo.c:
|
|
*
|
|
* there are 2.54 centimeters to an inch; so there are 25.4 millimeters.
|
|
*
|
|
* dpi = N pixels / (M millimeters / (25.4 millimeters / 1 inch))
|
|
* = N pixels / (M inch / 25.4)
|
|
* = N * 25.4 pixels / M inch
|
|
*/
|
|
|
|
*xres = (width * 25.4) / ((gdouble) widthMM);
|
|
*yres = (height * 25.4) / ((gdouble) heightMM);
|
|
}
|
|
|
|
|
|
void
|
|
gdisplay_new_view (GDisplay *gdisp)
|
|
{
|
|
GDisplay *new_gdisp;
|
|
|
|
/* make sure the image has been fully loaded... */
|
|
if (gdisp->gimage)
|
|
{
|
|
new_gdisp = gdisplay_new (gdisp->gimage, gdisp->scale);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
gdisplay_close_window (GDisplay *gdisp,
|
|
gboolean kill_it)
|
|
{
|
|
/* If the image has been modified, give the user a chance to save
|
|
* it before nuking it--this only applies if its the last view
|
|
* to an image canvas. (a gimage with disp_count = 1)
|
|
*/
|
|
if (! kill_it &&
|
|
(gdisp->gimage->disp_count == 1) &&
|
|
gdisp->gimage->dirty &&
|
|
confirm_on_close)
|
|
{
|
|
gdisplay_close_warning_dialog
|
|
(g_basename (gimage_filename (gdisp->gimage)), gdisp);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_destroy (gdisp->shell);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
gdisplay_shrink_wrap (GDisplay *gdisp)
|
|
{
|
|
/* FIXME: Still not perfect - have to hide and show the window to
|
|
* get it to work. Adding a queue_resize to the shell doesn't
|
|
* seem to help. Anyone have any good ideas here?
|
|
*/
|
|
gint x, y;
|
|
gint disp_width, disp_height;
|
|
gint width, height;
|
|
gint shell_x, shell_y;
|
|
gint shell_width, shell_height;
|
|
gint max_auto_width, max_auto_height;
|
|
gint border_x, border_y;
|
|
gint s_width, s_height;
|
|
|
|
s_width = gdk_screen_width ();
|
|
s_height = gdk_screen_height ();
|
|
|
|
width = SCALEX (gdisp, gdisp->gimage->width);
|
|
height = SCALEY (gdisp, gdisp->gimage->height);
|
|
|
|
disp_width = gdisp->disp_width;
|
|
disp_height = gdisp->disp_height;
|
|
|
|
shell_width = gdisp->shell->allocation.width;
|
|
shell_height = gdisp->shell->allocation.height;
|
|
|
|
border_x = shell_width - disp_width;
|
|
border_y = shell_height - disp_height;
|
|
|
|
max_auto_width = (s_width - border_x) * 0.75;
|
|
max_auto_height = (s_height - border_y) * 0.75;
|
|
|
|
/* If 1) the projected width & height are smaller than screen size, &
|
|
* 2) the current display size isn't already the desired size, expand
|
|
*/
|
|
if (((width + border_x) < s_width || (height + border_y) < s_height) &&
|
|
(width != disp_width || height != disp_height))
|
|
{
|
|
width = ((width + border_x) < s_width) ? width : max_auto_width;
|
|
height = ((height + border_y) < s_height) ? height : max_auto_height;
|
|
|
|
if (width < gdisp->statusarea->requisition.width)
|
|
{
|
|
width = gdisp->statusarea->requisition.width;
|
|
}
|
|
/* I don't know why, but if I don't hide the window, it doesn't work */
|
|
gtk_widget_hide(gdisp->shell);
|
|
gtk_drawing_area_size(GTK_DRAWING_AREA(gdisp->canvas),
|
|
width, height);
|
|
gtk_widget_show(gdisp->shell);
|
|
|
|
/*printf("1w:%d/%d d:%d/%d s:%d/%d b:%d/%d\n",
|
|
width, height,
|
|
disp_width, disp_height,
|
|
shell_width, shell_height,
|
|
border_x, border_y);fflush(stdout);*/
|
|
|
|
gdk_window_get_origin (gdisp->shell->window, &shell_x, &shell_y);
|
|
|
|
shell_width = width + border_x;
|
|
shell_height = height + border_y;
|
|
|
|
x = MIN (shell_x, CLAMP (s_width - shell_width, border_x, s_width));
|
|
y = MIN (shell_y, CLAMP (s_height - shell_height, border_y, s_height));
|
|
|
|
if (x != shell_x || y != shell_y)
|
|
gdk_window_move (gdisp->shell->window, x, y);
|
|
|
|
/* Set the new disp_width and disp_height values */
|
|
gdisp->disp_width = width;
|
|
gdisp->disp_height = height;
|
|
}
|
|
/* If the projected width is greater than current, but less than
|
|
* 3/4 of the screen size, expand automagically
|
|
*/
|
|
else if ((width > disp_width || height > disp_height) &&
|
|
(disp_width < max_auto_width || disp_height < max_auto_height))
|
|
{
|
|
max_auto_width = MIN (max_auto_width, width);
|
|
max_auto_height = MIN (max_auto_height, height);
|
|
|
|
if (width < gdisp->statusarea->requisition.width)
|
|
{
|
|
width = gdisp->statusarea->requisition.width;
|
|
}
|
|
|
|
/* I don't know why, but if I don't hide the window, it doesn't work */
|
|
gtk_widget_hide(gdisp->shell);
|
|
gtk_drawing_area_size(GTK_DRAWING_AREA(gdisp->canvas),
|
|
width, height);
|
|
gtk_widget_show(gdisp->shell);
|
|
|
|
/*printf("2w:%d/%d d:%d/%d s:%d/%d b:%d/%d\n",
|
|
width, height,
|
|
disp_width, disp_height,
|
|
shell_width, shell_height,
|
|
border_x, border_y);fflush(stdout);*/
|
|
|
|
gdk_window_get_origin (gdisp->shell->window, &shell_x, &shell_y);
|
|
|
|
shell_width = width + border_x;
|
|
shell_height = height + border_y;
|
|
|
|
x = MIN (shell_x, CLAMP (s_width - shell_width, border_x, s_width));
|
|
y = MIN (shell_y, CLAMP (s_height - shell_height, border_y, s_height));
|
|
|
|
if (x != shell_x || y != shell_y)
|
|
gdk_window_move (gdisp->shell->window, x, y);
|
|
|
|
/* Set the new disp_width and disp_height values */
|
|
gdisp->disp_width = max_auto_width;
|
|
gdisp->disp_height = max_auto_height;
|
|
}
|
|
/* Otherwise, reexpose by hand to reflect changes */
|
|
else
|
|
gdisplay_expose_full (gdisp);
|
|
|
|
/* If the width or height of the display has changed, recalculate
|
|
* the display offsets...
|
|
*/
|
|
if (disp_width != gdisp->disp_width ||
|
|
disp_height != gdisp->disp_height)
|
|
{
|
|
gdisp->offset_x += (disp_width - gdisp->disp_width) / 2;
|
|
gdisp->offset_y += (disp_height - gdisp->disp_height) / 2;
|
|
bounds_checking (gdisp);
|
|
}
|
|
}
|
|
|
|
|
|
gint
|
|
gdisplay_resize_image (GDisplay *gdisp)
|
|
{
|
|
gint sx, sy;
|
|
gint width, height;
|
|
|
|
/* Calculate the width and height of the new canvas */
|
|
sx = SCALEX (gdisp, gdisp->gimage->width);
|
|
sy = SCALEY (gdisp, gdisp->gimage->height);
|
|
width = MIN (sx, gdisp->disp_width);
|
|
height = MIN (sy, gdisp->disp_height);
|
|
|
|
/* if the new dimensions of the ximage are different than the old...resize */
|
|
if (width != gdisp->disp_width || height != gdisp->disp_height)
|
|
{
|
|
/* adjust the gdisplay offsets -- we need to set them so that the
|
|
* center of our viewport is at the center of the image.
|
|
*/
|
|
gdisp->offset_x = (sx / 2) - (width / 2);
|
|
gdisp->offset_y = (sy / 2) - (height / 2);
|
|
|
|
gdisp->disp_width = width;
|
|
gdisp->disp_height = height;
|
|
|
|
if (GTK_WIDGET_VISIBLE (gdisp->canvas))
|
|
gtk_widget_hide (gdisp->canvas);
|
|
|
|
gtk_widget_set_usize (gdisp->canvas,
|
|
gdisp->disp_width,
|
|
gdisp->disp_height);
|
|
|
|
gtk_widget_show (gdisp->canvas);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/********************************************************
|
|
* Routines to query before closing a dirty image *
|
|
********************************************************/
|
|
|
|
static void
|
|
gdisplay_close_warning_callback (GtkWidget *widget,
|
|
gboolean close,
|
|
gpointer data)
|
|
{
|
|
GDisplay *gdisp;
|
|
|
|
gdisp = (GDisplay *) data;
|
|
|
|
menus_set_sensitive ("<Image>/File/Close", TRUE);
|
|
|
|
if (close)
|
|
gtk_widget_destroy (gdisp->shell);
|
|
}
|
|
|
|
static void
|
|
gdisplay_close_warning_dialog (gchar *image_name,
|
|
GDisplay *gdisp)
|
|
{
|
|
GtkWidget *mbox;
|
|
gchar *warning_buf;
|
|
|
|
/* FIXUP this will raise any prexsisting close dialogs, which can be a
|
|
a bit confusing if you tried to close a new window because you had
|
|
forgotten the old dialog was still around */
|
|
/* If a warning dialog already exists raise the window and get out */
|
|
if (warning_dialog != NULL)
|
|
{
|
|
gdk_window_raise (warning_dialog->window);
|
|
return;
|
|
}
|
|
|
|
menus_set_sensitive ("<Image>/File/Close", FALSE);
|
|
|
|
warning_buf =
|
|
g_strdup_printf (_("Changes were made to %s.\nClose anyway?"), image_name);
|
|
|
|
warning_dialog = mbox =
|
|
gimp_query_boolean_box (image_name,
|
|
gimp_standard_help_func,
|
|
"dialogs/really_close.html",
|
|
FALSE,
|
|
warning_buf,
|
|
_("Close"), _("Cancel"),
|
|
NULL, NULL,
|
|
gdisplay_close_warning_callback,
|
|
gdisp);
|
|
|
|
g_free (warning_buf);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (mbox), "destroy",
|
|
GTK_SIGNAL_FUNC (gtk_widget_destroyed),
|
|
&warning_dialog);
|
|
|
|
gtk_widget_show (mbox);
|
|
}
|