2002-10-20 14:25:16 +00:00
|
|
|
#
|
|
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
|
|
#
|
2006-03-02 23:37:16 +00:00
|
|
|
# Copyright (C) 2000-2006 Donald N. Allingham
|
2011-05-08 03:29:36 +00:00
|
|
|
# Copyright (C) 2011 Adam Stein <adam@csh.rit.edu>
|
2002-10-20 14:25:16 +00:00
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
#
|
2007-09-10 22:14:33 +00:00
|
|
|
# This program is distributed in the hope that it will be useful,
|
2002-10-20 14:25:16 +00:00
|
|
|
# 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
|
|
|
|
#
|
|
|
|
|
2011-10-23 03:13:50 +00:00
|
|
|
# $Id$
|
2006-03-21 19:11:32 +00:00
|
|
|
|
2007-09-10 22:14:33 +00:00
|
|
|
"""
|
|
|
|
Image manipulation routines.
|
|
|
|
"""
|
|
|
|
|
2006-03-21 19:11:32 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Standard python modules
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2002-10-20 14:25:16 +00:00
|
|
|
import os
|
2006-03-29 03:21:29 +00:00
|
|
|
import tempfile
|
2006-03-21 19:11:32 +00:00
|
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# GTK/Gnome modules
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2002-10-20 14:25:16 +00:00
|
|
|
|
2010-10-09 12:33:47 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Gramps modules
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
import Utils
|
|
|
|
|
2006-03-21 19:11:32 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
2007-09-11 03:51:35 +00:00
|
|
|
# resize_to_jpeg
|
2006-03-21 19:11:32 +00:00
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2011-07-12 03:30:58 +00:00
|
|
|
def resize_to_jpeg(source, destination, width, height, crop=None):
|
2007-09-11 03:51:35 +00:00
|
|
|
"""
|
2008-02-24 13:55:55 +00:00
|
|
|
Create the destination, derived from the source, resizing it to the
|
2007-09-11 03:51:35 +00:00
|
|
|
specified size, while converting to JPEG.
|
|
|
|
|
2009-06-24 21:56:07 +00:00
|
|
|
:param source: source image file, in any format that gtk recognizes
|
|
|
|
:type source: unicode
|
|
|
|
:param destination: destination image file, output written in jpeg format
|
|
|
|
:type destination: unicode
|
|
|
|
:param width: desired width of the destination image
|
|
|
|
:type width: int
|
|
|
|
:param height: desired height of the destination image
|
|
|
|
:type height: int
|
2011-07-12 03:30:58 +00:00
|
|
|
:param crop: cropping coordinates
|
|
|
|
:type crop: array of integers ([start_x, start_y, end_x, end_y])
|
2007-09-11 03:51:35 +00:00
|
|
|
"""
|
2009-11-16 22:39:54 +00:00
|
|
|
import gtk
|
2011-07-12 03:30:58 +00:00
|
|
|
|
2007-09-11 03:51:35 +00:00
|
|
|
img = gtk.gdk.pixbuf_new_from_file(source)
|
2011-07-12 03:30:58 +00:00
|
|
|
|
|
|
|
if crop:
|
|
|
|
# Gramps cropping coorinates are [0, 100], so we need to convert to pixels
|
|
|
|
start_x = int((crop[0]/100.0)*img.get_width())
|
|
|
|
start_y = int((crop[1]/100.0)*img.get_height())
|
|
|
|
end_x = int((crop[2]/100.0)*img.get_width())
|
|
|
|
end_y = int((crop[3]/100.0)*img.get_height())
|
|
|
|
|
|
|
|
img = img.subpixbuf(start_x, start_y, end_x-start_x, end_y-start_y)
|
|
|
|
|
|
|
|
# Need to keep the ratio intact, otherwise scaled images look stretched
|
|
|
|
# if the dimensions aren't close in size
|
|
|
|
(width, height) = image_actual_size(width, height, img.get_width(), img.get_height())
|
|
|
|
|
|
|
|
scaled = img.scale_simple(int(width), int(height), gtk.gdk.INTERP_BILINEAR)
|
2007-09-11 03:51:35 +00:00
|
|
|
scaled.save(destination, 'jpeg')
|
2005-12-06 06:38:09 +00:00
|
|
|
|
2011-05-08 03:29:36 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# image_dpi
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
def image_dpi(source):
|
|
|
|
"""
|
|
|
|
Return the dpi found in the image header. None is returned if no dpi attribute
|
|
|
|
is available.
|
|
|
|
|
|
|
|
:param source: source image file, in any format that PIL recognizes
|
|
|
|
:type source: unicode
|
|
|
|
:rtype: int
|
|
|
|
:returns: the DPI setting in the image header
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
import PIL.Image
|
|
|
|
except ImportError:
|
|
|
|
import sys
|
|
|
|
print >> sys.stderr, _("WARNING: PIL module not loaded. "
|
|
|
|
"Image cropping in report files will not be available.")
|
|
|
|
|
|
|
|
dpi = None
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
img = PIL.Image.open(source)
|
|
|
|
except IOError:
|
|
|
|
dpi = None
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
dpi = img.info["dpi"]
|
2012-02-17 14:20:01 +00:00
|
|
|
except (AttributeError, KeyError):
|
2011-05-08 03:29:36 +00:00
|
|
|
dpi = None
|
|
|
|
|
|
|
|
return dpi
|
|
|
|
|
2006-03-21 19:11:32 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
2007-09-11 03:51:35 +00:00
|
|
|
# image_size
|
2006-03-21 19:11:32 +00:00
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2007-09-11 03:51:35 +00:00
|
|
|
def image_size(source):
|
2007-09-10 22:14:33 +00:00
|
|
|
"""
|
2008-02-24 13:55:55 +00:00
|
|
|
Return the width and size of the specified image.
|
2005-12-06 06:38:09 +00:00
|
|
|
|
2009-06-24 21:56:07 +00:00
|
|
|
:param source: source image file, in any format that gtk recongizes
|
|
|
|
:type source: unicode
|
|
|
|
:rtype: tuple(int, int)
|
|
|
|
:returns: a tuple consisting of the width and height
|
2007-09-11 03:51:35 +00:00
|
|
|
"""
|
2009-11-16 22:39:54 +00:00
|
|
|
import gtk
|
|
|
|
import gobject
|
2005-02-24 00:25:34 +00:00
|
|
|
try:
|
2007-09-11 03:51:35 +00:00
|
|
|
img = gtk.gdk.pixbuf_new_from_file(source)
|
2007-09-11 04:13:20 +00:00
|
|
|
width = img.get_width()
|
|
|
|
height = img.get_height()
|
2007-09-11 03:51:35 +00:00
|
|
|
except gobject.GError:
|
|
|
|
width = 0
|
|
|
|
height = 0
|
|
|
|
return (width, height)
|
2005-08-18 05:58:28 +00:00
|
|
|
|
2011-05-08 03:29:36 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
2011-05-13 03:58:41 +00:00
|
|
|
# image_actual_size
|
2011-05-08 03:29:36 +00:00
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
def image_actual_size(x_cm, y_cm, x, y):
|
|
|
|
"""
|
|
|
|
Calculate what the actual width & height of the image should be.
|
|
|
|
|
|
|
|
:param x_cm: width in centimeters
|
|
|
|
:type source: int
|
|
|
|
:param y_cm: height in centimeters
|
|
|
|
:type source: int
|
|
|
|
:param x: desired width in pixels
|
|
|
|
:type source: int
|
|
|
|
:param y: desired height in pixels
|
|
|
|
:type source: int
|
|
|
|
:rtype: tuple(int, int)
|
|
|
|
:returns: a tuple consisting of the width and height in centimeters
|
|
|
|
"""
|
|
|
|
|
|
|
|
ratio = float(x_cm)*float(y)/(float(y_cm)*float(x))
|
|
|
|
|
|
|
|
if ratio < 1:
|
|
|
|
act_width = x_cm
|
|
|
|
act_height = y_cm*ratio
|
|
|
|
else:
|
|
|
|
act_height = y_cm
|
|
|
|
act_width = x_cm/ratio
|
|
|
|
|
|
|
|
return (act_width, act_height)
|
|
|
|
|
2011-09-02 01:32:03 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# resize_to_buffer
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
def resize_to_buffer(source, size, crop=None):
|
|
|
|
"""
|
|
|
|
Loads the image and resizes it. Instead of saving the file, the data
|
|
|
|
is returned in a buffer.
|
|
|
|
|
|
|
|
:param source: source image file, in any format that gtk recognizes
|
|
|
|
:type source: unicode
|
|
|
|
:param size: desired size of the destination image ([width, height])
|
|
|
|
:type size: list
|
|
|
|
:param crop: cropping coordinates
|
|
|
|
:type crop: array of integers ([start_x, start_y, end_x, end_y])
|
|
|
|
:rtype: buffer of data
|
|
|
|
:returns: raw data
|
|
|
|
"""
|
|
|
|
import gtk
|
|
|
|
img = gtk.gdk.pixbuf_new_from_file(source)
|
|
|
|
|
|
|
|
if crop:
|
|
|
|
# Gramps cropping coorinates are [0, 100], so we need to convert to pixels
|
|
|
|
start_x = int((crop[0]/100.0)*img.get_width())
|
|
|
|
start_y = int((crop[1]/100.0)*img.get_height())
|
|
|
|
end_x = int((crop[2]/100.0)*img.get_width())
|
|
|
|
end_y = int((crop[3]/100.0)*img.get_height())
|
|
|
|
|
|
|
|
img = img.subpixbuf(start_x, start_y, end_x-start_x, end_y-start_y)
|
|
|
|
|
|
|
|
# Need to keep the ratio intact, otherwise scaled images look stretched
|
|
|
|
# if the dimensions aren't close in size
|
|
|
|
(size[0], size[1]) = image_actual_size(size[0], size[1], img.get_width(), img.get_height())
|
|
|
|
|
|
|
|
scaled = img.scale_simple(int(size[0]), int(size[1]), gtk.gdk.INTERP_BILINEAR)
|
|
|
|
|
|
|
|
return scaled
|
|
|
|
|
2007-09-11 03:51:35 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# resize_to_jpeg_buffer
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2011-07-27 03:26:12 +00:00
|
|
|
def resize_to_jpeg_buffer(source, size, crop=None):
|
2007-09-11 03:51:35 +00:00
|
|
|
"""
|
|
|
|
Loads the image, converting the file to JPEG, and resizing it. Instead of
|
|
|
|
saving the file, the data is returned in a buffer.
|
|
|
|
|
2009-06-24 21:56:07 +00:00
|
|
|
:param source: source image file, in any format that gtk recognizes
|
|
|
|
:type source: unicode
|
2011-07-27 03:26:12 +00:00
|
|
|
:param size: desired size of the destination image ([width, height])
|
|
|
|
:type size: list
|
|
|
|
:param crop: cropping coordinates
|
|
|
|
:type crop: array of integers ([start_x, start_y, end_x, end_y])
|
2009-06-24 21:56:07 +00:00
|
|
|
:rtype: buffer of data
|
|
|
|
:returns: jpeg image as raw data
|
2007-09-11 03:51:35 +00:00
|
|
|
"""
|
2009-11-16 22:39:54 +00:00
|
|
|
import gtk
|
2007-09-11 04:13:20 +00:00
|
|
|
filed, dest = tempfile.mkstemp()
|
2007-09-11 03:51:35 +00:00
|
|
|
img = gtk.gdk.pixbuf_new_from_file(source)
|
2011-07-27 03:26:12 +00:00
|
|
|
|
|
|
|
if crop:
|
|
|
|
# Gramps cropping coorinates are [0, 100], so we need to convert to pixels
|
|
|
|
start_x = int((crop[0]/100.0)*img.get_width())
|
|
|
|
start_y = int((crop[1]/100.0)*img.get_height())
|
|
|
|
end_x = int((crop[2]/100.0)*img.get_width())
|
|
|
|
end_y = int((crop[3]/100.0)*img.get_height())
|
|
|
|
|
|
|
|
img = img.subpixbuf(start_x, start_y, end_x-start_x, end_y-start_y)
|
|
|
|
|
|
|
|
# Need to keep the ratio intact, otherwise scaled images look stretched
|
|
|
|
# if the dimensions aren't close in size
|
|
|
|
(size[0], size[1]) = image_actual_size(size[0], size[1], img.get_width(), img.get_height())
|
|
|
|
|
|
|
|
scaled = img.scale_simple(int(size[0]), int(size[1]), gtk.gdk.INTERP_BILINEAR)
|
2010-04-16 15:36:31 +00:00
|
|
|
os.close(filed)
|
2010-10-09 12:33:47 +00:00
|
|
|
dest = Utils.get_unicode_path_from_env_var(dest)
|
2007-09-11 03:51:35 +00:00
|
|
|
scaled.save(dest, 'jpeg')
|
2007-09-11 04:13:20 +00:00
|
|
|
ofile = open(dest, mode='rb')
|
|
|
|
data = ofile.read()
|
|
|
|
ofile.close()
|
2007-09-11 03:51:35 +00:00
|
|
|
try:
|
|
|
|
os.unlink(dest)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
return data
|
2006-02-02 05:20:42 +00:00
|
|
|
|