From: Melissa LeBlanc-Williams Date: Thu, 25 Jun 2020 21:47:21 +0000 (-0700) Subject: Image processing optimizations X-Git-Tag: 0.4.0^2~2 X-Git-Url: https://git.ayoreis.com/hackapet/Adafruit_Blinka_Displayio.git/commitdiff_plain/e7f558f790efd2492cb53f43c077dce21fb5428b Image processing optimizations --- diff --git a/displayio/bitmap.py b/displayio/bitmap.py index 1bc90aa..7ac075c 100644 --- a/displayio/bitmap.py +++ b/displayio/bitmap.py @@ -36,6 +36,7 @@ displayio for Blinka """ from recordclass import recordclass +from PIL import Image __version__ = "0.0.0-auto.0" __repo__ = "https://github.com/adafruit/Adafruit_Blinka_displayio.git" @@ -75,7 +76,7 @@ class Bitmap: ): raise NotImplementedError("Invalid bits per value") - self._data = (width * height) * [0] + self._image = Image.new("P", (width, height), 0) self._dirty_area = Rectangle(0, 0, width, height) def __getitem__(self, index): @@ -84,10 +85,16 @@ class Bitmap: an x,y tuple or an int equal to `y * width + x`. """ if isinstance(index, (tuple, list)): - index = (index[1] * self._width) + index[0] - if index >= len(self._data): + x, y = index + elif isinstance(index, int): + x = index % self._width + y = index // self._width + else: + raise TypeError("Index is not an int, list, or tuple") + + if x > self._image.width or y > self._image.height: raise ValueError("Index {} is out of range".format(index)) - return self._data[index] + return self._image.getpixel((x, y)) def __setitem__(self, index, value): """ @@ -103,7 +110,7 @@ class Bitmap: elif isinstance(index, int): x = index % self._width y = index // self._width - self._data[index] = value + self._image.putpixel((x, y), value) if self._dirty_area.x1 == self._dirty_area.x2: self._dirty_area.x1 = x self._dirty_area.x2 = x + 1 @@ -125,7 +132,7 @@ class Bitmap: def fill(self, value): """Fills the bitmap with the supplied palette index value.""" - self._data = (self._width * self._height) * [value] + self._image = Image.new("P", (self._width, self._height), value) self._dirty_area = Rectangle(0, 0, self._width, self._height) @property diff --git a/displayio/display.py b/displayio/display.py index 84cdc25..eb3d635 100644 --- a/displayio/display.py +++ b/displayio/display.py @@ -353,7 +353,7 @@ class Display: if 0 <= float(value) <= 1.0: self._brightness = value if self._backlight_type == BACKLIGHT_IN_OUT: - self._backlight.value = int(round(self._brightness)) + self._backlight.value = round(self._brightness) # PWM not currently implemented # Command-based brightness not implemented else: diff --git a/displayio/palette.py b/displayio/palette.py index 1812dbf..ae3d27c 100644 --- a/displayio/palette.py +++ b/displayio/palette.py @@ -111,3 +111,19 @@ class Palette: """Set the palette index to be an opaque color""" self._colors[palette_index]["transparent"] = False self._update_rgba(palette_index) + + def _get_palette(self): + """Generate a palette for use with PIL""" + palette = [] + for color in self._colors: + palette += color["rgba"][0:3] + return palette + + def _get_alpha_palette(self): + """Generate an alpha channel palette with white being + opaque and black being transparent""" + palette = [] + for color in self._colors: + for _ in range(3): + palette += [0 if color["transparent"] else 255] + return palette diff --git a/displayio/tilegrid.py b/displayio/tilegrid.py index 7308342..861384e 100644 --- a/displayio/tilegrid.py +++ b/displayio/tilegrid.py @@ -197,6 +197,25 @@ class TileGrid: return self._pixel_shader.convert(pixel_value) return pixel_value + def _apply_palette(self, image): + if isinstance(self._pixel_shader, Palette): + image.putpalette( + self._pixel_shader._get_palette() + ) # pylint: disable=protected-access + if isinstance(self._pixel_shader, ColorConverter): + # This will be needed for eInks, grayscale, and monochrome displays + pass + + def _add_alpha(self, image): + if isinstance(self._pixel_shader, Palette): + alpha = self._bitmap._image.copy().convert( + "P" + ) # pylint: disable=protected-access + alpha.putpalette( + self._pixel_shader._get_alpha_palette() + ) # pylint: disable=protected-access + image.putalpha(alpha.convert("L")) + # pylint: disable=too-many-locals def _fill_area(self, buffer): """Draw onto the image""" @@ -218,14 +237,20 @@ class TileGrid: tile_index = self._tiles[tile_y * self._width + tile_x] tile_index_x = tile_index % tile_count_x tile_index_y = tile_index // tile_count_x - for pixel_x in range(self._tile_width): - for pixel_y in range(self._tile_height): - image_x = (tile_x * self._tile_width) + pixel_x - image_y = (tile_y * self._tile_height) + pixel_y - bitmap_x = (tile_index_x * self._tile_width) + pixel_x - bitmap_y = (tile_index_y * self._tile_height) + pixel_y - pixel_color = self._shade(self._bitmap[bitmap_x, bitmap_y]) - image.putpixel((image_x, image_y), pixel_color) + tile_image = self._bitmap._image.copy().convert( + "P" + ) # pylint: disable=protected-access + self._apply_palette(tile_image) + tile_image = tile_image.convert("RGBA") + self._add_alpha(tile_image) + image.alpha_composite( + tile_image, + dest=(tile_x * self._tile_width, tile_y * self._tile_height), + source=( + tile_index_x * self._tile_width, + tile_index_y * self._tile_height, + ), + ) if self._absolute_transform is not None: if self._absolute_transform.scale > 1: