目前我正在使用:
os.chdir(album.path)
images = glob.glob('*.*')
# thumbs size
size = 80,80
for image in images:
#create thumb
file, ext = os.path.splitext(image)
im = Image.open(os.path.join(album.path,image))
im.thumbnail(size, Image.ANTIALIAS)
thumb_path = os.path.join(album.path, 'thumbs', file + ".thumb" + ".jpeg")
im.save(thumb_path)
虽然这样可行,但我最终会得到不同尺寸的图像(有些是肖像,有些是风景),但我希望所有图像都具有精确的尺寸。也许是一种明智的裁剪?
更新
我不介意裁剪一小部分图像。当我说合理的裁剪时,我的意思是这样的algorythm:
if image is portrait:
make width 80px
crop the height (will be more than 80px)
else if image is landscape:
make height 80px
crop the width to 80px (will be more than 80px)
答案 0 :(得分:18)
这是我对图像进行填充适合的尝试:
#!/usr/bin/env python
from PIL import Image, ImageChops
F_IN = "/path/to/image_in.jpg"
F_OUT = "/path/to/image_out.jpg"
size = (80,80)
image = Image.open(F_IN)
image.thumbnail(size, Image.ANTIALIAS)
image_size = image.size
thumb = image.crop( (0, 0, size[0], size[1]) )
offset_x = max( (size[0] - image_size[0]) / 2, 0 )
offset_y = max( (size[1] - image_size[1]) / 2, 0 )
thumb = ImageChops.offset(thumb, offset_x, offset_y)
thumb.save(F_OUT)
它首先使用缩略图操作将图像降低到原始范围内并保留方面。然后将它裁剪回来以实际填充边界的大小(因为除非原始图像是正方形,现在它将更小),并且我们找到适当的偏移以使图像居中。图像偏移到中心,因此您最终得到黑色填充但没有图像裁剪。
除非您能够在合适的中心裁剪上做出真正明智的猜测,而不会丢失边缘可能出现的重要图像数据,否则填充合身的方法会更好。
<强>更新强>
这是一个可以进行中心裁剪或打击垫适合的版本。
#!/usr/bin/env python
from PIL import Image, ImageChops, ImageOps
def makeThumb(f_in, f_out, size=(80,80), pad=False):
image = Image.open(f_in)
image.thumbnail(size, Image.ANTIALIAS)
image_size = image.size
if pad:
thumb = image.crop( (0, 0, size[0], size[1]) )
offset_x = max( (size[0] - image_size[0]) / 2, 0 )
offset_y = max( (size[1] - image_size[1]) / 2, 0 )
thumb = ImageChops.offset(thumb, offset_x, offset_y)
else:
thumb = ImageOps.fit(image, size, Image.ANTIALIAS, (0.5, 0.5))
thumb.save(f_out)
source = "/path/to/source/image.JPG"
makeThumb(source, "/path/to/source/image_padded.JPG", pad=True)
makeThumb(source, "/path/to/source/image_centerCropped.JPG", pad=False)
答案 1 :(得分:2)
显然,您需要裁剪或填充图像。您可以执行以下操作,根据缩略图的宽高比(未经测试)获得最大居中作物:
aspect = lambda size: float(size[0]) / float(size[1])
sa = aspect(size)
if aspect(im.size) > sa:
width = int(sa * im.size[1])
left = (im.size[0] - width) / 2
im = im.crop((left, 0, left + width, im.size[1]))
else:
height = int(im.size[0] / sa)
top = (im.size[1] - height) / 2
im = im.crop((0, top, im.size[0], top + height))
im.thumbnail(size, Image.ANTIALIAS)
答案 2 :(得分:0)
如果您使用简易缩略图,则需要将crop
设置为True
并将upscale
设置为True
以便始终填充空间(具有完全相同的尺寸)。
例如:使image_2适合image_1尺寸:
thumbnailer = get_thumbnailer(image_2)
thumbnail = thumbnailer.generate_thumbnail(thumbnail_options={
'crop': True,
'upscale': True,
'size': image_1.size
})
image_2 = thumbnail.image