|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
""" |
|
(Experimental) WCK-style drawing interface operations |
|
|
|
.. seealso:: :py:mod:`PIL.ImageDraw` |
|
""" |
|
from __future__ import annotations |
|
|
|
from typing import Any, AnyStr, BinaryIO |
|
|
|
from . import Image, ImageColor, ImageDraw, ImageFont, ImagePath |
|
from ._typing import Coords, StrOrBytesPath |
|
|
|
|
|
class Pen: |
|
"""Stores an outline color and width.""" |
|
|
|
def __init__(self, color: str, width: int = 1, opacity: int = 255) -> None: |
|
self.color = ImageColor.getrgb(color) |
|
self.width = width |
|
|
|
|
|
class Brush: |
|
"""Stores a fill color""" |
|
|
|
def __init__(self, color: str, opacity: int = 255) -> None: |
|
self.color = ImageColor.getrgb(color) |
|
|
|
|
|
class Font: |
|
"""Stores a TrueType font and color""" |
|
|
|
def __init__( |
|
self, color: str, file: StrOrBytesPath | BinaryIO, size: float = 12 |
|
) -> None: |
|
|
|
self.color = ImageColor.getrgb(color) |
|
self.font = ImageFont.truetype(file, size) |
|
|
|
|
|
class Draw: |
|
""" |
|
(Experimental) WCK-style drawing interface |
|
""" |
|
|
|
def __init__( |
|
self, |
|
image: Image.Image | str, |
|
size: tuple[int, int] | list[int] | None = None, |
|
color: float | tuple[float, ...] | str | None = None, |
|
) -> None: |
|
if isinstance(image, str): |
|
if size is None: |
|
msg = "If image argument is mode string, size must be a list or tuple" |
|
raise ValueError(msg) |
|
image = Image.new(image, size, color) |
|
self.draw = ImageDraw.Draw(image) |
|
self.image = image |
|
self.transform: tuple[float, float, float, float, float, float] | None = None |
|
|
|
def flush(self) -> Image.Image: |
|
return self.image |
|
|
|
def render( |
|
self, |
|
op: str, |
|
xy: Coords, |
|
pen: Pen | Brush | None, |
|
brush: Brush | Pen | None = None, |
|
**kwargs: Any, |
|
) -> None: |
|
|
|
outline = fill = None |
|
width = 1 |
|
if isinstance(pen, Pen): |
|
outline = pen.color |
|
width = pen.width |
|
elif isinstance(brush, Pen): |
|
outline = brush.color |
|
width = brush.width |
|
if isinstance(brush, Brush): |
|
fill = brush.color |
|
elif isinstance(pen, Brush): |
|
fill = pen.color |
|
|
|
if self.transform: |
|
path = ImagePath.Path(xy) |
|
path.transform(self.transform) |
|
xy = path |
|
|
|
if op in ("arc", "line"): |
|
kwargs.setdefault("fill", outline) |
|
else: |
|
kwargs.setdefault("fill", fill) |
|
kwargs.setdefault("outline", outline) |
|
if op == "line": |
|
kwargs.setdefault("width", width) |
|
getattr(self.draw, op)(xy, **kwargs) |
|
|
|
def settransform(self, offset: tuple[float, float]) -> None: |
|
"""Sets a transformation offset.""" |
|
(xoffset, yoffset) = offset |
|
self.transform = (1, 0, xoffset, 0, 1, yoffset) |
|
|
|
def arc( |
|
self, |
|
xy: Coords, |
|
pen: Pen | Brush | None, |
|
start: float, |
|
end: float, |
|
*options: Any, |
|
) -> None: |
|
""" |
|
Draws an arc (a portion of a circle outline) between the start and end |
|
angles, inside the given bounding box. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.arc` |
|
""" |
|
self.render("arc", xy, pen, *options, start=start, end=end) |
|
|
|
def chord( |
|
self, |
|
xy: Coords, |
|
pen: Pen | Brush | None, |
|
start: float, |
|
end: float, |
|
*options: Any, |
|
) -> None: |
|
""" |
|
Same as :py:meth:`~PIL.ImageDraw2.Draw.arc`, but connects the end points |
|
with a straight line. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.chord` |
|
""" |
|
self.render("chord", xy, pen, *options, start=start, end=end) |
|
|
|
def ellipse(self, xy: Coords, pen: Pen | Brush | None, *options: Any) -> None: |
|
""" |
|
Draws an ellipse inside the given bounding box. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.ellipse` |
|
""" |
|
self.render("ellipse", xy, pen, *options) |
|
|
|
def line(self, xy: Coords, pen: Pen | Brush | None, *options: Any) -> None: |
|
""" |
|
Draws a line between the coordinates in the ``xy`` list. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.line` |
|
""" |
|
self.render("line", xy, pen, *options) |
|
|
|
def pieslice( |
|
self, |
|
xy: Coords, |
|
pen: Pen | Brush | None, |
|
start: float, |
|
end: float, |
|
*options: Any, |
|
) -> None: |
|
""" |
|
Same as arc, but also draws straight lines between the end points and the |
|
center of the bounding box. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.pieslice` |
|
""" |
|
self.render("pieslice", xy, pen, *options, start=start, end=end) |
|
|
|
def polygon(self, xy: Coords, pen: Pen | Brush | None, *options: Any) -> None: |
|
""" |
|
Draws a polygon. |
|
|
|
The polygon outline consists of straight lines between the given |
|
coordinates, plus a straight line between the last and the first |
|
coordinate. |
|
|
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.polygon` |
|
""" |
|
self.render("polygon", xy, pen, *options) |
|
|
|
def rectangle(self, xy: Coords, pen: Pen | Brush | None, *options: Any) -> None: |
|
""" |
|
Draws a rectangle. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.rectangle` |
|
""" |
|
self.render("rectangle", xy, pen, *options) |
|
|
|
def text(self, xy: tuple[float, float], text: AnyStr, font: Font) -> None: |
|
""" |
|
Draws the string at the given position. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.text` |
|
""" |
|
if self.transform: |
|
path = ImagePath.Path(xy) |
|
path.transform(self.transform) |
|
xy = path |
|
self.draw.text(xy, text, font=font.font, fill=font.color) |
|
|
|
def textbbox( |
|
self, xy: tuple[float, float], text: AnyStr, font: Font |
|
) -> tuple[float, float, float, float]: |
|
""" |
|
Returns bounding box (in pixels) of given text. |
|
|
|
:return: ``(left, top, right, bottom)`` bounding box |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.textbbox` |
|
""" |
|
if self.transform: |
|
path = ImagePath.Path(xy) |
|
path.transform(self.transform) |
|
xy = path |
|
return self.draw.textbbox(xy, text, font=font.font) |
|
|
|
def textlength(self, text: AnyStr, font: Font) -> float: |
|
""" |
|
Returns length (in pixels) of given text. |
|
This is the amount by which following text should be offset. |
|
|
|
.. seealso:: :py:meth:`PIL.ImageDraw.ImageDraw.textlength` |
|
""" |
|
return self.draw.textlength(text, font=font.font) |
|
|