Pillow/src/PIL/PSDraw.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

238 lines
6.8 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library
# $Id$
#
2020-07-10 11:48:02 +03:00
# Simple PostScript graphics interface
2010-07-31 06:52:47 +04:00
#
# History:
# 1996-04-20 fl Created
# 1999-01-10 fl Added gsave/grestore to image method
# 2005-05-04 fl Fixed floating point issue in image (from Eric Etheridge)
#
# Copyright (c) 1997-2005 by Secret Labs AB. All rights reserved.
# Copyright (c) 1996 by Fredrik Lundh.
#
# See the README file for information on usage and redistribution.
#
from __future__ import annotations
2010-07-31 06:52:47 +04:00
import sys
2024-05-18 09:06:50 +03:00
from typing import TYPE_CHECKING
from . import EpsImagePlugin
2014-08-26 17:47:10 +04:00
2010-07-31 06:52:47 +04:00
##
2020-07-10 11:48:02 +03:00
# Simple PostScript graphics interface.
2010-07-31 06:52:47 +04:00
class PSDraw:
2013-10-14 09:30:00 +04:00
"""
2020-09-01 20:16:46 +03:00
Sets up printing to the given file. If ``fp`` is omitted,
``sys.stdout.buffer`` or ``sys.stdout`` is assumed.
2013-10-14 09:30:00 +04:00
"""
2010-07-31 06:52:47 +04:00
def __init__(self, fp=None):
if not fp:
try:
fp = sys.stdout.buffer
except AttributeError:
fp = sys.stdout
2010-07-31 06:52:47 +04:00
self.fp = fp
2024-05-18 09:06:50 +03:00
def begin_document(self, id: str | None = None) -> None:
2020-07-10 11:48:02 +03:00
"""Set up printing of a document. (Write PostScript DSC header.)"""
2010-07-31 06:52:47 +04:00
# FIXME: incomplete
self.fp.write(
b"%!PS-Adobe-3.0\n"
b"save\n"
b"/showpage { } def\n"
b"%%EndComments\n"
b"%%BeginDocument\n"
2014-12-28 00:07:41 +03:00
)
# self.fp.write(ERROR_PS) # debugging!
self.fp.write(EDROFF_PS)
self.fp.write(VDI_PS)
self.fp.write(b"%%EndProlog\n")
2024-05-18 09:06:50 +03:00
self.isofont: dict[bytes, int] = {}
2010-07-31 06:52:47 +04:00
2024-05-11 03:48:09 +03:00
def end_document(self) -> None:
2020-07-10 11:48:02 +03:00
"""Ends printing. (Write PostScript DSC footer.)"""
self.fp.write(b"%%EndDocument\nrestore showpage\n%%End\n")
2010-07-31 06:52:47 +04:00
if hasattr(self.fp, "flush"):
self.fp.flush()
2024-05-18 09:06:50 +03:00
def setfont(self, font: str, size: int) -> None:
2013-10-14 09:30:00 +04:00
"""
Selects which font to use.
2020-07-10 11:48:02 +03:00
:param font: A PostScript font name
2013-10-14 09:30:00 +04:00
:param size: Size in points.
"""
2024-05-18 09:06:50 +03:00
font_bytes = bytes(font, "UTF-8")
if font_bytes not in self.isofont:
2010-07-31 06:52:47 +04:00
# reencode font
2024-05-18 09:06:50 +03:00
self.fp.write(
b"/PSDraw-%s ISOLatin1Encoding /%s E\n" % (font_bytes, font_bytes)
)
self.isofont[font_bytes] = 1
2010-07-31 06:52:47 +04:00
# rough
2024-05-18 09:06:50 +03:00
self.fp.write(b"/F0 %d /PSDraw-%s F\n" % (size, font_bytes))
2010-07-31 06:52:47 +04:00
2024-05-18 09:06:50 +03:00
def line(self, xy0: tuple[int, int], xy1: tuple[int, int]) -> None:
2013-10-14 09:30:00 +04:00
"""
Draws a line between the two points. Coordinates are given in
2020-07-10 11:48:02 +03:00
PostScript point coordinates (72 points per inch, (0, 0) is the lower
2013-10-14 09:30:00 +04:00
left corner of the page).
"""
self.fp.write(b"%d %d %d %d Vl\n" % (*xy0, *xy1))
2010-07-31 06:52:47 +04:00
2024-05-18 09:06:50 +03:00
def rectangle(self, box: tuple[int, int, int, int]) -> None:
2013-10-14 09:30:00 +04:00
"""
Draws a rectangle.
:param box: A tuple of four integers, specifying left, bottom, width and
height.
2013-10-14 09:30:00 +04:00
"""
2022-07-10 23:54:23 +03:00
self.fp.write(b"%d %d M 0 %d %d Vr\n" % box)
2010-07-31 06:52:47 +04:00
2024-05-18 09:06:50 +03:00
def text(self, xy: tuple[int, int], text: str) -> None:
2013-10-14 09:30:00 +04:00
"""
Draws text at the given position. You must use
:py:meth:`~PIL.PSDraw.PSDraw.setfont` before calling this method.
"""
2024-05-18 09:06:50 +03:00
text_bytes = bytes(text, "UTF-8")
text_bytes = b"\\(".join(text_bytes.split(b"("))
text_bytes = b"\\)".join(text_bytes.split(b")"))
self.fp.write(b"%d %d M (%s) S\n" % (xy + (text_bytes,)))
2010-07-31 06:52:47 +04:00
2024-05-18 09:06:50 +03:00
if TYPE_CHECKING:
from . import Image
def image(
self, box: tuple[int, int, int, int], im: Image.Image, dpi: int | None = None
) -> None:
2013-10-14 09:30:00 +04:00
"""Draw a PIL image, centered in the given box."""
2010-07-31 06:52:47 +04:00
# default resolution depends on mode
if not dpi:
if im.mode == "1":
dpi = 200 # fax
else:
2023-10-19 11:12:01 +03:00
dpi = 100 # grayscale
2010-07-31 06:52:47 +04:00
# image size (on paper)
x = im.size[0] * 72 / dpi
y = im.size[1] * 72 / dpi
2010-07-31 06:52:47 +04:00
# max allowed size
xmax = float(box[2] - box[0])
ymax = float(box[3] - box[1])
if x > xmax:
y = y * xmax / x
x = xmax
if y > ymax:
x = x * ymax / y
y = ymax
dx = (xmax - x) / 2 + box[0]
dy = (ymax - y) / 2 + box[1]
self.fp.write(b"gsave\n%f %f translate\n" % (dx, dy))
2010-07-31 06:52:47 +04:00
if (x, y) != im.size:
# EpsImagePlugin._save prints the image at (0,0,xsize,ysize)
sx = x / im.size[0]
sy = y / im.size[1]
self.fp.write(b"%f %f scale\n" % (sx, sy))
2024-06-11 16:26:00 +03:00
EpsImagePlugin._save(im, self.fp, "", 0)
self.fp.write(b"\ngrestore\n")
2010-07-31 06:52:47 +04:00
2019-03-21 16:28:20 +03:00
2010-07-31 06:52:47 +04:00
# --------------------------------------------------------------------
2020-07-10 11:48:02 +03:00
# PostScript driver
2010-07-31 06:52:47 +04:00
#
2020-07-10 11:48:02 +03:00
# EDROFF.PS -- PostScript driver for Edroff 2
2010-07-31 06:52:47 +04:00
#
# History:
# 94-01-25 fl: created (edroff 2.04)
#
# Copyright (c) Fredrik Lundh 1994.
#
2018-03-03 12:54:00 +03:00
EDROFF_PS = b"""\
2010-07-31 06:52:47 +04:00
/S { show } bind def
/P { moveto show } bind def
/M { moveto } bind def
/X { 0 rmoveto } bind def
/Y { 0 exch rmoveto } bind def
/E { findfont
dup maxlength dict begin
{
1 index /FID ne { def } { pop pop } ifelse
} forall
/Encoding exch def
dup /FontName exch def
currentdict end definefont pop
} bind def
/F { findfont exch scalefont dup setfont
[ exch /setfont cvx ] cvx bind def
} bind def
"""
#
2020-07-10 11:48:02 +03:00
# VDI.PS -- PostScript driver for VDI meta commands
2010-07-31 06:52:47 +04:00
#
# History:
# 94-01-25 fl: created (edroff 2.04)
#
# Copyright (c) Fredrik Lundh 1994.
#
VDI_PS = b"""\
2010-07-31 06:52:47 +04:00
/Vm { moveto } bind def
/Va { newpath arcn stroke } bind def
/Vl { moveto lineto stroke } bind def
/Vc { newpath 0 360 arc closepath } bind def
/Vr { exch dup 0 rlineto
exch dup 0 exch rlineto
2010-07-31 06:52:47 +04:00
exch neg 0 rlineto
0 exch neg rlineto
2022-07-11 12:38:46 +03:00
setgray fill } bind def
2010-07-31 06:52:47 +04:00
/Tm matrix def
/Ve { Tm currentmatrix pop
translate scale newpath 0 0 .5 0 360 arc closepath
Tm setmatrix
} bind def
/Vf { currentgray exch setgray fill setgray } bind def
"""
#
# ERROR.PS -- Error handler
#
# History:
# 89-11-21 fl: created (pslist 1.10)
#
ERROR_PS = b"""\
2010-07-31 06:52:47 +04:00
/landscape false def
/errorBUF 200 string def
/errorNL { currentpoint 10 sub exch pop 72 exch moveto } def
errordict begin /handleerror {
initmatrix /Courier findfont 10 scalefont setfont
newpath 72 720 moveto $error begin /newerror false def
(PostScript Error) show errorNL errorNL
(Error: ) show
/errorname load errorBUF cvs show errorNL errorNL
(Command: ) show
/command load dup type /stringtype ne { errorBUF cvs } if show
errorNL errorNL
(VMstatus: ) show
vmstatus errorBUF cvs show ( bytes available, ) show
errorBUF cvs show ( bytes used at level ) show
errorBUF cvs show errorNL errorNL
(Operand stargck: ) show errorNL /ostargck load {
dup type /stringtype ne { errorBUF cvs } if 72 0 rmoveto show errorNL
} forall errorNL
(Execution stargck: ) show errorNL /estargck load {
dup type /stringtype ne { errorBUF cvs } if 72 0 rmoveto show errorNL
} forall
end showpage
} def end
"""