2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library.
|
|
|
|
# $Id$
|
|
|
|
#
|
|
|
|
# EPS file handling
|
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 1995-09-01 fl Created (0.1)
|
|
|
|
# 1996-05-18 fl Don't choke on "atend" fields, Ghostscript interface (0.2)
|
|
|
|
# 1996-08-22 fl Don't choke on floating point BoundingBox values
|
|
|
|
# 1996-08-23 fl Handle files from Macintosh (0.3)
|
|
|
|
# 2001-02-17 fl Use 're' instead of 'regex' (Python 2.1) (0.4)
|
|
|
|
# 2003-09-07 fl Check gs.close status (from Federico Di Gregorio) (0.5)
|
2014-08-26 17:47:10 +04:00
|
|
|
# 2014-05-07 e Handling of EPS with binary preview and fixed resolution
|
|
|
|
# resizing
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# Copyright (c) 1997-2003 by Secret Labs AB.
|
|
|
|
# Copyright (c) 1995-2003 by Fredrik Lundh
|
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
|
|
|
|
2012-10-24 07:21:19 +04:00
|
|
|
import io
|
2016-11-24 04:48:56 +03:00
|
|
|
import os
|
2019-07-06 23:40:53 +03:00
|
|
|
import re
|
2019-10-07 16:28:36 +03:00
|
|
|
import subprocess
|
2015-08-25 15:27:18 +03:00
|
|
|
import sys
|
2019-10-07 16:28:36 +03:00
|
|
|
import tempfile
|
2019-07-06 23:40:53 +03:00
|
|
|
|
2017-01-17 16:22:18 +03:00
|
|
|
from . import Image, ImageFile
|
2017-03-03 13:32:31 +03:00
|
|
|
from ._binary import i32le as i32
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
#
|
|
|
|
# --------------------------------------------------------------------
|
|
|
|
|
|
|
|
split = re.compile(r"^%%([^:]*):[ \t]*(.*)[ \t]*$")
|
|
|
|
field = re.compile(r"^%[%!\w]([^:]*)[ \t]*$")
|
|
|
|
|
2013-03-06 21:36:22 +04:00
|
|
|
gs_windows_binary = None
|
2019-03-21 16:28:20 +03:00
|
|
|
if sys.platform.startswith("win"):
|
2013-03-06 21:36:22 +04:00
|
|
|
import shutil
|
2019-03-21 16:28:20 +03:00
|
|
|
|
|
|
|
for binary in ("gswin32c", "gswin64c", "gs"):
|
2019-09-26 15:12:28 +03:00
|
|
|
if shutil.which(binary) is not None:
|
2013-03-06 21:36:22 +04:00
|
|
|
gs_windows_binary = binary
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
gs_windows_binary = False
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-04-05 00:33:54 +04:00
|
|
|
def has_ghostscript():
|
|
|
|
if gs_windows_binary:
|
|
|
|
return True
|
2019-03-21 16:28:20 +03:00
|
|
|
if not sys.platform.startswith("win"):
|
2014-04-05 00:33:54 +04:00
|
|
|
try:
|
2019-10-08 11:24:36 +03:00
|
|
|
subprocess.check_call(["gs", "--version"], stdout=subprocess.DEVNULL)
|
2014-04-05 00:33:54 +04:00
|
|
|
return True
|
|
|
|
except OSError:
|
2018-09-26 14:38:44 +03:00
|
|
|
# No Ghostscript
|
2014-04-05 00:33:54 +04:00
|
|
|
pass
|
|
|
|
return False
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2014-04-05 00:33:54 +04:00
|
|
|
|
2013-11-17 11:26:44 +04:00
|
|
|
def Ghostscript(tile, size, fp, scale=1):
|
2013-03-06 21:36:22 +04:00
|
|
|
"""Render an image using Ghostscript"""
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# Unpack decoder tile
|
|
|
|
decoder, tile, offset, data = tile[0]
|
|
|
|
length, bbox = data
|
2014-08-26 17:47:10 +04:00
|
|
|
|
|
|
|
# Hack to support hi-res rendering
|
2013-11-17 11:26:44 +04:00
|
|
|
scale = int(scale) or 1
|
2014-08-26 17:47:10 +04:00
|
|
|
# orig_size = size
|
|
|
|
# orig_bbox = bbox
|
2013-11-17 11:26:44 +04:00
|
|
|
size = (size[0] * scale, size[1] * scale)
|
2014-08-26 17:47:10 +04:00
|
|
|
# resolution is dependent on bbox and size
|
2019-03-21 16:28:20 +03:00
|
|
|
res = (
|
2020-01-26 17:21:41 +03:00
|
|
|
72.0 * size[0] / (bbox[2] - bbox[0]),
|
|
|
|
72.0 * size[1] / (bbox[3] - bbox[1]),
|
2019-03-21 16:28:20 +03:00
|
|
|
)
|
2013-11-17 11:26:44 +04:00
|
|
|
|
2014-03-15 02:56:41 +04:00
|
|
|
out_fd, outfile = tempfile.mkstemp()
|
|
|
|
os.close(out_fd)
|
2014-09-03 10:09:04 +04:00
|
|
|
|
|
|
|
infile_temp = None
|
2019-03-21 16:28:20 +03:00
|
|
|
if hasattr(fp, "name") and os.path.exists(fp.name):
|
2014-09-03 10:09:04 +04:00
|
|
|
infile = fp.name
|
|
|
|
else:
|
|
|
|
in_fd, infile_temp = tempfile.mkstemp()
|
|
|
|
os.close(in_fd)
|
|
|
|
infile = infile_temp
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2018-09-26 14:38:44 +03:00
|
|
|
# Ignore length and offset!
|
|
|
|
# Ghostscript can read it
|
|
|
|
# Copy whole file to read in Ghostscript
|
2019-03-21 16:28:20 +03:00
|
|
|
with open(infile_temp, "wb") as f:
|
2014-09-03 10:09:04 +04:00
|
|
|
# fetch length of fp
|
2019-01-13 05:05:46 +03:00
|
|
|
fp.seek(0, io.SEEK_END)
|
2014-09-03 10:09:04 +04:00
|
|
|
fsize = fp.tell()
|
|
|
|
# ensure start position
|
|
|
|
# go back
|
|
|
|
fp.seek(0)
|
|
|
|
lengthfile = fsize
|
|
|
|
while lengthfile > 0:
|
2019-03-21 16:28:20 +03:00
|
|
|
s = fp.read(min(lengthfile, 100 * 1024))
|
2014-09-03 10:09:04 +04:00
|
|
|
if not s:
|
|
|
|
break
|
|
|
|
lengthfile -= len(s)
|
|
|
|
f.write(s)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2018-09-26 14:38:44 +03:00
|
|
|
# Build Ghostscript command
|
2019-03-21 16:28:20 +03:00
|
|
|
command = [
|
|
|
|
"gs",
|
|
|
|
"-q", # quiet mode
|
|
|
|
"-g%dx%d" % size, # set output geometry (pixels)
|
|
|
|
"-r%fx%f" % res, # set input DPI (dots per inch)
|
|
|
|
"-dBATCH", # exit after processing
|
|
|
|
"-dNOPAUSE", # don't pause between pages
|
|
|
|
"-dSAFER", # safe mode
|
|
|
|
"-sDEVICE=ppmraw", # ppm driver
|
2020-07-16 12:43:29 +03:00
|
|
|
f"-sOutputFile={outfile}", # output file
|
2019-03-21 16:28:20 +03:00
|
|
|
# adjust for image origin
|
|
|
|
"-c",
|
2020-07-16 12:43:29 +03:00
|
|
|
f"{-bbox[0]} {-bbox[1]} translate",
|
2019-03-21 16:28:20 +03:00
|
|
|
"-f",
|
|
|
|
infile, # input file
|
|
|
|
# showpage (see https://bugs.ghostscript.com/show_bug.cgi?id=698272)
|
|
|
|
"-c",
|
|
|
|
"showpage",
|
|
|
|
]
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2013-03-06 21:36:22 +04:00
|
|
|
if gs_windows_binary is not None:
|
2014-01-09 07:07:35 +04:00
|
|
|
if not gs_windows_binary:
|
2019-09-30 17:56:31 +03:00
|
|
|
raise OSError("Unable to locate Ghostscript on paths")
|
2013-03-06 21:36:22 +04:00
|
|
|
command[0] = gs_windows_binary
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2018-09-26 14:38:44 +03:00
|
|
|
# push data through Ghostscript
|
2010-07-31 06:52:47 +04:00
|
|
|
try:
|
2019-02-06 11:07:53 +03:00
|
|
|
startupinfo = None
|
2019-03-21 16:28:20 +03:00
|
|
|
if sys.platform.startswith("win"):
|
2019-02-06 11:07:53 +03:00
|
|
|
startupinfo = subprocess.STARTUPINFO()
|
|
|
|
startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
|
|
|
|
subprocess.check_call(command, startupinfo=startupinfo)
|
2020-02-18 12:49:05 +03:00
|
|
|
out_im = Image.open(outfile)
|
|
|
|
out_im.load()
|
2010-07-31 06:52:47 +04:00
|
|
|
finally:
|
2014-01-22 11:18:24 +04:00
|
|
|
try:
|
|
|
|
os.unlink(outfile)
|
2014-09-14 13:08:31 +04:00
|
|
|
if infile_temp:
|
2014-09-03 10:09:04 +04:00
|
|
|
os.unlink(infile_temp)
|
2015-12-02 08:23:49 +03:00
|
|
|
except OSError:
|
2014-09-14 13:08:31 +04:00
|
|
|
pass
|
|
|
|
|
2020-02-18 12:49:05 +03:00
|
|
|
im = out_im.im.copy()
|
|
|
|
out_im.close()
|
|
|
|
return im
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
|
2019-09-30 17:56:31 +03:00
|
|
|
class PSFile:
|
2014-09-14 13:08:31 +04:00
|
|
|
"""
|
|
|
|
Wrapper for bytesio object that treats either CR or LF as end of line.
|
|
|
|
"""
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def __init__(self, fp):
|
|
|
|
self.fp = fp
|
|
|
|
self.char = None
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2019-01-13 05:05:46 +03:00
|
|
|
def seek(self, offset, whence=io.SEEK_SET):
|
2010-07-31 06:52:47 +04:00
|
|
|
self.char = None
|
|
|
|
self.fp.seek(offset, whence)
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def readline(self):
|
2014-08-09 02:47:39 +04:00
|
|
|
s = self.char or b""
|
|
|
|
self.char = None
|
|
|
|
|
|
|
|
c = self.fp.read(1)
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
while c not in b"\r\n":
|
2010-07-31 06:52:47 +04:00
|
|
|
s = s + c
|
|
|
|
c = self.fp.read(1)
|
|
|
|
|
2014-08-09 02:47:39 +04:00
|
|
|
self.char = self.fp.read(1)
|
|
|
|
# line endings can be 1 or 2 of \r \n, in either order
|
|
|
|
if self.char in b"\r\n":
|
|
|
|
self.char = None
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
return s.decode("latin-1")
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
def _accept(prefix):
|
2019-03-21 16:28:20 +03:00
|
|
|
return prefix[:4] == b"%!PS" or (len(prefix) >= 4 and i32(prefix) == 0xC6D3D0C5)
|
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
##
|
2020-07-10 11:48:02 +03:00
|
|
|
# Image plugin for Encapsulated PostScript. This plugin supports only
|
2010-07-31 06:52:47 +04:00
|
|
|
# a few variants of this format.
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
class EpsImageFile(ImageFile.ImageFile):
|
|
|
|
"""EPS File Parser for the Python Imaging Library"""
|
|
|
|
|
|
|
|
format = "EPS"
|
|
|
|
format_description = "Encapsulated Postscript"
|
|
|
|
|
2016-04-13 11:27:46 +03:00
|
|
|
mode_map = {1: "L", 2: "LAB", 3: "RGB", 4: "CMYK"}
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
def _open(self):
|
2014-08-05 02:28:57 +04:00
|
|
|
(length, offset) = self._find_offset(self.fp)
|
2014-05-07 15:36:35 +04:00
|
|
|
|
2014-08-05 02:28:57 +04:00
|
|
|
# Rewrap the open file pointer in something that will
|
|
|
|
# convert line endings and decode to latin-1.
|
2016-11-01 16:16:44 +03:00
|
|
|
fp = PSFile(self.fp)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
# go to offset - start of "%!PS"
|
2010-07-31 06:52:47 +04:00
|
|
|
fp.seek(offset)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
box = None
|
|
|
|
|
|
|
|
self.mode = "RGB"
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = 1, 1 # FIXME: huh?
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
#
|
|
|
|
# Load EPS header
|
|
|
|
|
2017-12-22 09:01:54 +03:00
|
|
|
s_raw = fp.readline()
|
2019-03-21 16:28:20 +03:00
|
|
|
s = s_raw.strip("\r\n")
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2017-12-22 09:01:54 +03:00
|
|
|
while s_raw:
|
|
|
|
if s:
|
|
|
|
if len(s) > 255:
|
|
|
|
raise SyntaxError("not an EPS file")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-12-22 09:01:54 +03:00
|
|
|
try:
|
|
|
|
m = split.match(s)
|
2020-06-21 13:13:35 +03:00
|
|
|
except re.error as e:
|
|
|
|
raise SyntaxError("not an EPS file") from e
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
if m:
|
2017-12-22 09:01:54 +03:00
|
|
|
k, v = m.group(1, 2)
|
|
|
|
self.info[k] = v
|
|
|
|
if k == "BoundingBox":
|
|
|
|
try:
|
|
|
|
# Note: The DSC spec says that BoundingBox
|
|
|
|
# fields should be integers, but some drivers
|
|
|
|
# put floating point values there anyway.
|
|
|
|
box = [int(float(i)) for i in v.split()]
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = box[2] - box[0], box[3] - box[1]
|
2019-03-21 16:28:20 +03:00
|
|
|
self.tile = [
|
|
|
|
("eps", (0, 0) + self.size, offset, (length, box))
|
|
|
|
]
|
2018-09-26 14:38:44 +03:00
|
|
|
except Exception:
|
2017-12-22 09:01:54 +03:00
|
|
|
pass
|
2012-10-24 07:21:19 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
else:
|
2017-12-22 09:01:54 +03:00
|
|
|
m = field.match(s)
|
|
|
|
if m:
|
|
|
|
k = m.group(1)
|
|
|
|
|
|
|
|
if k == "EndComments":
|
|
|
|
break
|
|
|
|
if k[:8] == "PS-Adobe":
|
|
|
|
self.info[k[:8]] = k[9:]
|
|
|
|
else:
|
|
|
|
self.info[k] = ""
|
2019-03-21 16:28:20 +03:00
|
|
|
elif s[0] == "%":
|
2020-07-10 11:48:02 +03:00
|
|
|
# handle non-DSC PostScript comments that some
|
2017-12-22 09:01:54 +03:00
|
|
|
# tools mistakenly put in the Comments section
|
|
|
|
pass
|
|
|
|
else:
|
2019-09-30 17:56:31 +03:00
|
|
|
raise OSError("bad EPS header")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-12-22 09:01:54 +03:00
|
|
|
s_raw = fp.readline()
|
2019-03-21 16:28:20 +03:00
|
|
|
s = s_raw.strip("\r\n")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-12-22 09:01:54 +03:00
|
|
|
if s and s[:1] != "%":
|
2010-07-31 06:52:47 +04:00
|
|
|
break
|
|
|
|
|
|
|
|
#
|
|
|
|
# Scan for an "ImageData" descriptor
|
|
|
|
|
2015-03-26 15:06:54 +03:00
|
|
|
while s[:1] == "%":
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
if len(s) > 255:
|
2012-10-11 07:52:53 +04:00
|
|
|
raise SyntaxError("not an EPS file")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
if s[:11] == "%ImageData:":
|
2014-08-08 03:41:10 +04:00
|
|
|
# Encoded bitmapped image.
|
2015-04-24 11:24:52 +03:00
|
|
|
x, y, bi, mo = s[11:].split(None, 7)[:4]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-08-09 02:47:39 +04:00
|
|
|
if int(bi) != 8:
|
2010-07-31 06:52:47 +04:00
|
|
|
break
|
2014-08-09 02:47:39 +04:00
|
|
|
try:
|
|
|
|
self.mode = self.mode_map[int(mo)]
|
2015-12-09 08:24:32 +03:00
|
|
|
except ValueError:
|
2010-07-31 06:52:47 +04:00
|
|
|
break
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = int(x), int(y)
|
2014-08-08 03:41:10 +04:00
|
|
|
return
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
s = fp.readline().strip("\r\n")
|
2010-07-31 06:52:47 +04:00
|
|
|
if not s:
|
|
|
|
break
|
|
|
|
|
|
|
|
if not box:
|
2019-09-30 17:56:31 +03:00
|
|
|
raise OSError("cannot determine EPS bounding box")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-08-05 02:28:57 +04:00
|
|
|
def _find_offset(self, fp):
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2014-08-05 02:28:57 +04:00
|
|
|
s = fp.read(160)
|
2014-09-14 13:08:31 +04:00
|
|
|
|
2014-08-05 02:28:57 +04:00
|
|
|
if s[:4] == b"%!PS":
|
|
|
|
# for HEAD without binary preview
|
2019-01-13 05:05:46 +03:00
|
|
|
fp.seek(0, io.SEEK_END)
|
2014-08-05 02:28:57 +04:00
|
|
|
length = fp.tell()
|
|
|
|
offset = 0
|
|
|
|
elif i32(s[0:4]) == 0xC6D3D0C5:
|
2014-09-14 13:08:31 +04:00
|
|
|
# FIX for: Some EPS file not handled correctly / issue #302
|
2014-08-05 02:28:57 +04:00
|
|
|
# EPS can contain binary data
|
|
|
|
# or start directly with latin coding
|
2014-09-14 13:08:31 +04:00
|
|
|
# more info see:
|
2017-02-18 07:53:01 +03:00
|
|
|
# https://web.archive.org/web/20160528181353/http://partners.adobe.com/public/developer/en/ps/5002.EPSF_Spec.pdf
|
2014-08-05 02:28:57 +04:00
|
|
|
offset = i32(s[4:8])
|
|
|
|
length = i32(s[8:12])
|
|
|
|
else:
|
|
|
|
raise SyntaxError("not an EPS file")
|
|
|
|
|
|
|
|
return (length, offset)
|
|
|
|
|
2013-11-17 11:26:44 +04:00
|
|
|
def load(self, scale=1):
|
2010-07-31 06:52:47 +04:00
|
|
|
# Load EPS via Ghostscript
|
|
|
|
if not self.tile:
|
|
|
|
return
|
2013-11-17 11:26:44 +04:00
|
|
|
self.im = Ghostscript(self.tile, self.size, self.fp, scale)
|
2010-07-31 06:52:47 +04:00
|
|
|
self.mode = self.im.mode
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = self.im.size
|
2010-07-31 06:52:47 +04:00
|
|
|
self.tile = []
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
def load_seek(self, *args, **kwargs):
|
2014-01-22 11:17:47 +04:00
|
|
|
# we can't incrementally load, so force ImageFile.parser to
|
2014-08-26 17:47:10 +04:00
|
|
|
# use our custom load method by defining this method.
|
2014-01-22 11:17:47 +04:00
|
|
|
pass
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# --------------------------------------------------------------------
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def _save(im, fp, filename, eps=1):
|
|
|
|
"""EPS Writer for the Python Imaging Library."""
|
|
|
|
|
|
|
|
#
|
|
|
|
# make sure image data is available
|
|
|
|
im.load()
|
|
|
|
|
|
|
|
#
|
2020-07-10 11:48:02 +03:00
|
|
|
# determine PostScript image mode
|
2010-07-31 06:52:47 +04:00
|
|
|
if im.mode == "L":
|
2012-10-24 07:21:19 +04:00
|
|
|
operator = (8, 1, "image")
|
2010-07-31 06:52:47 +04:00
|
|
|
elif im.mode == "RGB":
|
2012-10-24 07:21:19 +04:00
|
|
|
operator = (8, 3, "false 3 colorimage")
|
2010-07-31 06:52:47 +04:00
|
|
|
elif im.mode == "CMYK":
|
2012-10-24 07:21:19 +04:00
|
|
|
operator = (8, 4, "false 4 colorimage")
|
2010-07-31 06:52:47 +04:00
|
|
|
else:
|
2012-10-11 07:52:53 +04:00
|
|
|
raise ValueError("image mode is not supported")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2012-10-24 07:21:19 +04:00
|
|
|
base_fp = fp
|
2016-11-11 22:18:27 +03:00
|
|
|
wrapped_fp = False
|
2015-07-31 13:59:59 +03:00
|
|
|
if fp != sys.stdout:
|
2019-09-26 15:12:28 +03:00
|
|
|
fp = io.TextIOWrapper(fp, encoding="latin-1")
|
|
|
|
wrapped_fp = True
|
2012-10-24 07:21:19 +04:00
|
|
|
|
2016-11-11 22:18:27 +03:00
|
|
|
try:
|
|
|
|
if eps:
|
|
|
|
#
|
|
|
|
# write EPS header
|
|
|
|
fp.write("%!PS-Adobe-3.0 EPSF-3.0\n")
|
|
|
|
fp.write("%%Creator: PIL 0.1 EpsEncode\n")
|
|
|
|
# fp.write("%%CreationDate: %s"...)
|
|
|
|
fp.write("%%%%BoundingBox: 0 0 %d %d\n" % im.size)
|
|
|
|
fp.write("%%Pages: 1\n")
|
|
|
|
fp.write("%%EndComments\n")
|
|
|
|
fp.write("%%Page: 1 1\n")
|
|
|
|
fp.write("%%ImageData: %d %d " % im.size)
|
2019-03-21 16:28:20 +03:00
|
|
|
fp.write('%d %d 0 1 1 "%s"\n' % operator)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2016-11-11 22:18:27 +03:00
|
|
|
#
|
|
|
|
# image header
|
|
|
|
fp.write("gsave\n")
|
|
|
|
fp.write("10 dict begin\n")
|
2020-07-16 12:43:29 +03:00
|
|
|
fp.write(f"/buf {im.size[0] * operator[1]} string def\n")
|
2016-11-11 22:18:27 +03:00
|
|
|
fp.write("%d %d scale\n" % im.size)
|
|
|
|
fp.write("%d %d 8\n" % im.size) # <= bits
|
2020-07-16 12:43:29 +03:00
|
|
|
fp.write(f"[{im.size[0]} 0 0 -{im.size[1]} 0 {im.size[1]}]\n")
|
2016-11-11 22:18:27 +03:00
|
|
|
fp.write("{ currentfile buf readhexstring pop } bind\n")
|
|
|
|
fp.write(operator[2] + "\n")
|
|
|
|
if hasattr(fp, "flush"):
|
|
|
|
fp.flush()
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
ImageFile._save(im, base_fp, [("eps", (0, 0) + im.size, 0, None)])
|
2016-11-11 22:18:27 +03:00
|
|
|
|
|
|
|
fp.write("\n%%%%EndBinary\n")
|
|
|
|
fp.write("grestore end\n")
|
|
|
|
if hasattr(fp, "flush"):
|
|
|
|
fp.flush()
|
|
|
|
finally:
|
|
|
|
if wrapped_fp:
|
|
|
|
fp.detach()
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# --------------------------------------------------------------------
|
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
Image.register_open(EpsImageFile.format, EpsImageFile, _accept)
|
|
|
|
|
|
|
|
Image.register_save(EpsImageFile.format, _save)
|
|
|
|
|
2016-04-25 07:59:02 +03:00
|
|
|
Image.register_extensions(EpsImageFile.format, [".ps", ".eps"])
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
Image.register_mime(EpsImageFile.format, "application/postscript")
|