Source code for bruhanimate.bruhutil.bruhscreen

"""
Copyright 2023 Ethan Christensen
Copied, Guided, and Adapted from Asciimatics <https://github.com/peterbrittain/asciimatics/blob/master/asciimatics/screen.py>

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""

from __future__ import absolute_import, division, print_function, unicode_literals

import os
import signal
import sys

ENABLE_EXTENDED_FLAGS = 0x0080
ENABLE_QUICK_EDIT_MODE = 0x0040
ENABLE_VIRTUAL_TERMINAL_PROCESSING = 0x0004

if sys.platform == "win32":
    import pywintypes
    import win32con
    import win32console
    import win32event  # -->. for keyboard events
    import win32file

    class Screen:
        """
        Class for creating and managing a terminal screen in a WINDOWS OS terminal
        """

        # Standard extended key codes.
        KEY_ESCAPE = -1
        KEY_F1 = -2
        KEY_F2 = -3
        KEY_F3 = -4
        KEY_F4 = -5
        KEY_F5 = -6
        KEY_F6 = -7
        KEY_F7 = -8
        KEY_F8 = -9
        KEY_F9 = -10
        KEY_F10 = -11
        KEY_F11 = -12
        KEY_F12 = -13
        KEY_F13 = -14
        KEY_F14 = -15
        KEY_F15 = -16
        KEY_F16 = -17
        KEY_F17 = -18
        KEY_F18 = -19
        KEY_F19 = -20
        KEY_F20 = -21
        KEY_F21 = -22
        KEY_F22 = -23
        KEY_F23 = -24
        KEY_F24 = -25
        KEY_PRINT_SCREEN = -100
        KEY_INSERT = -101
        KEY_DELETE = -102
        KEY_HOME = -200
        KEY_END = -201
        KEY_LEFT = -203
        KEY_UP = -204
        KEY_RIGHT = -205
        KEY_DOWN = -206
        KEY_PAGE_UP = -207
        KEY_PAGE_DOWN = -208
        KEY_BACK = -300
        KEY_TAB = -301
        KEY_BACK_TAB = -302
        KEY_NUMPAD0 = -400
        KEY_NUMPAD1 = -401
        KEY_NUMPAD2 = -402
        KEY_NUMPAD3 = -403
        KEY_NUMPAD4 = -404
        KEY_NUMPAD5 = -405
        KEY_NUMPAD6 = -406
        KEY_NUMPAD7 = -407
        KEY_NUMPAD8 = -408
        KEY_NUMPAD9 = -409
        KEY_MULTIPLY = -410
        KEY_ADD = -411
        KEY_SUBTRACT = -412
        KEY_DECIMAL = -413
        KEY_DIVIDE = -414
        KEY_CAPS_LOCK = -500
        KEY_NUM_LOCK = -501
        KEY_SCROLL_LOCK = -502
        KEY_SHIFT = -600
        KEY_CONTROL = -601
        KEY_MENU = -602

        def __init__(self, stdout, stdin, old_out, old_in):
            info = stdout.GetConsoleScreenBufferInfo()["Window"]
            self.width = info.Right - info.Left + 1
            self.height = info.Bottom - info.Top + 1
            self._stdout = stdout
            self._stdin = stdin
            self._last_width = self.width
            self._last_height = self.height
            self._last_start = 0
            self._old_out = old_out
            self._old_in = old_in
            self._current_x = 0
            self._current_y = 0
            self._buffering = False
            self._frame_parts = []
            self._KEY_MAP = {
                win32con.VK_ESCAPE: Screen.KEY_ESCAPE,
                win32con.VK_F1: Screen.KEY_F1,
                win32con.VK_F2: Screen.KEY_F2,
                win32con.VK_F3: Screen.KEY_F3,
                win32con.VK_F4: Screen.KEY_F4,
                win32con.VK_F5: Screen.KEY_F5,
                win32con.VK_F6: Screen.KEY_F6,
                win32con.VK_F7: Screen.KEY_F7,
                win32con.VK_F8: Screen.KEY_F8,
                win32con.VK_F9: Screen.KEY_F9,
                win32con.VK_F10: Screen.KEY_F10,
                win32con.VK_F11: Screen.KEY_F11,
                win32con.VK_F12: Screen.KEY_F12,
                win32con.VK_F13: Screen.KEY_F13,
                win32con.VK_F14: Screen.KEY_F14,
                win32con.VK_F15: Screen.KEY_F15,
                win32con.VK_F16: Screen.KEY_F16,
                win32con.VK_F17: Screen.KEY_F17,
                win32con.VK_F18: Screen.KEY_F18,
                win32con.VK_F19: Screen.KEY_F19,
                win32con.VK_F20: Screen.KEY_F20,
                win32con.VK_F21: Screen.KEY_F21,
                win32con.VK_F22: Screen.KEY_F22,
                win32con.VK_F23: Screen.KEY_F23,
                win32con.VK_F24: Screen.KEY_F24,
                win32con.VK_PRINT: Screen.KEY_PRINT_SCREEN,
                win32con.VK_INSERT: Screen.KEY_INSERT,
                win32con.VK_DELETE: Screen.KEY_DELETE,
                win32con.VK_HOME: Screen.KEY_HOME,
                win32con.VK_END: Screen.KEY_END,
                win32con.VK_LEFT: Screen.KEY_LEFT,
                win32con.VK_UP: Screen.KEY_UP,
                win32con.VK_RIGHT: Screen.KEY_RIGHT,
                win32con.VK_DOWN: Screen.KEY_DOWN,
                win32con.VK_PRIOR: Screen.KEY_PAGE_UP,
                win32con.VK_NEXT: Screen.KEY_PAGE_DOWN,
                win32con.VK_BACK: Screen.KEY_BACK,
                win32con.VK_TAB: Screen.KEY_TAB,
            }
            self._keys = set()
            self._map_all = False

        def close(self, restore=True):
            """
            Close the screen and restore the previous console buffer and input modes.

            :param restore: Boolean flag to restore the original console buffer and input mode
                            when closing the screen. Default is True.
            """
            if restore:
                self._old_out.SetConsoleActiveScreenBuffer()
                self._stdin.SetConsoleMode(self._old_in)

        def print_at(self, text, x, y, width):
            """
            Print text at a specified position (x, y) on the console screen.

            :param text: The text to print on the screen.
            :param x: The x-coordinate where the text begins.
            :param y: The y-coordinate where the text begins.
            :param width: The width of the text to be displayed, used to update the cursor position.
            """
            try:
                if self._buffering:
                    cursor = (
                        f"\x1b[{y + 1};{x + 1}H"
                        if x != self._current_x or y != self._current_y
                        else ""
                    )
                    self._frame_parts.append(cursor + str(text))
                    self._current_x = x + width
                    self._current_y = y
                else:
                    if x != self._current_x or y != self._current_y:
                        self._stdout.SetConsoleCursorPosition(
                            win32console.PyCOORDType(x, y)
                        )
                    self._stdout.WriteConsole(str(text))
                    self._current_x = x + width
                    self._current_y = y
            except pywintypes.error:
                pass

        def begin_frame(self):
            """Enter frame-batching mode; print_at() calls accumulate until flush_frame()."""
            self._buffering = True
            self._frame_parts = []

        def flush_frame(self):
            """
            Flush accumulated frame as a single WriteConsole call.
            ANSI cursor-positioning sequences replace SetConsoleCursorPosition,
            reducing Win32 syscalls from O(changed_cells) to 1 per frame.
            Requires ENABLE_VIRTUAL_TERMINAL_PROCESSING on the console output buffer.
            """
            if self._frame_parts:
                try:
                    self._stdout.WriteConsole("".join(self._frame_parts))
                except pywintypes.error:
                    pass
                self._frame_parts = []
            self._buffering = False

        def print_center(self, text, y, width):
            """
            Print text centrally aligned horizontally at a given y-coordinate.

            :param text: The text to print on the screen.
            :param y: The y-coordinate where the text is to be centered.
            :param width: The width of the text to be displayed, used to calculate and update the left padding and cursor position.
            """
            try:
                left_pad = (self.width - width) // 2
                self._stdout.SetConsoleCursorPosition(
                    win32console.PyCOORDType(left_pad, y)
                )
                self._stdout.WriteConsole(str(text))
                self._current_x = left_pad + width
                self._current_y = y
            except pywintypes.error:
                pass

        def clear(self):
            """
            Clear the console screen by filling the console buffer with spaces and resetting the cursor position.
            """
            info = self._stdout.GetConsoleScreenBufferInfo()["Window"]
            width = info.Right - info.Left + 1
            height = info.Bottom - info.Top + 1
            box_size = width * height
            self._stdout.FillConsoleOutputAttribute(
                0, box_size, win32console.PyCOORDType(0, 0)
            )
            self._stdout.FillConsoleOutputCharacter(
                " ", box_size, win32console.PyCOORDType(0, 0)
            )
            self._stdout.SetConsoleCursorPosition(win32console.PyCOORDType(0, 0))
            self._current_x = 0
            self._current_y = 0

        def has_resized(self):
            """
            Determine if the console screen has been resized since the last check.

            :return: Boolean indicating whether the screen has been resized.
            """
            re_sized = False
            info = self._stdout.GetConsoleScreenBufferInfo()["Window"]
            width = info.Right - info.Left + 1
            height = info.Bottom - info.Top + 1
            if width != self._last_width or height != self._last_height:
                re_sized = True
            return re_sized

        def set_title(self, title: str) -> None:
            """
            Set the console window title.

            :param title: The title to set for the console window.
            """
            win32console.SetConsoleTitle(title)

        def wait_for_input(self, timeout: int) -> None:
            """
            Wait for user input in the console with a specified timeout period.

            :param timeout: The time in seconds to wait for user input before timing out.
            :raises RuntimeError: If an unexpected return code is encountered during the wait.
            """
            rc = win32event.WaitForSingleObject(self._stdin, int(timeout * 1000))
            if rc not in [0, 258]:
                raise RuntimeError(rc)

        def get_event(self):
            """
            Check for any console input event without waiting and return the corresponding key code.

            This method processes the input events in the console's event queue to find key press events.
            It translates these key events into a keyboard event object and returns the appropriate key code.
            Modifier states are not mapped in a cross-platform manner, but standard bindings for extended keys are considered.

            :return: The translated key code for the key event if available, otherwise None.
            """
            # Look for a new event and consume it if there is one.
            while len(self._stdin.PeekConsoleInput(1)) > 0:
                event = self._stdin.ReadConsoleInput(1)[0]
                if event.EventType == win32console.KEY_EVENT:
                    # Handle key-down events and Alt key states for unicode pasting.
                    key_code = ord(event.Char)
                    if event.KeyDown or (
                        key_code > 0
                        and key_code not in self._keys
                        and event.VirtualKeyCode == win32con.VK_MENU
                    ):
                        # Record any keys that were pressed.
                        if event.KeyDown:
                            self._keys.add(key_code)

                        # Translate keys into a KeyboardEvent object.
                        if event.VirtualKeyCode in self._KEY_MAP:
                            key_code = self._KEY_MAP[event.VirtualKeyCode]

                        # Handle cross-platform key mapping or any special mappings.
                        if (
                            self._map_all
                            and event.VirtualKeyCode in self._EXTRA_KEY_MAP
                        ):
                            key_code = self._EXTRA_KEY_MAP[event.VirtualKeyCode]
                        elif (
                            event.VirtualKeyCode == win32con.VK_TAB
                            and event.ControlKeyState & win32con.SHIFT_PRESSED
                        ):
                            key_code = Screen.KEY_BACK_TAB
                        elif event.VirtualKeyCode == win32con.VK_RETURN:
                            key_code = 10
                        elif event.VirtualKeyCode == win32con.VK_TAB:
                            key_code = 11

                        # Return the key code if a valid mapping exists.
                        if key_code:
                            return key_code
                    else:
                        # Remove any previously pressed key that is no longer down.
                        if key_code in self._keys:
                            self._keys.remove(key_code)

            # Return None if no interesting event was found.
            return None

        @classmethod
        def open(cls):
            """
            Opens and initializes a new console screen buffer for the Screen class on a Windows platform.

            This method creates a new console screen buffer, sets it as the active buffer, configures
            input and output modes, and prepares the screen for rendering text. It disables the cursor
            visibility and scroll settings to provide an optimized setup for terminal-based applications.

            :return: An initialized Screen object with the configured console screen buffer.
            """
            old_out = win32console.PyConsoleScreenBufferType(
                win32file.CreateFile(
                    "CONOUT$",
                    win32file.GENERIC_READ | win32file.GENERIC_WRITE,
                    win32file.FILE_SHARE_WRITE,
                    None,
                    win32file.OPEN_ALWAYS,
                    0,
                    None,
                )
            )
            try:
                info = old_out.GetConsoleScreenBufferInfo()
            except pywintypes.error:
                info = None
            win_out = win32console.CreateConsoleScreenBuffer()
            if info:
                win_out.SetConsoleScreenBufferSize(info["Size"])
            else:
                win_out.SetStdHandle(win32console.STD_OUTPUT_HANDLE)
            win_out.SetConsoleActiveScreenBuffer()
            win_in = win32console.PyConsoleScreenBufferType(
                win32file.CreateFile(
                    "CONIN$",
                    win32file.GENERIC_READ | win32file.GENERIC_WRITE,
                    win32file.FILE_SHARE_READ,
                    None,
                    win32file.OPEN_ALWAYS,
                    0,
                    None,
                )
            )
            win_in.SetStdHandle(win32console.STD_INPUT_HANDLE)
            # Hide Cursor
            win_out.SetConsoleCursorInfo(1, 0)
            # Disable scroll and enable ANSI/VT sequence processing
            out_mode = win_out.GetConsoleMode()
            out_mode &= ~win32console.ENABLE_WRAP_AT_EOL_OUTPUT
            out_mode |= ENABLE_VIRTUAL_TERMINAL_PROCESSING
            try:
                win_out.SetConsoleMode(out_mode)
            except pywintypes.error:
                # Older Windows without VT support — strip the VT flag and fall back
                win_out.SetConsoleMode(out_mode & ~ENABLE_VIRTUAL_TERMINAL_PROCESSING)
            in_mode = win_in.GetConsoleMode()
            new_mode = in_mode | win32console.ENABLE_MOUSE_INPUT | ENABLE_EXTENDED_FLAGS
            new_mode &= ~ENABLE_QUICK_EDIT_MODE
            # new_mode &= ~win32console.ENABLE_PROCESSED_INPUT
            win_in.SetConsoleMode(new_mode)

            screen = Screen(win_out, win_in, old_out, in_mode)
            return screen

        @classmethod
        def show(cls, function, args=None):
            """
            Display the provided function's output on the terminal screen with optional arguments.

            This method initializes the terminal screen, invokes the provided function, and passes
            any additional arguments to it. After the function execution, the screen is closed to
            restore the previous terminal settings.

            :param function: A callable to be executed within the context of the screen.
            :param args: An optional list of arguments to be passed to the function.
            """
            screen = Screen.open()
            try:
                if args:
                    return function(screen, *args)
                else:
                    return function(screen)
            finally:
                screen.close()

else:
    import curses

[docs] class Screen: KEY_ESCAPE = -1 KEY_F1 = -2 KEY_F2 = -3 KEY_F3 = -4 KEY_F4 = -5 KEY_F5 = -6 KEY_F6 = -7 KEY_F7 = -8 KEY_F8 = -9 KEY_F9 = -10 KEY_F10 = -11 KEY_F11 = -12 KEY_F12 = -13 KEY_F13 = -14 KEY_F14 = -15 KEY_F15 = -16 KEY_F16 = -17 KEY_F17 = -18 KEY_F18 = -19 KEY_F19 = -20 KEY_F20 = -21 KEY_F21 = -22 KEY_F22 = -23 KEY_F23 = -24 KEY_F24 = -25 KEY_PRINT_SCREEN = -100 KEY_INSERT = -101 KEY_DELETE = -102 KEY_HOME = -200 KEY_END = -201 KEY_LEFT = -203 KEY_UP = -204 KEY_RIGHT = -205 KEY_DOWN = -206 KEY_PAGE_UP = -207 KEY_PAGE_DOWN = -208 KEY_BACK = -300 KEY_TAB = -301 KEY_BACK_TAB = -302 KEY_NUMPAD0 = -400 KEY_NUMPAD1 = -401 KEY_NUMPAD2 = -402 KEY_NUMPAD3 = -403 KEY_NUMPAD4 = -404 KEY_NUMPAD5 = -405 KEY_NUMPAD6 = -406 KEY_NUMPAD7 = -407 KEY_NUMPAD8 = -408 KEY_NUMPAD9 = -409 KEY_MULTIPLY = -410 KEY_ADD = -411 KEY_SUBTRACT = -412 KEY_DECIMAL = -413 KEY_DIVIDE = -414 KEY_CAPS_LOCK = -500 KEY_NUM_LOCK = -501 KEY_SCROLL_LOCK = -502 KEY_SHIFT = -600 KEY_CONTROL = -601 KEY_MENU = -602
[docs] def get_event(self): c = self.screen.getch() if c != curses.ERR: if c in (curses.KEY_BACKSPACE, 127, 8): return self.KEY_BACK elif c in (10, 13, curses.KEY_ENTER): return 10 elif c == 27: return self.KEY_ESCAPE return c return None
[docs] def __init__(self, window, height=None): self.screen = window self.screen.keypad(1) self.height = window.getmaxyx()[0] self.width = window.getmaxyx()[1] curses.curs_set(0) self.screen.nodelay(1) self.signal_state = SignalState() self._re_sized = False self.signal_state.set(signal.SIGWINCH, self._resize_handler) self.signal_state.set(signal.SIGCONT, self._continue_handler) curses.mousemask(curses.ALL_MOUSE_EVENTS | curses.REPORT_MOUSE_POSITION) self._move_y_x = curses.tigetstr("cup") self._up_line = curses.tigetstr("ri").decode("utf-8") self._down_line = curses.tigetstr("ind").decode("utf-8") self._fg_color = curses.tigetstr("setaf") self._bg_color = curses.tigetstr("setab") self._default_colours = curses.tigetstr("op") if curses.tigetflag("hs"): self._start_title = curses.tigetstr("tsl").decode("utf-8") self._end_title = curses.tigetstr("fsl").decode("utf-8") else: self._start_title = self._end_title = None self._a_normal = curses.tigetstr("sgr0").decode("utf-8") self._a_bold = curses.tigetstr("bold").decode("utf-8") self._a_reverse = curses.tigetstr("rev").decode("utf-8") self._a_underline = curses.tigetstr("smul").decode("utf-8") self._clear_screen = curses.tigetstr("clear").decode("utf-8") self._bytes_to_read = 0 self._bytes_to_return = b"" self._cur_x = 0 self._cur_y = 0 self._buffering = False self._frame_parts = []
def _resize_handler(self, *_): curses.endwin() curses.initscr() self._re_sized = True def _continue_handler(self, *_): self.force_update(full_refresh=True)
[docs] def has_resized(self): return self._re_sized
[docs] def close(self, restore=True): self.signal_state.restore() if restore: self.screen.keypad(0) curses.echo() curses.nocbreak() curses.endwin()
[docs] def clear(self): try: sys.stdout.flush() os.system("clear") self._cur_x = 0 self._cur_y = 0 except IOError: pass
[docs] @classmethod def open(cls): stdcrs = curses.initscr() curses.noecho() curses.cbreak() stdcrs.keypad(1) screen = Screen(stdcrs) return screen
@staticmethod def _safe_write(msg): try: sys.stdout.write(msg) except IOError: pass
[docs] def print_at(self, text, x, y, width): cursor = "" if x != self._cur_x or y != self._cur_y: cursor = curses.tparm(self._move_y_x, y, x).decode("utf-8") try: piece = cursor + str(text) if self._buffering: self._frame_parts.append(piece) else: self._safe_write(piece) except UnicodeEncodeError: piece = cursor + "?" * len(str(text)) if self._buffering: self._frame_parts.append(piece) else: self._safe_write(piece) self._cur_x = x + width self._cur_y = y
[docs] def begin_frame(self): """Enter frame-batching mode; print_at() calls accumulate until flush_frame().""" self._buffering = True self._frame_parts = []
[docs] def flush_frame(self): """ Flush accumulated frame output as a single sys.stdout.write() + flush(). Goes from O(cells_changed) write() calls to 1 write + 1 flush per frame. """ if self._frame_parts: self._safe_write("".join(self._frame_parts)) self._frame_parts = [] sys.stdout.flush() self._buffering = False
[docs] @classmethod def show(cls, function, args=None): os.system("clear") screen = Screen.open() try: if args: return function(screen, *args) else: return function(screen) finally: screen.close() input() os.system("clear")
[docs] class SignalState(object):
[docs] def __init__(self): self._old_signal_states = []
[docs] def set(self, signalnum, handler): old_handler = signal.getsignal(signalnum) if old_handler is None: old_handler = signal.SIG_DFL self._old_signal_states.append((signalnum, old_handler)) signal.signal(signalnum, handler)
[docs] def restore(self): for signalnum, handler in self._old_signal_states: signal.signal(signalnum, handler) self._old_signal_states = []