Source code for thurible.panel

"""
.. _panels:

######
Panels
######

:dfn:`Panels` format and display data in a terminal. If you are used
to working with graphical user interfaces, you can think of them like
windows. They are how the application puts data on the screen.

.. _interfaces:

Common Interfaces
*****************

In :mod:`thurible`, a panel is a subclass of :class:`thurible.panel.Panel.`
That class and a couple additional protocols define much of the common
behavior of panel objects.

For more details on how panels are sized within the terminal, see
:ref:`sizing`.

For more details on how to define how panels react to user input,
see :ref:`active`.

.. autoclass:: thurible.panel.Panel
    :members:
.. autoclass:: thurible.panel.Frame
    :members:
.. autoclass:: thurible.panel.Content
    :members:
.. autoclass:: thurible.panel.Scroll
    :members:
.. autoclass:: thurible.panel.Title
    :members:

.. _sizing:

Sizing Panels
*************
Panels attempt to allow for the relative sizing of an element
within a terminal. What does that mean?

A terminal window has a size in rows and columns. These rows
and columns are measured in relation to a fixed-width character.
A row is the height of one character. A column is the width of
one character. For reasons that go back to the era of punch
cards and hardware terminals, the common default size of a
terminal window is 24 rows by 80 columns.

However, terminal widows do not have to be that standard size.
Most terminal emulators that I've used allow you to set any
size you want for the size of the terminal window, and you can
resize the window after you open it. That creates a problem if
you are trying to create a consistent interface for a terminal
application. Sure, you can usually assume that a terminal is
going to be 24×80, but if you run into a terminal that is
48×132, things might get weird.

Panel tries to solve that by allowing you to set the size of
a panel relative to the terminal window, no matter what size
that terminal window is. Now, there are some limitations to that.
If the terminal window is 1×1, there isn't much that can be
shown in that terminal. However, it still should be useful for
most terminal sizes you are going to run into.

.. _position_onion:

The Positioning Onion
^^^^^^^^^^^^^^^^^^^^^

While it varies a little depending upon what classes a panel
inherits, there are generally four layers of positioning that
can happen within a panel:

*   Absolute
*   Frame (requires inheriting from :class:`thurible.panel.Frame`)
*   Inner
*   Content (requires inheriting from :class:`thurible.panel.Content`)

.. _absolute-layer:

The Absolute Layer
==================

The absolute layer is the first layer, and it is tracked in the
following attributes:

height
    The number of rows from the top of the panel to the bottom.
    If you don't specify a height, it will default to the
    total number of rows in the current terminal window.
width
    The number of columns from the left side of the panel to
    the right side. If you don't specify a width, it will
    default to the number of columns in the current terminal
    window.
origin_x
    The left-most column of the panel. If you don't specify an
    origin_x, it will default to the left-most row of the
    terminal window.
origin_y
    The top-most row of the panel. If you don't specify an
    origin_y, it will default to the top-most row of the
    terminal window.

.. note::
    While you can set these manually, doing so will prevent the
    panel from being resized when the terminal is resized. The
    only intended use case for setting these is to simplify
    the writing of unit tests. In all other cases, it's recommended
    to allow these to default to the full size of the terminal
    window and use other parameters to position the panel.

.. _frame-layer:

The Frame Layer
===============

For subclasses of :class:`thurible.panel.Frame`, the next layer in
positions the panel's frame. It's tracked by the following properties:

frame_height
    The number of rows from the top of the panel's frame to the bottom.
frame_width
    The number of columns from the left side of the panel's frame to
    the right side.
frame_x
    The left-most column of the panel's frame.
frame_y
    The top-most row of the panel's frame.

These are properties, so they cannot be set directly. You have to use
relative positioning attributes to affect these. That is done with
the attributes described in the :ref:`inner-layer` below.

.. _inner-layer:

The Inner Layer
===============

The next positioning layer is the :dfn:`inner layer`. It's tracked
by the following properties:

inner_height
    The number of rows from the top of the panel's interior to the
    bottom.
inner_width
    The number of columns from the left side of the panel's
    interior to the right side.
inner_x
    The left-most column of the panel's interior.
inner_y
    The top-most row of the panel's interior.

These are properties, so they cannot be set directly. You have to
use the following attributes to set their value:

panel_relative_height
    The proportional height of the interior of the panel as
    compared to the absoute height of the panel. The proportion
    is given a :class:`float` with a value between 0.0 and 1.0,
    inclusive.
panel_relative_width
    The proportional width of the interior of the panel as
    compared to the absoute width of the panel. The proportion
    is given a :class:`float` with a value between 0.0 and 1.0,
    inclusive.
panel_align_h
    The horizontal alignment of the interior of the panel within
    the absolute width of the panel. The alignment is given as
    one of the following strings: "left", "center", or "right".
panel_align_v
    The vertical alignment of the interior of the panel within
    the absolute height of the panel. The alignment is given
    as one of the following strings: "top", "middle", "bottom".

.. note::
    There are two additional attributes that are involved:
    `panel_pad_left` and `panel_pad_right`. While it is
    currently possible to set these yourself, that
    ability will likely be removed in future versions. It
    is recommended to avoid using these two attributes.

So how does this actually work? Let's say we have the following
panel in a standard 80 character wide terminal:

.. testcode::

    import thurible

    panel = thurible.panel.Panel(
        origin_x=0,
        panel_relative_width=0.8,
        panel_align_h='right'
    )

The width of the interior of the panel, as given by
:attr:`panel.inner_width` is:

.. testsetup:: inner_layer

    import thurible

    panel = thurible.panel.Panel(
        origin_x=0,
        width=80,
        panel_relative_width=0.8,
        panel_align_h='right'
    )

.. testcode:: inner_layer

    panel.width == 80
    panel.panel_relative_width == 0.8
    int(panel.width * panel.panel_relative_width) == 64

Since the interior is aligned "right", the starting point of the
interior, as given by :attr:`panel.inner_x` is:

.. testcode:: inner_layer

    panel.origin_x == 0
    panel.width == 80
    panel.inner_width == 16
    panel.origin_x + (panel.width - panel.inner_width) == 16

Had the alignment been "center", it would have been:

.. testcode:: inner_layer

    panel.origin_x == 0
    panel.width == 80
    panel.inner_width == 16
    panel.origin_x + (panel.width - panel.inner_width) // 2 == 8

Had the alignment been "left", it would have been:

.. testcode:: inner_layer

    panel.origin_x == 0

.. warning::
    The above is not completely accurate for the current version
    of :mod:`thurible`. Instead, the panel uses
    :attr:`panel.panel_relative_width` to calculate the values of
    :attr:`panel.panel_pad_left` and :attr:`panel.panel_pad_right`.
    It then uses those values to determine the values for the
    `inner_*` attributes. However, the explanation above should
    be close to the results provided by the actual method, and
    future versions of :mod:`thurible` should move to the model
    described above.

.. note::
    In subclasses of :class:`thurible.panel.Frame`, adding a
    frame will affect the values of the `inner_*` attributes.
    The frame shrinks the interior by one character on each side
    of the interior. Therefore, if there had been a frame in the
    example above, :attr:`panel.inner_width` would have been 62
    and the :attr:`panel.inner_x` would've been 17.

.. _content-layer:

The Content Layer
=================

For subclasses of :class:`thurible.panel.Content`, the next layer in
positions the panel's content. It's tracked by the following properties:

content_width
    The number of columns from the left side of the panel's content
    to the right side.
content_x
    The left-most column of the panel's content.

These are properties, so they cannot be set directly. You have to
use the following attributes to set their value:

content_relative_width
    The proportional width of the content of the panel as
    compared to the inner width of the panel. The proportion
    is given a :class:`float` with a value between 0.0 and 1.0,
    inclusive.
content_align_h
    The horizontal alignment of the content of the panel within
    the inner width of the panel. The alignment is given as
    one of the following strings: "left", "center", or "right".

.. note::
    There are two additional attributes that are involved:
    `content_pad_left` and `content_pad_right`. While it is
    currently possible to set these yourself, that
    ability will likely be removed in future versions. It
    is recommended to avoid using these two attributes.

These attributes work similar to the relative width and alignment
attributes in :ref:`inner-layer` above.


.. _active:

Active Keys
***********

An :dfn:`active key` is a keyboard key that, when pressed by the user,
will be intercepted and handled by the panel rather than passed on to
the application.

An :dfn:`action handler` is a method that accepts a key press, as
represented by a :class:`blessed.keyboard.Keystroke` object returned
by :meth:`blessed.Terminal.inkey.` It defines the behavior of the panel
when the key is pressed, and it returns a :class:`str` with any updates
that need to be made to the terminal display.

The :mod:`thurible` library displays data to the user of a terminal
application. In some cases, the user needs to navigate within that
data. For example, the text displayed by a panel may be longer than
the number of rows in the current terminal window, so the user needs
to scroll down in the text to read all of it. Given a menu of options
the user needs to select the option they want. :mod:`Thurible` panels
will handle this sort of navigation for you through these active keys
and action handlers.

.. note::
    Active keys do not send any data back to your application. It's
    not intended for your application to even be aware they were
    pressed. Any input that needs to go back to your application
    should be handled in :meth:`Panel.action` and returned as the data
    :class:`str.`

"""
from collections.abc import Callable
from typing import Optional

