# SPDX-FileCopyrightText: 2020 Melissa LeBlanc-Williams for Adafruit Industries # # SPDX-License-Identifier: MIT """ `displayio.tilegrid` ================================================================================ displayio for Blinka **Software and Dependencies:** * Adafruit Blinka: https://github.com/adafruit/Adafruit_Blinka/releases * Author(s): Melissa LeBlanc-Williams """ from typing import Union, Optional, Tuple from PIL import Image from ._bitmap import Bitmap from ._colorconverter import ColorConverter from ._ondiskbitmap import OnDiskBitmap from ._shape import Shape from ._palette import Palette from ._structs import RectangleStruct, TransformStruct __version__ = "0.0.0-auto.0" __repo__ = "https://github.com/adafruit/Adafruit_Blinka_displayio.git" class TileGrid: # pylint: disable=too-many-instance-attributes """Position a grid of tiles sourced from a bitmap and pixel_shader combination. Multiple grids can share bitmaps and pixel shaders. A single tile grid is also known as a Sprite. """ def __init__( self, bitmap: Union[Bitmap, OnDiskBitmap, Shape], *, pixel_shader: Union[ColorConverter, Palette], width: int = 1, height: int = 1, tile_width: Optional[int] = None, tile_height: Optional[int] = None, default_tile: int = 0, x: int = 0, y: int = 0, ): """Create a TileGrid object. The bitmap is source for 2d pixels. The pixel_shader is used to convert the value and its location to a display native pixel color. This may be a simple color palette lookup, a gradient, a pattern or a color transformer. tile_width and tile_height match the height of the bitmap by default. """ if not isinstance(bitmap, (Bitmap, OnDiskBitmap, Shape)): raise ValueError("Unsupported Bitmap type") self._bitmap = bitmap bitmap_width = bitmap.width bitmap_height = bitmap.height if pixel_shader is not None and not isinstance( pixel_shader, (ColorConverter, Palette) ): raise ValueError("Unsupported Pixel Shader type") self._pixel_shader = pixel_shader if isinstance(self._pixel_shader, ColorConverter): self._pixel_shader._rgba = True # pylint: disable=protected-access self._hidden_tilegrid = False self._x = x self._y = y self._width = width # Number of Tiles Wide self._height = height # Number of Tiles High self._transpose_xy = False self._flip_x = False self._flip_y = False self._top_left_x = 0 self._top_left_y = 0 if tile_width is None or tile_width == 0: tile_width = bitmap_width if tile_height is None or tile_width == 0: tile_height = bitmap_height if tile_width < 1 or tile_height < 1: raise ValueError("Tile width and height must be greater than 0") if bitmap_width % tile_width != 0: raise ValueError("Tile width must exactly divide bitmap width") self._tile_width = tile_width if bitmap_height % tile_height != 0: raise ValueError("Tile height must exactly divide bitmap height") self._tile_height = tile_height if not 0 <= default_tile <= 255: raise ValueError("Default Tile is out of range") self._pixel_width = width * tile_width self._pixel_height = height * tile_height self._tiles = (self._width * self._height) * [default_tile] self._in_group = False self._absolute_transform = TransformStruct(0, 0, 1, 1, 1, False, False, False) self._current_area = RectangleStruct( 0, 0, self._pixel_width, self._pixel_height ) self._moved = False def _update_transform(self, absolute_transform): """Update the parent transform and child transforms""" self._absolute_transform = absolute_transform if self._absolute_transform is not None: self._update_current_x() self._update_current_y() def _update_current_x(self): if self._transpose_xy: width = self._pixel_height else: width = self._pixel_width if self._absolute_transform.transpose_xy: self._current_area.y1 = ( self._absolute_transform.y + self._absolute_transform.dy * self._x ) self._current_area.y2 = ( self._absolute_transform.y + self._absolute_transform.dy * (self._x + width) ) if self._current_area.y2 < self._current_area.y1: self._current_area.y1, self._current_area.y2 = ( self._current_area.y2, self._current_area.y1, ) else: self._current_area.x1 = ( self._absolute_transform.x + self._absolute_transform.dx * self._x ) self._current_area.x2 = ( self._absolute_transform.x + self._absolute_transform.dx * (self._x + width) ) if self._current_area.x2 < self._current_area.x1: self._current_area.x1, self._current_area.x2 = ( self._current_area.x2, self._current_area.x1, ) def _update_current_y(self): if self._transpose_xy: height = self._pixel_width else: height = self._pixel_height if self._absolute_transform.transpose_xy: self._current_area.x1 = ( self._absolute_transform.x + self._absolute_transform.dx * self._y ) self._current_area.x2 = ( self._absolute_transform.x + self._absolute_transform.dx * (self._y + height) ) if self._current_area.x2 < self._current_area.x1: self._current_area.x1, self._current_area.x2 = ( self._current_area.x2, self._current_area.x1, ) else: self._current_area.y1 = ( self._absolute_transform.y + self._absolute_transform.dy * self._y ) self._current_area.y2 = ( self._absolute_transform.y + self._absolute_transform.dy * (self._y + height) ) if self._current_area.y2 < self._current_area.y1: self._current_area.y1, self._current_area.y2 = ( self._current_area.y2, self._current_area.y1, ) def _shade(self, pixel_value): if isinstance(self._pixel_shader, Palette): return self._pixel_shader[pixel_value] if isinstance(self._pixel_shader, ColorConverter): return self._pixel_shader.convert(pixel_value) return pixel_value def _apply_palette(self, image): image.putpalette( self._pixel_shader._get_palette() # pylint: disable=protected-access ) def _add_alpha(self, image): alpha = self._bitmap._image.copy().convert( # pylint: disable=protected-access "P" ) alpha.putpalette( self._pixel_shader._get_alpha_palette() # pylint: disable=protected-access ) image.putalpha(alpha.convert("L")) def _fill_area(self, buffer): # pylint: disable=too-many-locals,too-many-branches,too-many-statements """Draw onto the image""" if self._hidden_tilegrid: return if self._bitmap.width <= 0 or self._bitmap.height <= 0: return # Copy class variables to local variables in case something changes x = self._x y = self._y width = self._width height = self._height tile_width = self._tile_width tile_height = self._tile_height bitmap_width = self._bitmap.width pixel_width = self._pixel_width pixel_height = self._pixel_height tiles = self._tiles absolute_transform = self._absolute_transform pixel_shader = self._pixel_shader bitmap = self._bitmap tiles = self._tiles tile_count_x = bitmap_width // tile_width image = Image.new( "RGBA", (width * tile_width, height * tile_height), (0, 0, 0, 0), ) for tile_x in range(width): for tile_y in range(height): tile_index = tiles[tile_y * width + tile_x] tile_index_x = tile_index % tile_count_x tile_index_y = tile_index // tile_count_x tile_image = bitmap._image # pylint: disable=protected-access if isinstance(pixel_shader, Palette): tile_image = tile_image.copy().convert("P") self._apply_palette(tile_image) tile_image = tile_image.convert("RGBA") self._add_alpha(tile_image) elif isinstance(pixel_shader, ColorConverter): # This will be needed for eInks, grayscale, and monochrome displays pass image.alpha_composite( tile_image, dest=(tile_x * tile_width, tile_y * tile_height), source=( tile_index_x * tile_width, tile_index_y * tile_height, tile_index_x * tile_width + tile_width, tile_index_y * tile_height + tile_height, ), ) if absolute_transform is not None: if absolute_transform.scale > 1: image = image.resize( ( int(pixel_width * absolute_transform.scale), int( pixel_height * absolute_transform.scale, ), ), resample=Image.NEAREST, ) if absolute_transform.mirror_x: image = image.transpose(Image.FLIP_LEFT_RIGHT) if absolute_transform.mirror_y: image = image.transpose(Image.FLIP_TOP_BOTTOM) if absolute_transform.transpose_xy: image = image.transpose(Image.TRANSPOSE) x *= absolute_transform.dx y *= absolute_transform.dy x += absolute_transform.x y += absolute_transform.y source_x = source_y = 0 if x < 0: source_x = round(0 - x) x = 0 if y < 0: source_y = round(0 - y) y = 0 x = round(x) y = round(y) if ( x <= buffer.width and y <= buffer.height and source_x <= image.width and source_y <= image.height ): buffer.alpha_composite(image, (x, y), source=(source_x, source_y)) def _finish_refresh(self): pass @property def hidden(self) -> bool: """True when the TileGrid is hidden. This may be False even when a part of a hidden Group.""" return self._hidden_tilegrid @hidden.setter def hidden(self, value: bool): if not isinstance(value, (bool, int)): raise ValueError("Expecting a boolean or integer value") self._hidden_tilegrid = bool(value) @property def x(self) -> int: """X position of the left edge in the parent.""" return self._x @x.setter def x(self, value: int): if not isinstance(value, int): raise TypeError("X should be a integer type") if self._x != value: self._x = value self._update_current_x() @property def y(self) -> int: """Y position of the top edge in the parent.""" return self._y @y.setter def y(self, value: int): if not isinstance(value, int): raise TypeError("Y should be a integer type") if self._y != value: self._y = value self._update_current_y() @property def flip_x(self) -> bool: """If true, the left edge rendered will be the right edge of the right-most tile.""" return self._flip_x @flip_x.setter def flip_x(self, value: bool): if not isinstance(value, bool): raise TypeError("Flip X should be a boolean type") if self._flip_x != value: self._flip_x = value @property def flip_y(self) -> bool: """If true, the top edge rendered will be the bottom edge of the bottom-most tile.""" return self._flip_y @flip_y.setter def flip_y(self, value: bool): if not isinstance(value, bool): raise TypeError("Flip Y should be a boolean type") if self._flip_y != value: self._flip_y = value @property def transpose_xy(self) -> bool: """If true, the TileGrid's axis will be swapped. When combined with mirroring, any 90 degree rotation can be achieved along with the corresponding mirrored version. """ return self._transpose_xy @transpose_xy.setter def transpose_xy(self, value: bool): if not isinstance(value, bool): raise TypeError("Transpose XY should be a boolean type") if self._transpose_xy != value: self._transpose_xy = value self._update_current_x() self._update_current_y() @property def pixel_shader(self) -> Union[ColorConverter, Palette]: """The pixel shader of the tilegrid.""" return self._pixel_shader @pixel_shader.setter def pixel_shader(self, new_pixel_shader: Union[ColorConverter, Palette]) -> None: if not isinstance(new_pixel_shader, ColorConverter) and not isinstance( new_pixel_shader, Palette ): raise TypeError( "Unsupported Type: new_pixel_shader must be ColorConverter or Palette" ) self._pixel_shader = new_pixel_shader @property def bitmap(self) -> Union[Bitmap, OnDiskBitmap, Shape]: """The Bitmap, OnDiskBitmap, or Shape that is assigned to this TileGrid""" return self._bitmap @bitmap.setter def bitmap(self, new_bitmap: Union[Bitmap, OnDiskBitmap, Shape]) -> None: if ( not isinstance(new_bitmap, Bitmap) and not isinstance(new_bitmap, OnDiskBitmap) and not isinstance(new_bitmap, Shape) ): raise TypeError( "Unsupported Type: new_bitmap must be Bitmap, OnDiskBitmap, or Shape" ) if ( new_bitmap.width != self.bitmap.width or new_bitmap.height != self.bitmap.height ): raise ValueError("New bitmap must be same size as old bitmap") self._bitmap = new_bitmap def _extract_and_check_index(self, index): if isinstance(index, (tuple, list)): x = index[0] y = index[1] index = y * self._width + x elif isinstance(index, int): x = index % self._width y = index // self._width if x > self._width or y > self._height or index >= len(self._tiles): raise ValueError("Tile index out of bounds") return index def __getitem__(self, index: Union[Tuple[int, int], int]) -> int: """Returns the tile index at the given index. The index can either be an x,y tuple or an int equal to ``y * width + x``'. """ index = self._extract_and_check_index(index) return self._tiles[index] def __setitem__(self, index: Union[Tuple[int, int], int], value: int) -> None: """Sets the tile index at the given index. The index can either be an x,y tuple or an int equal to ``y * width + x``. """ index = self._extract_and_check_index(index) if not 0 <= value <= 255: raise ValueError("Tile value out of bounds") self._tiles[index] = value @property def width(self) -> int: """Width in tiles""" return self._width @property def height(self) -> int: """Height in tiles""" return self._height @property def tile_width(self) -> int: """Width of each tile in pixels""" return self._tile_width @property def tile_height(self) -> int: """Height of each tile in pixels""" return self._tile_height