Pillow/PIL/SgiImagePlugin.py

178 lines
4.8 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library.
# $Id$
#
# SGI image file handling
#
# See "The SGI Image File Format (Draft version 0.97)", Paul Haeberli.
# <ftp://ftp.sgi.com/graphics/SGIIMAGESPEC>
#
#
2010-07-31 06:52:47 +04:00
# History:
2017-07-22 10:34:06 +03:00
# 2017-22-07 mb Add RLE decompression
# 2016-16-10 mb Add save method without compression
2010-07-31 06:52:47 +04:00
# 1995-09-10 fl Created
#
# Copyright (c) 2016 by Mickael Bonfill.
2010-07-31 06:52:47 +04:00
# Copyright (c) 2008 by Karsten Hiddemann.
# Copyright (c) 1997 by Secret Labs AB.
# Copyright (c) 1995 by Fredrik Lundh.
#
# See the README file for information on usage and redistribution.
#
from . import Image, ImageFile
from ._binary import i8, o8, i16be as i16
import struct
import os
2010-07-31 06:52:47 +04:00
__version__ = "0.3"
2015-08-25 15:27:18 +03:00
2010-07-31 06:52:47 +04:00
def _accept(prefix):
return len(prefix) >= 2 and i16(prefix) == 474
2010-07-31 06:52:47 +04:00
2014-07-16 22:30:41 +04:00
2010-07-31 06:52:47 +04:00
##
# Image plugin for SGI images.
class SgiImageFile(ImageFile.ImageFile):
format = "SGI"
format_description = "SGI Image File Format"
def _open(self):
# HEAD
2017-07-22 10:34:06 +03:00
offset = 512
s = self.fp.read(offset)
# magic number : 474
2010-07-31 06:52:47 +04:00
if i16(s) != 474:
raise ValueError("Not an SGI image file")
2010-07-31 06:52:47 +04:00
2017-07-22 10:34:06 +03:00
# compression : verbatim or RLE
compression = i8(s[2])
2010-07-31 06:52:47 +04:00
2017-07-22 10:34:06 +03:00
# depth : 1 or 2 bytes (8bits or 16bits)
depth = i8(s[3]) * 8
# dimension : 1, 2 or 3 (depending on xsize, ysize and zsize)
dimension = i16(s[4:])
# xsize : width
xsize = i16(s[6:])
# ysize : height
ysize = i16(s[8:])
2010-07-31 06:52:47 +04:00
2017-07-22 10:34:06 +03:00
# zsize : channels count
zsize = i16(s[10:])
# layout
layout = depth, dimension, zsize
# determine mode from bits/zsize
if layout == (8, 2, 1) or layout == (8, 1, 1):
2010-07-31 06:52:47 +04:00
self.mode = "L"
2017-07-22 10:34:06 +03:00
elif layout == (8, 3, 3):
2010-07-31 06:52:47 +04:00
self.mode = "RGB"
2017-07-22 10:34:06 +03:00
elif layout == (8, 3, 4):
2010-07-31 06:52:47 +04:00
self.mode = "RGBA"
else:
raise ValueError("Unsupported SGI image mode")
2010-07-31 06:52:47 +04:00
2017-07-22 10:34:06 +03:00
self.size = xsize, ysize
# orientation -1 : scanlines begins at the bottom-left corner
orientation = -1
2010-07-31 06:52:47 +04:00
# decoder info
if compression == 0:
2017-07-22 10:34:06 +03:00
pagesize = xsize * ysize * (depth / 8)
2010-07-31 06:52:47 +04:00
self.tile = []
for layer in self.mode:
2014-07-16 22:30:41 +04:00
self.tile.append(
2017-07-22 10:34:06 +03:00
("raw", (0, 0) + self.size,
offset, (layer, 0, orientation)))
2010-07-31 06:52:47 +04:00
offset = offset + pagesize
elif compression == 1:
2017-07-22 10:34:06 +03:00
self.tile = [("sgi_rle", (0, 0) + self.size,
offset, (self.mode, orientation, depth))]
2010-07-31 06:52:47 +04:00
def _save(im, fp, filename):
if im.mode != "RGB" and im.mode != "RGBA" and im.mode != "L":
raise ValueError("Unsupported SGI image mode")
# Flip the image, since the origin of SGI file is the bottom-left corner
im = im.transpose(Image.FLIP_TOP_BOTTOM)
# Define the file as SGI File Format
magicNumber = 474
# Run-Length Encoding Compression - Unsupported at this time
rle = 0
# Byte-per-pixel precision, 1 = 8bits per pixel
bpc = 1
# Number of dimensions (x,y,z)
dim = 3
# X Dimension = width / Y Dimension = height
x, y = im.size
if im.mode == "L" and y == 1:
dim = 1
elif im.mode == "L":
dim = 2
# Z Dimension: Number of channels
z = len(im.mode)
if dim == 1 or dim == 2:
z = 1
# assert we've got the right number of bands.
if len(im.getbands()) != z:
raise ValueError("incorrect number of bands in SGI write: %s vs %s" %
(z, len(im.getbands())))
# Minimum Byte value
pinmin = 0
# Maximum Byte value (255 = 8bits per pixel)
pinmax = 255
2016-12-31 02:10:47 +03:00
# Image name (79 characters max, truncated below in write)
imgName = os.path.splitext(os.path.basename(filename))[0]
if str is not bytes:
imgName = imgName.encode('ascii', 'ignore')
# Standard representation of pixel in the file
colormap = 0
fp.write(struct.pack('>h', magicNumber))
2016-12-31 02:10:47 +03:00
fp.write(o8(rle))
fp.write(o8(bpc))
fp.write(struct.pack('>H', dim))
fp.write(struct.pack('>H', x))
fp.write(struct.pack('>H', y))
fp.write(struct.pack('>H', z))
fp.write(struct.pack('>l', pinmin))
fp.write(struct.pack('>l', pinmax))
2016-12-31 02:10:47 +03:00
2017-04-20 14:14:23 +03:00
fp.write(struct.pack('4s', b'')) # dummy
fp.write(struct.pack('79s', imgName)) # truncates to 79 chars
fp.write(struct.pack('s', b'')) # force null byte after imgname
fp.write(struct.pack('>l', colormap))
2016-12-31 02:10:47 +03:00
2017-04-20 14:14:23 +03:00
fp.write(struct.pack('404s', b'')) # dummy
2016-12-31 02:10:47 +03:00
for channel in im.split():
fp.write(channel.tobytes())
fp.close()
2010-07-31 06:52:47 +04:00
#
# registry
Image.register_open(SgiImageFile.format, SgiImageFile, _accept)
Image.register_save(SgiImageFile.format, _save)
Image.register_mime(SgiImageFile.format, "image/sgi")
Image.register_mime(SgiImageFile.format, "image/rgb")
2016-04-25 07:59:02 +03:00
Image.register_extensions(SgiImageFile.format, [".bw", ".rgb", ".rgba", ".sgi"])
# End of file