|
2 | 2 | Recycle microbits exercises in python using 5 * 5 strings and some coloring.
|
3 | 3 | """
|
4 | 4 |
|
5 |
| -TODO = 42 |
| 5 | +__all__ = ["Pycrobit", "DEFAULT_COLOR_MAP"] |
| 6 | + |
| 7 | +import os |
| 8 | +import time |
| 9 | +from typing import Dict |
| 10 | + |
| 11 | +from colorama import Fore, init |
| 12 | +from colorama.ansi import AnsiFore |
| 13 | + |
| 14 | +init() |
| 15 | +COLORAMA_AVAILABLE = True |
| 16 | +DEFAULT_COLOR_MAP = {".": "", "*": Fore.RED} |
| 17 | +ColorMap = Dict[str, AnsiFore] |
| 18 | + |
| 19 | + |
| 20 | +def validate_pycrobit_str(pycrobit_string: str) -> None: |
| 21 | + """Validate that the pycrobit string is correct.""" |
| 22 | + strip_line_feed = pycrobit_string.strip("\n ") |
| 23 | + lines = strip_line_feed.split("\n") |
| 24 | + assert len(lines) == 5, f"We expected 5 lines in the string and got {lines} " |
| 25 | + for i, line in enumerate(lines): |
| 26 | + stripped_line = line.strip(" ") |
| 27 | + if len(stripped_line) != 5: |
| 28 | + raise ValueError( |
| 29 | + f"We expected 5 characters on line n°{i+1} and got " |
| 30 | + f"{len(stripped_line)} ({stripped_line})" |
| 31 | + ) |
| 32 | + |
| 33 | + |
| 34 | +def validate_color_map(color_map: ColorMap) -> None: |
| 35 | + """Validate that the color map is correct.""" |
| 36 | + for character, color in color_map.items(): |
| 37 | + if not isinstance(color, str): |
| 38 | + msg = f"The color '{color}' for '{character}' is invalid." |
| 39 | + raise ValueError(msg) |
| 40 | + |
| 41 | + |
| 42 | +def clear_terminal() -> None: |
| 43 | + """Clear the terminal (refresh the screen).""" |
| 44 | + os.system("cls" if os.name == "nt" else "clear") |
| 45 | + |
| 46 | + |
| 47 | +def colorize(pycrobit_string: str, color_map: ColorMap) -> str: |
| 48 | + """Colorize a string to be displayed in terminal.""" |
| 49 | + validate_pycrobit_str(pycrobit_string) |
| 50 | + validate_color_map(color_map) |
| 51 | + return "".join(f"{color_map.get(px, '')}{px}{Fore.RESET}" for px in pycrobit_string) |
| 52 | + |
| 53 | + |
| 54 | +class Pycrobit: |
| 55 | + |
| 56 | + """A context manager that act similarly as a microbit but in Python.""" |
| 57 | + |
| 58 | + DEFAULT_FRAMERATE_SECOND = 1.0 |
| 59 | + |
| 60 | + def __init__(self, framerate=DEFAULT_FRAMERATE_SECOND): |
| 61 | + # No delay for the first display |
| 62 | + self.framerate = framerate |
| 63 | + self._switch_frame_after = 0 |
| 64 | + |
| 65 | + def wait(self, time_to_wait: float = 0.0): |
| 66 | + """Wait for an additional 'time_to_wait'. |
| 67 | +
|
| 68 | + It can be negative to accelerate the framerate.""" |
| 69 | + if self._switch_frame_after + time_to_wait < 0: |
| 70 | + raise ValueError("pycrobit was asked to wait for a negative time.") |
| 71 | + self._switch_frame_after += time_to_wait |
| 72 | + |
| 73 | + def display(self, pycrobit_string: str, color_map: ColorMap = None): |
| 74 | + """Display the string""" |
| 75 | + # print(f"entering {pycrobit_string}") |
| 76 | + clear_terminal() |
| 77 | + if color_map is None: |
| 78 | + color_map = DEFAULT_COLOR_MAP |
| 79 | + print(colorize(pycrobit_string, color_map)) |
| 80 | + time.sleep(self._switch_frame_after) |
| 81 | + self._switch_frame_after = self.framerate |
0 commit comments