from blessed import Terminal
from blessed.keyboard import Keystroke

from thurible.util import Box, get_terminal


# Exceptions.
class InvalidDimensionsError(Exception):
    """The parameters that determine the relative width or height
    must add up to one.
    """


class InvalidTitleAlignmentError(Exception):
    """The value given for the title alignment was invalid."""


class NoFrameTypeForFrameError(Exception):
    """A paramters that requires a frame type was set without setting a
    frame type.
    """


class PanelPaddingAndAlignmentSetError(Exception):
    """You cannot set both panel padding and alignment."""


# Base classes.
class Message:
    """A base class to allow all messages to be identified."""


[docs] class Panel: """Create a new :class:`Panel` object. This class serves as a parent class for all panels, providing the core code relating to the area the panel fills in the terminal window. :param height: (Optional.) The height of the pane. :param width: (Optional.) The width of the pane. :param term: (Optional.) A :class:`blessed.Terminal` instance for formatting text for terminal display. :param origin_y: (Optional.) The terminal row for the top of the panel. :param origin_x: (Optional.) The terminal column for the left side of the panel. :param fg: (Optional.) A string describing the foreground color of the pane. See the documentation for :mod:`blessed` for more detail on the available options. :param bg: (Optional.) A string describing the background color of the pane. See the documentation for :mod:`blessed` for more detail on the available options. :param panel_pad_top: (Optional.) Distance between the Y origin of the panel and the top of the interior of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_relative_height: (Optional.) The height of the interior of the panel in comparison of the full `height` of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_pad_bottom: (Optional.) Distance between the full height of the panel and the interior of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_pad_left: (Optional.) Distance between the X origin of the panel and the left side of the interior of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_relative_width: (Optional.) The width of the interior of the panel in comparison of the full width of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_pad_right: (Optional.) Distance between the full width of the panel and the right side of the interior of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_align_h: (Optional.) If the interior of the panel is smaller than the full width of the panel, this sets how the interior of the panel is aligned within the full height. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :param panel_align_v: (Optional.) If the interior of the panel is smaller than the full width of the panel, this sets how the interior of the panel is aligned within the full height. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. See :ref:`sizing` for more information. :return: A :class:`Panel` object. :rtype: thurible.panel.Panel :usage: To create a new :class:`thurible.panel.Panel` subclass: .. testcode:: from thurible import panel class Spam(panel.Panel): def wobble(self): pass spam = Spam(height=10, width=10) Information on the sizing of :class:`thurible.panel.Panel` objects can be found in the :ref:`sizing` section below. """ # Magic methods. def __init__( self, height: Optional[int] = None, width: Optional[int] = None, term: Optional[Terminal] = None, origin_y: Optional[int] = None, origin_x: Optional[int] = None, bg: str = '', fg: str = '', panel_pad_bottom: Optional[float] = None, panel_pad_left: Optional[float] = None, panel_pad_right: Optional[float] = None, panel_pad_top: Optional[float] = None, panel_relative_height: Optional[float] = None, panel_relative_width: Optional[float] = None, panel_align_h: Optional[str] = None, panel_align_v: Optional[str] = None ) -> None: self.term = term if term else get_terminal() self._set_relative_horizontal_dimensions( panel_pad_left, panel_relative_width, panel_pad_right, panel_align_h ) self._set_relative_vertical_dimensions( panel_pad_top, panel_relative_height, panel_pad_bottom, panel_align_v ) self.height = height if height is not None else self.term.height self.width = width if width is not None else self.term.width self.origin_y = origin_y if origin_y else 0 self.origin_x = origin_x if origin_x else 0 self.bg = bg self.fg = fg # Private attributes. self._active_keys: dict[str, Callable] = {} def __eq__(self, other) -> bool: if not isinstance(other, self.__class__): return NotImplemented return ( self.height == other.height and self.width == other.width and self.origin_y == other.origin_y and self.origin_x == other.origin_x and self.bg == other.bg and self.fg == other.fg ) def __str__(self) -> str: """Return a string that will draw the entire display.""" result = '' result += self.clear_contents() return result # Properties @property def active_keys(self) -> dict[str, Callable]: """The key presses the class will react to and the handler that acts on that key press. :return: A :class:`dict` object where the keys are the representation of the :class:`blessed.keyboard.Keystroke` object emitted when the key is pressed and the values are the action handler methods called when the key is pressed. :rtype: dict """ return self._active_keys.copy() @property def inner_height(self) -> int: """The number of rows in the terminal contained within the interior of the panel. :return: An :class:`int` object. :rtype: int """ height = self.height height -= self._panel_pad_offset_top height -= self._panel_pad_offset_bottom return height @property def inner_width(self) -> int: """The number of columns in the terminal contained within the interior of the panel. :return: An :class:`int` object. :rtype: int """ width = self.width width -= self._panel_pad_offset_left width -= self._panel_pad_offset_right return width @property def inner_x(self) -> int: """The left-most column in the terminal of the interior of the panel. :return: An :class:`int` object. :rtype: int """ return self.origin_x + self._panel_pad_offset_left @property def inner_y(self) -> int: """The top-most row in the terminal of the interior of the panel. :return: An :class:`int` object. :rtype: int """ return self.origin_y + self._panel_pad_offset_top @property def _panel_pad_offset_bottom(self) -> int: offset = self.height * self.panel_pad_bottom return int(offset) @property def _panel_pad_offset_left(self) -> int: offset = self.width * self.panel_pad_left return int(offset) @property def _panel_pad_offset_right(self) -> int: offset = self.width * self.panel_pad_right return int(offset) @property def _panel_pad_offset_top(self) -> int: offset = self.height * self.panel_pad_top return int(offset) # Public methods.
[docs] def action(self, key: Keystroke) -> tuple[str, str]: """Act on a keystroke typed by the user. :param key: A :class:`blessed.keyboard.Keystroke` object representing the key pressed by the user. :return: A :class:`tuple` object containing two :class:`str` objects. The first string is any data that needs to be sent to the application. The second string contains any updates needed to be made to the terminal display. :rtype: tuple """ data = str(key) update = '' return (data, update)
[docs] def clear_contents(self) -> str: """Clear the interior area of the panel. :return: A :class:`str` object containing the update needed to be made to the terminal display. :rtype: str """ # Set up. height = self.inner_height width = self.inner_width y = self.inner_y x = self.inner_x color = self._get_color(self.fg, self.bg) result = color # Create the clearing string and return. for i in range(height): result += self.term.move(y + i, x) + ' ' * width if color: result += self.term.normal return result
[docs] def register_key(self, key: str, handler: Callable) -> None: """Declare the key presses the class will react to, and define the action the class will take when that key is pressed. :param key: The name of the key pressed as returned by the representation of the :class:`blessed.keyboard.Keystroke` emitted by the key press. :param handler: And action handler to invoke when the key is pressed. An action handler is a function that takes an optional :class:`blessed.keyboard.Keystroke` object and returns a string that contains any changes that need to be made to the terminal display as a result of the key press. :return: None. :rtype: NoneType """ items = [getattr(self, name) for name in self.__dict__] if handler not in items: setattr(self, key, handler) self._active_keys[key] = handler
[docs] def update(self, msg: Message) -> str: """Act on a message sent by the application. :param msg: A message sent by the application. :return: A :class:`str` object containing any updates needed to be made to the terminal display. :rtype: str """ return ''
# Private helper methods. def _get_color(self, fg: str = '', bg: str = '') -> str: color = fg if color and bg: color += f'_on_{bg}' elif bg: color += f'on_{bg}' return getattr(self.term, color) def _set_relative_dimenstion( self, left: Optional[float] = None, width: Optional[float] = None, right: Optional[float] = None, align: Optional[str] = None, align_default: str = 'center', alignments: tuple[str, str, str] = ('left', 'center', 'right'), attr_names: tuple[str, str, str] = ( 'panel_pad_left', 'panel_relative_width', 'panel_pad_right', ) ) -> tuple[float, float, float, str]: # This function needs to check which parameters are None a lot. # So, the answer to that is stored in this tuple to help # reduce the verbosity of the rest of the function. were_set = (left is not None, width is not None, right is not None) LEFT, WIDTH, RIGHT = 0, 1, 2 # Since we aren't directly checking these values to see if they # are None, mypy will get confused. So, replace Nones with 0.0 # to keep mypy happy. left = 0.0 if left is None else left width = 0.0 if width is None else width right = 0.0 if right is None else right # If both padding and alignment are set, raise an exception # because the intended behavior of the panel would be ambiguous. if align is not None and any((were_set[LEFT], were_set[RIGHT])): msg = 'Cannot set both panel padding and panel alignment.' raise PanelPaddingAndAlignmentSetError(msg) # If only width was set and align wasn't, set align to the default. elif align is None: align = align_default # If none are set, use the default values. if not any(were_set): left = 0.0 right = 0.0 width = 1.0 # If all three values are set, they must add up to one. elif all(were_set) and left + right + width != 1.0: msg = ( f'If {attr_names[LEFT]}, {attr_names[RIGHT]}, and ' f'{attr_names[WIDTH]} are set, the sum of ' 'the three must equal one. The given values were: ' f'{attr_names[LEFT]}={left}, ' f'{attr_names[RIGHT]}={right}, ' f'and {attr_names[WIDTH]}={width}.' ) raise InvalidDimensionsError(msg) # If only left was set, the rest goes to width. elif sum(were_set) == 1 and were_set[LEFT]: width = 1.0 - left right = 0.0 # If only width was set, the padding depends on the alignment. elif sum(were_set) == 1 and were_set[WIDTH]: total = 1.0 - width if align == alignments[LEFT]: left = 0.0 right = total elif align == alignments[RIGHT]: left = total right = 0.0 else: left = total / 2 right = total / 2 # If only right was set, the rest goes to width. elif sum(were_set) == 1 and were_set[RIGHT]: left = 0.0 width = 1.0 - right # If only left wasn't set, width is everything that remains. elif sum(were_set) == 2 and not were_set[LEFT]: left = 1.0 - width - right # If only width wasn't set, width is everything that's not pad. elif sum(were_set) == 2 and not were_set[WIDTH]: width = 1.0 - left - right # If only right wasn't set, right is everything that remains. elif sum(were_set) == 2 and not were_set[RIGHT]: right = 1.0 - left - width # Return the values. return left, width, right, align def _set_relative_horizontal_dimensions( self, left: Optional[float] = None, width: Optional[float] = None, right: Optional[float] = None, align: Optional[str] = None, ) -> None: """Ensure the horizontal relative dimensions are set correctly.""" # Calculate the correct values. left, width, right, align = self._set_relative_dimenstion( left, width, right, align ) # Set the attributes. self.panel_pad_left = left self.panel_relative_width = width self.panel_pad_right = right self.panel_align_h = align def _set_relative_vertical_dimensions( self, top: Optional[float] = None, height: Optional[float] = None, bottom: Optional[float] = None, align: Optional[str] = None ) -> None: # Calculate the correct values. top, height, bottom, align = self._set_relative_dimenstion( top, height, bottom, align, align_default='middle', alignments=('top', 'middle', 'bottom'), attr_names=( 'panel_pad_top', 'panel_relative_height', 'panel_pad_bottom', ) ) # Set the attributes. self.panel_pad_top = top self.panel_relative_height = height self.panel_pad_bottom = bottom self.panel_align_v = align
# Protocols.
[docs] class Frame(Panel): """Create a new :class:`thurible.panel.Frame` object. This class serves as a parent class for all panels that can have a frame surrounding the interior of the panel. As a subclass of :class:`thurible.panel.Panel`, it can also take those parameters and has those public methods. :param frame_type: (Optional.) If a string, the string determines the frame used for the pane. If None, the pane doesn't have a frame. :param frame_fg: (Optional.) A string describing the foreground color of the frame. See the documentation for :mod:`blessed` for more detail on the available options. If `fg` is set and this is not, the frame will have the `fg` color. :param frame_bg: (Optional.) A string describing the background color of the frame. See the documentation for `blessed` for more detail on the available options. If `bg` is set and this is not, the frame will have the `bg` color. :return: A :class:`Frame` object. :rtype: thurible.panel.Frame :usage: To create a new :class:`thurible.panel.Frame` subclass: .. testcode:: from thurible import panel class Spam(panel.Frame): def wobble(self): pass spam = Spam(height=10, width=10) Information on the sizing of :class:`thurible.panel.Frame` objects can be found in the :ref:`sizing` section below. """ def __init__( self, frame_type: Optional[str] = None, frame_bg: str = '', frame_fg: str = '', *args, **kwargs ) -> None: super().__init__(*args, **kwargs) self.frame_type = frame_type self.frame_bg = frame_bg self.frame_fg = frame_fg def __eq__(self, other) -> bool: if not isinstance(other, self.__class__): return NotImplemented return ( super().__eq__(other) and self.frame_type == other.frame_type and self.frame_bg == other.frame_bg and self.frame_fg == other.frame_fg ) def __str__(self) -> str: """Return a string that will draw the entire display.""" result = super().__str__() result += self.frame return result # Properties @property def frame(self) -> str: """A string that will print panel's frame in a terminal. :return: A :class:`str` object. :rtype: str """ # Handle frame coloration. bg = self.frame_bg if self.frame_bg else self.bg fg = self.frame_fg if self.frame_fg else self.fg # Create the frame string and return. result = '' if self.frame_type is not None: result += self._frame( frame_type=self.frame_type, height=self.frame_height, width=self.frame_width, origin_y=self.frame_origin_y, origin_x=self.frame_origin_x, foreground=fg, background=bg ) return result @property def frame_height(self) -> int: """The height in rows of the frame in the terminal. :return: A :class:`str` object. :rtype: str """ return super().inner_height @property def frame_width(self) -> int: """The width in columns of the frame in the terminal. :return: A :class:`str` object. :rtype: str """ return super().inner_width @property def frame_origin_x(self) -> int: """The left-most column of the frame in the terminal. :return: A :class:`str` object. :rtype: str """ return super().inner_x @property def frame_origin_y(self) -> int: """The top-most row of the frame in the terminal. :return: A :class:`str` object. :rtype: str """ return super().inner_y @property def inner_height(self) -> int: height = super().inner_height if self.frame_type: height -= 2 return height @property def inner_width(self) -> int: width = super().inner_width if self.frame_type: width -= 2 return width @property def inner_x(self) -> int: x = super().inner_x if self.frame_type: x += 1 return x @property def inner_y(self) -> int: y = super().inner_y if self.frame_type: y += 1 return y # Private helper methods. def _frame( self, frame_type: str, height: int, width: int, origin_y: int = 0, origin_x: int = 0, foreground: str = '', background: str = '' ) -> str: frame = Box(frame_type) result = self._get_color(foreground, background) result += ( self.term.move(origin_y, origin_x) + frame.ltop + frame.top * (width - 2) + frame.rtop ) for y in range(origin_y + 1, origin_y + height - 1): line = ( self.term.move(y, origin_x) + frame.side + self.term.move(y, origin_x + width - 1) + frame.side ) result += line result += ( self.term.move(origin_y + height - 1, origin_x) + frame.lbot + frame.bot * (width - 2) + frame.rbot ) if background or foreground: result += self.term.normal return result
[docs] class Content(Frame): """Create a new :class:`thurible.panel.Content` object. This class serves as a parent class for all panels that allow padding between the frame surrounding the interior of the panel and the content contained by the panel. The nature of that content is defined by the subclass. As a subclass of :class:`thurible.panel.Frame`, it can also take those parameters and has those public methods. :param content_align_h: (Optional.) The horizontal alignment of the contents of the panel. It defaults to center. :param content_align_v: (Optional.) The vertical alignment of the contents of the penal. It defaults to middle. :param content_pad_left: (Optional.) The amount of padding between the left inner margin of the panel and the content. It is measured as a float between 0.0 and 1.0, where 0.0 is no padding and 1.0 is the entire width of the panel is padding. The default is 0.0. :param content_pad_right: (Optional.) The amount of padding between the right inner margin of the panel and the content. It is measured as a float between 0.0 and 1.0, where 0.0 is no padding and 1.0 is the entire width of the panel is padding. The default is 0.0. :param panel_relative_width: (Optional.) The width of the content of the panel in comparison of the full width of the panel. It is a percentage expressed as a :class:`float` between 0.0 and 1.0, inclusive. The default is 1.0. :return: A :class:`Content` object. :rtype: thurible.panel.Content :usage: To create a new :class:`thurible.panel.Content` subclass: .. testcode:: from thurible import panel class Spam(panel.Content): def wobble(self): pass spam = Spam(height=10, width=10) Information on the sizing of :class:`thurible.panel.Content` objects can be found in the :ref:`sizing` section below. """ # Magic methods. def __init__( self, content_align_h: Optional[str] = None, content_align_v: str = 'middle', content_pad_left: Optional[float] = None, content_pad_right: Optional[float] = None, content_relative_width: Optional[float] = None, *args, **kwargs ) -> None: self.content_align_v = content_align_v self._set_content_relative_horizontal_dimensions( content_pad_left, content_relative_width, content_pad_right, content_align_h ) super().__init__(*args, **kwargs) def __eq__(self, other) -> bool: if not isinstance(other, self.__class__): return NotImplemented return ( super().__eq__(other) and self.content_align_h == other.content_align_h and self.content_align_v == other.content_align_v and self.content_pad_left == other.content_pad_left and self.content_pad_right == other.content_pad_right ) # Properties. @property def content_width(self) -> int: """The width available to content within the panel after padding has been taken into account. :return: A :class:`int` object. :rtype: int """ width = self.inner_width width -= self._offset_left width -= self._offset_right return width @property def content_x(self) -> int: """The left-most column available to content within the panel after padding has been taken into account. :return: A :class:`int` object. :rtype: int """ x = self.inner_x x += self._offset_left return x @property def lines(self) -> list[str]: """The lines available to be displayed in the panel. :return: A :class:`list` object containing each line as a :class:`str`. :rtype: list """ # This is an abstract version of the property. It will need to # be reimplemented for the classes that follow the Content # protocol. return ['',] @property def _offset_left(self) -> int: offset = super().inner_width * self.content_pad_left return int(offset) @property def _offset_right(self) -> int: offset = super().inner_width * self.content_pad_right return int(offset) # Private helper methods. def _align_h(self, align: str, length: int, width: int) -> int: """Return the amount offset the column for the horizontal alignment. """ if align == 'left': x_mod = 0 elif align == 'center': h_space = width - length x_mod = h_space // 2 elif align == 'right': h_space = width - length x_mod = h_space return x_mod def _align_v(self, align: str, length: int, height: int) -> int: """Return the amount offset the column for the vertical alignment. """ if align == 'middle': v_space = height - length y_mod = v_space // 2 elif align == 'top': y_mod = 0 elif align == 'bottom': v_space = height - length y_mod = v_space return y_mod def _set_content_relative_horizontal_dimensions( self, left: Optional[float] = None, width: Optional[float] = None, right: Optional[float] = None, align: Optional[str] = None, ) -> None: """Ensure the horizontal relative dimensions are set correctly.""" # Calculate the correct values. left, width, right, align = self._set_relative_dimenstion( left, width, right, align, attr_names=( 'content_pad_top', 'content_relative_height', 'content_pad_bottom', ) ) # Set the attributes. self.content_pad_left = left self.content_relative_width = width self.content_pad_right = right self.content_align_h = align
[docs] class Scroll(Content): """ Create a new :class:`thurible.panel.Scroll` object. This class serves as a parent class for all panels that allow the user to scroll through content that overflows the interior of the panel. As a subclass of :class:`thurible.panel.Content`, it can also take those parameters and has those public methods. :return: A :class:`Panel` object. :rtype: thurible.panel.Panel :usage: To create a new :class:`thurible.panel.Content` subclass: .. testcode:: from thurible import panel class Spam(panel.Content): def wobble(self): pass spam = Spam(height=10, width=10) Information on the sizing of :class:`thurible.panel.Content` objects can be found in the :ref:`sizing` section below. :active keys: This class defines the following :ref:`active keys<active>`: * KEY_END: Scroll to the end of the content. * KEY_DOWN: Scroll down in the content. * KEY_HOME: Scroll to the top of the content. * KEY_PGDOWN: Scroll one screen down in the content. * KEY_PGUP: Scroll one page up in the content. * KEY_UP: Scroll one line up in the content. """ def __init__( self, *args, **kwargs ) -> None: super().__init__(*args, **kwargs) self._lines: list[str] = [] self._ofbot = '[▼]' self._oftop = '[▲]' self._overflow_bottom = False self._overflow_top = False self._start = 0 self._stop = self.inner_height self._wrapped_width = -1 # Register the keyboard input the class responds to. self.register_key('KEY_END', self._end) self.register_key('KEY_DOWN', self._line_down) self.register_key('KEY_HOME', self._home) self.register_key('KEY_PGDOWN', self._page_down) self.register_key('KEY_PGUP', self._page_up) self.register_key('KEY_UP', self._line_up) def __eq__(self, other) -> bool: if not isinstance(other, self.__class__): return NotImplemented return super().__eq__(other) # Property. @property def inner_height(self) -> int: height = super().inner_height if self._overflow_bottom: height -= 1 if self._overflow_top: height -= 1 return height @property def inner_y(self) -> int: y = super().inner_y if self._overflow_top: y += 1 return y # Public methods.
[docs] def action(self, key: Keystroke) -> tuple[str, str]: # These are the results that are returned. data = '' update = '' # Initial set up. height = self.inner_height width = self.inner_width y = self.inner_y x = self.inner_x lines = self.lines length = len(lines) # Handle input. if length <= height and key.name in self._active_keys: ... elif key.name in self._active_keys: action = self._active_keys[key.name] action() update, height, y = self._flow( update, length, height, width, y, x ) self._overscroll(length, height) update += self.clear_contents() update += self._visible(lines, width, y, x) else: data = str(key) # Return the results. return data, update
# Private action handlers. def _end(self, key: Optional[Keystroke] = None) -> str: """Scroll to the bottom of the content.""" length = len(self.lines) self._start = length - self.inner_height self._stop = length return '' def _home(self, key: Optional[Keystroke] = None) -> str: """Scroll to the top of the content.""" self._start = 0 self._stop = self.inner_height return '' def _line_down(self, key: Optional[Keystroke] = None) -> str: """Scroll a line down in the content.""" self._start += 1 self._stop += 1 return '' def _line_up(self, key: Optional[Keystroke] = None) -> str: """Scroll a line up in the content.""" self._start -= 1 self._stop -= 1 return '' def _page_down(self, key: Optional[Keystroke] = None) -> str: """Scroll down one page in the content.""" height = self.inner_height self._start += height self._stop += height if not self._overflow_top: self._start -= 1 self._stop -= 1 return '' def _page_up(self, key: Optional[Keystroke] = None) -> str: """Scroll up one page in the content.""" height = self.inner_height self._start -= height self._stop -= height if not self._overflow_bottom: self._start += 1 self._stop += 1 return '' # Private helper methods. def _flow( self, update: str, length: int, height: int, width: int, y: int, x: int ) -> tuple[str, int, int]: """Manage text that overflows the visible answer.""" if self._stop >= length - 1 and self._overflow_bottom: self._stop += 1 self._overflow_bottom = False height += 1 end = y + height - 1 update += self._get_color(self.fg, self.bg) update += self.term.move(end, x) + ' ' * width if self.fg or self.bg: update += self.term.normal if self._stop < length and not self._overflow_bottom: self._stop -= 1 self._overflow_bottom = True height -= 1 end = y + height x_mod = self._align_h('center', len(self._ofbot), width) update += self._get_color(self.fg, self.bg) update += self.term.move(end, x) + ' ' * width update += self.term.move(end, x + x_mod) + self._ofbot if self.fg or self.bg: update += self.term.normal if self._start <= 1 and self._overflow_top: self._start -= 1 self._overflow_top = False height += 1 y -= 1 update += self._get_color(self.fg, self.bg) update += self.term.move(y, x) + ' ' * width if self.fg or self.bg: update += self.term.normal if self._start > 0 and not self._overflow_top: self._start += 1 self._overflow_top = True x_mod = self._align_h('center', len(self._oftop), width) update += self._get_color(self.fg, self.bg) update += self.term.move(y, x) + ' ' * width update += self.term.move(y, x + x_mod) + self._oftop if self.fg or self.bg: update += self.term.normal height -= 1 y += 1 return update, height, y def _overscroll(self, length: int, height: int) -> None: """Manage situations where the visible area is scrolled beyond the text. """ if self._start < 0: self._start = 0 self._stop = height elif self._stop > length and length > height: self._stop = length self._start = self._stop - height elif self._stop > length: self._stop = length self._start = 0 def _visible(self, lines: list[str], width: int, y: int, x: int) -> str: """Output the lines in the display.""" update = '' update += self._get_color(self.fg, self.bg) for i, line in enumerate(lines[self._start: self._stop]): x_mod = self._align_h(self.content_align_h, len(line), width) update += self.term.move(y + i, x + x_mod) + line if self.fg or self.bg: update += self.term.normal return update
[docs] class Title(Frame): """Create a new :class:`thurible.panel.Title` object. This class serves as a parent class for all panels that all the user to put a title on the top of the panel and a footer on the bottom of the frame. As a subclass of :class:`thurible.panel.Frame`, it can alse take those parameters and has those public methods and properties. :param footer_align: (Optional.) The horizontal alignment of the footer. The available options are "left", "center", and "right". :param footer_frame: (Optional.) Whether the frame should be capped on either side of the footer. :param footer_text: (Optional.) The text contained within the footer. :param title_align: (Optional.) The horizontal alignment of the title. The available options are "left", "center", and "right". :param title_bg: (Optional.) The background color of the title and footer. See the documentation for :mod:`blessed` for more detail on the available options. :param title_fg: (Optional.) The foreground color of the title and footer. See the documentation for :mod:`blessed` for more detail on the available options. :param title_frame: (Optional.) Whether the frame should be capped on either side of the title. :param title_text: (Optional.) The text contained within the title. :return: A :class:`Title` object. :rtype: thurible.panel.Title :usage: To create a new :class:`thurible.panel.Title` subclass: .. testcode:: from thurible import panel class Spam(panel.Title): def wobble(self): pass spam = Spam(height=10, width=10) Information on the sizing of :class:`thurible.panel.Title` objects can be found in the :ref:`sizing` section below. """ # Magic methods. def __init__( self, footer_align: str = 'left', footer_frame: bool = False, footer_text: str = '', title_align: str = 'left', title_bg: str = '', title_fg: str = '', title_frame: bool = False, title_text: str = '', *args, **kwargs ) -> None: super().__init__(*args, **kwargs) self.footer_text = footer_text self.footer_align = footer_align self.footer_frame = footer_frame self.title_text = title_text self.title_align = title_align self.title_frame = title_frame self.title_bg = title_bg self.title_fg = title_fg self._ofr = '[▸]' def __eq__(self, other) -> bool: if not isinstance(other, self.__class__): return NotImplemented return ( super().__eq__(other) and self.footer_text == other.footer_text and self.footer_align == other.footer_align and self.footer_frame == other.footer_frame and self.title_text == other.title_text and self.title_align == other.title_align and self.title_frame == other.title_frame and self.title_bg == other.title_bg and self.title_fg == other.title_fg ) def __repr__(self) -> str: name = self.__class__.__name__ return f"{name}(title_text='{self.title_text}')" def __str__(self) -> str: """Return a string that will draw the entire display.""" result = super().__str__() result += self.title result += self.footer return result # Properties. @property def footer(self) -> str: """The footer as a string that could be used to update the terminal. :return: A :class:`str` object. :rtype: str """ y = self.frame_origin_y + self.frame_height - 1 text = self.footer_text frame = self.footer_frame return self._title(text, self.footer_align, y, frame) @property def footer_frame(self) -> bool: return self._Title__footer_frame @footer_frame.setter def footer_frame(self, value: bool) -> None: if value and not self.frame_type: msg = 'You must set frame_type if you set footer_frame.' raise NoFrameTypeForFrameError(msg) self._Title__footer_frame = value @property def frame(self) -> str: result = super().frame if not result: height = self.frame_height width = self.frame_width y = self.frame_origin_y x = self.frame_origin_x bg = self.title_bg if self.title_bg else self.bg fg = self.title_fg if self.title_fg else self.fg if not result and self.title_text: result += self._get_color(fg, bg) result += self.term.move(y, x) + ' ' * width if bg or fg: result += self.term.normal if not result and self.footer_text: result += self._get_color(fg, bg) result += self.term.move(y + height - 1, x) + ' ' * width if bg or fg: result += self.term.normal return result @property def inner_height(self) -> int: height = super().inner_height if self.frame_type is None and self.title_text: height -= 1 if self.frame_type is None and self.footer_text: height -= 1 return height @property def inner_y(self) -> int: y = super().inner_y if self.frame_type is None and self.title: y += 1 return y @property def title(self) -> str: """The title as a string that could be used to update the terminal. :return: A :class:`str` object. :rtype: str """ text = self.title_text frame = self.title_frame return self._title(text, self.title_align, self.frame_origin_y, frame) @property def title_frame(self) -> bool: return self._Title__title_frame @title_frame.setter def title_frame(self, value: bool) -> None: if value and not self.frame_type: msg = 'You must set frame_type if you set title_frame.' raise NoFrameTypeForFrameError(msg) self._Title__title_frame = value # Private helper methods. def _title( self, title: str, align: str, y: int, frame: bool = False ) -> str: """Build and return a string to write the title of the panel to the terminal. """ result = '' # Bail out early if there is no title. if not title: return result # Set up. x = self.inner_x width = self.inner_width # Set the color and frame for the title. if len(title) >= width: title = title[:width - 3] + self._ofr title = self._title_color(title) if isinstance(self.frame_type, str) and frame: title = self._title_frame(title, self.frame_type) # Align the title. if align == 'left': ... elif align == 'center': space = width - len(title) x += space // 2 elif align == 'right': x += width - len(title) else: msg = f'Invalid title alignment: {self.title_align}.' raise InvalidTitleAlignmentError(msg) # Create the title and return. result += self.term.move(y, x) + title return result def _title_color(self, title: str) -> str: """Apply the title color to a title or footer.""" bg = self.title_bg if self.title_bg else self.bg fg = self.title_fg if self.title_fg else self.fg result = self._get_color(fg, bg) result += title if bg or fg: result += self.term.normal return result def _title_frame(self, title: str, frame_type: str) -> str: """Apply the frame cap to the title or footer.""" frame = Box(frame_type) bg = self.frame_bg if self.frame_bg else self.bg fg = self.frame_fg if self.frame_fg else self.fg color = self._get_color(fg, bg) normal = self.term.normal if color else '' lside = color + frame.rside + normal rside = color + frame.lside + normal return lside + title + rside