|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| """
|
| (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)
|
|
|