import colorsys from dataclasses import dataclass from typing import Self import numpy as np from PIL import Image, ImageDraw, ImageFont @dataclass class AdventImage: img: Image.Image @classmethod async def load_standard(cls, fp) -> Self: """ Bild laden und einen quadratischen Ausschnitt aus der Mitte nehmen """ # Bild laden img = Image.open(fp=fp) # Größen bestimmen width, height = img.size square = min(width, height) # Bild zuschneiden und skalieren img = img.crop(box=( int((width - square)/2), int((height - square)/2), int((width + square)/2), int((height + square)/2), )) img = img.resize( size=(500, 500), resample=Image.LANCZOS, ) # Farbmodell festlegen return cls(img=img.convert("RGB")) async def get_text_box( self, xy: tuple[float, float], text: str | bytes, font: "ImageFont._Font", anchor: str | None = "mm", **text_kwargs, ) -> tuple[int, int, int, int] | None: """ Koordinaten (links, oben, rechts, unten) des betroffenen Rechtecks bestimmen, wenn das Bild mit einem Text versehen wird """ # Neues 1-Bit Bild, gleiche Größe mask = Image.new(mode="1", size=self.img.size, color=0) # Text auf Maske auftragen ImageDraw.Draw(mask).text( xy=xy, text=text, font=font, anchor=anchor, fill=1, **text_kwargs, ) # betroffenen Pixelbereich bestimmen return mask.getbbox() async def get_average_color( self, box: tuple[int, int, int, int], ) -> tuple[int, int, int]: """ Durchschnittsfarbe eines rechteckigen Ausschnitts in einem Bild berechnen """ pixel_data = self.img.crop(box).getdata() mean_color: np.ndarray = np.mean(pixel_data, axis=0) return tuple(mean_color.astype(int).tolist()) async def hide_text( self, xy: tuple[float, float], text: str | bytes, font: "ImageFont._Font", anchor: str | None = "mm", **text_kwargs, ) -> None: """ Text `text` in Bild an Position `xy` verstecken. Weitere Parameter wie bei `ImageDraw.text()`. """ # betroffenen Bildbereich bestimmen text_box = await self.get_text_box( xy=xy, text=text, font=font, anchor=anchor, **text_kwargs ) if text_box is not None: # Durchschnittsfarbe bestimmen text_color = await self.get_average_color( box=text_box, ) # etwas heller/dunkler machen tc_h, tc_s, tc_v = colorsys.rgb_to_hsv(*text_color) tc_v = int((tc_v - 127) * 0.97) + 127 if tc_v < 127: tc_v += 3 else: tc_v -= 3 text_color = colorsys.hsv_to_rgb(tc_h, tc_s, tc_v) text_color = tuple(int(val) for val in text_color) # Buchstaben verstecken ImageDraw.Draw(self.img).text( xy=xy, text=text, font=font, fill=text_color, anchor=anchor, **text_kwargs )