Spaces:
Sleeping
Sleeping
File size: 11,224 Bytes
2d876d1 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 |
"""Module contains shared utility functions and typing aliases."""
import math
import os
import shutil
from typing import (
TYPE_CHECKING,
Any,
Callable,
Dict,
List,
NamedTuple,
Optional,
Tuple,
Union,
)
from prompt_toolkit import print_formatted_text
from prompt_toolkit.application import run_in_terminal
from prompt_toolkit.application.current import get_app
from prompt_toolkit.formatted_text import FormattedText
from prompt_toolkit.styles import Style
from prompt_toolkit.validation import Validator
from InquirerPy.exceptions import InvalidArgument
if TYPE_CHECKING:
from prompt_toolkit.filters.base import FilterOrBool
from InquirerPy.base.control import Choice
__all__ = [
"get_style",
"calculate_height",
"InquirerPyStyle",
"patched_print",
"color_print",
]
class InquirerPyStyle(NamedTuple):
"""`InquirerPy` Style class.
Used as a helper class to enforce the method `get_style` to be used
while also avoiding :class:`dict` to be passed into prompts.
Note:
The class is an instance of :class:`typing.NamedTuple`.
Warning:
You should not directly be using this class besides for type hinting
purposes. Obtain an instance of this class using :func:`.get_style`.
"""
dict: Dict[str, str]
InquirerPySessionResult = Dict[Union[str, int], Optional[Union[str, bool, List[Any]]]]
InquirerPyChoice = Union[List[Any], List["Choice"], List[Dict[str, Any]]]
InquirerPyListChoices = Union[
Callable[["InquirerPySessionResult"], InquirerPyChoice],
InquirerPyChoice,
]
InquirerPyValidate = Union[Callable[[Any], bool], "Validator"]
InquirerPyQuestions = Union[List[Dict[str, Any]], Dict[str, Any]]
InquirerPyMessage = Union[str, Callable[["InquirerPySessionResult"], str]]
InquirerPyDefault = Union[Any, Callable[["InquirerPySessionResult"], Any]]
InquirerPyKeybindings = Dict[
str, List[Dict[str, Union[str, "FilterOrBool", List[str]]]]
]
def get_style(
style: Optional[Dict[str, str]] = None, style_override: bool = True
) -> InquirerPyStyle:
"""Obtain an :class:`.InquirerPyStyle` instance which can be consumed by the `style` parameter in prompts.
Tip:
This function supports ENV variables.
For all the color ENV variable names, refer to the :ref:`ENV <pages/env:Style>` documentation.
Note:
If no style is provided, then a default theme based on `one dark <https://github.com/joshdick/onedark.vim#color-reference>`_
color palette is applied.
Note:
Priority: style parameter -> ENV variable -> default style
Args:
style: The dictionary of style classes and their colors, If nothing is passed, the style will be resolved to the :ref:`pages/style:Default Style`.
style_override: A boolean to determine if the supplied `style` parameter should be merged with the :ref:`pages/style:Default Style` or override them.
By default, the supplied style will overwrite the :ref:`pages/style:Default Style`.
Returns:
An instance of :class:`.InquirerPyStyle`.
Examples:
>>> from InquirerPy import get_style
>>> from InquirerPy import inquirer
>>> style = get_style({"questionmark": "#ffffff", "answer": "#000000"}, style_override=False)
>>> result = inquirer.confirm(message="Confirm?", style=style).execute()
"""
if not style_override or style is None:
if not style:
style = {}
result = {
"questionmark": os.getenv("INQUIRERPY_STYLE_QUESTIONMARK", "#e5c07b"),
"answermark": os.getenv("INQUIRERPY_STYLE_ANSWERMARK", "#e5c07b"),
"answer": os.getenv("INQUIRERPY_STYLE_ANSWER", "#61afef"),
"input": os.getenv("INQUIRERPY_STYLE_INPUT", "#98c379"),
"question": os.getenv("INQUIRERPY_STYLE_QUESTION", ""),
"answered_question": os.getenv("INQUIRERPY_STYLE_ANSWERED_QUESTION", ""),
"instruction": os.getenv("INQUIRERPY_STYLE_INSTRUCTION", "#abb2bf"),
"long_instruction": os.getenv(
"INQUIRERPY_STYLE_LONG_INSTRUCTION", "#abb2bf"
),
"pointer": os.getenv("INQUIRERPY_STYLE_POINTER", "#61afef"),
"checkbox": os.getenv("INQUIRERPY_STYLE_CHECKBOX", "#98c379"),
"separator": os.getenv("INQUIRERPY_STYLE_SEPARATOR", ""),
"skipped": os.getenv("INQUIRERPY_STYLE_SKIPPED", "#5c6370"),
"validator": os.getenv("INQUIRERPY_STYLE_VALIDATOR", ""),
"marker": os.getenv("INQUIRERPY_STYLE_MARKER", "#e5c07b"),
"fuzzy_prompt": os.getenv("INQUIRERPY_STYLE_FUZZY_PROMPT", "#c678dd"),
"fuzzy_info": os.getenv("INQUIRERPY_STYLE_FUZZY_INFO", "#abb2bf"),
"fuzzy_border": os.getenv("INQUIRERPY_STYLE_FUZZY_BORDER", "#4b5263"),
"fuzzy_match": os.getenv("INQUIRERPY_STYLE_FUZZY_MATCH", "#c678dd"),
"spinner_pattern": os.getenv("INQUIRERPY_STYLE_SPINNER_PATTERN", "#e5c07b"),
"spinner_text": os.getenv("INQUIRERPY_STYLE_SPINNER_TEXT", ""),
**style,
}
else:
result = {
"questionmark": os.getenv("INQUIRERPY_STYLE_QUESTIONMARK", ""),
"answermark": os.getenv("INQUIRERPY_STYLE_ANSWERMARK", ""),
"answer": os.getenv("INQUIRERPY_STYLE_ANSWER", ""),
"input": os.getenv("INQUIRERPY_STYLE_INPUT", ""),
"question": os.getenv("INQUIRERPY_STYLE_QUESTION", ""),
"answered_question": os.getenv("INQUIRERPY_STYLE_ANSWERED_QUESTION", ""),
"instruction": os.getenv("INQUIRERPY_STYLE_INSTRUCTION", ""),
"long_instruction": os.getenv("INQUIRERPY_STYLE_LONG_INSTRUCTION", ""),
"pointer": os.getenv("INQUIRERPY_STYLE_POINTER", ""),
"checkbox": os.getenv("INQUIRERPY_STYLE_CHECKBOX", ""),
"separator": os.getenv("INQUIRERPY_STYLE_SEPARATOR", ""),
"skipped": os.getenv("INQUIRERPY_STYLE_SKIPPED", ""),
"validator": os.getenv("INQUIRERPY_STYLE_VALIDATOR", ""),
"marker": os.getenv("INQUIRERPY_STYLE_MARKER", ""),
"fuzzy_prompt": os.getenv("INQUIRERPY_STYLE_FUZZY_PROMPT", ""),
"fuzzy_info": os.getenv("INQUIRERPY_STYLE_FUZZY_INFO", ""),
"fuzzy_border": os.getenv("INQUIRERPY_STYLE_FUZZY_BORDER", ""),
"fuzzy_match": os.getenv("INQUIRERPY_STYLE_FUZZY_MATCH", ""),
"spinner_pattern": os.getenv("INQUIRERPY_STYLE_SPINNER_PATTERN", ""),
"spinner_text": os.getenv("INQUIRERPY_STYLE_SPINNER_TEXT", ""),
**style,
}
if result.get("fuzzy_border"):
result["frame.border"] = result.pop("fuzzy_border")
if result.get("validator"):
result["validation-toolbar"] = result.pop("validator")
result["bottom-toolbar"] = "noreverse"
return InquirerPyStyle(result)
def calculate_height(
height: Optional[Union[int, str]],
max_height: Optional[Union[int, str]],
height_offset: int = 2,
) -> Tuple[Optional[int], int]:
"""Calculate the `height` and `max_height` for the main question contents.
Tip:
The parameter `height`/`max_height` can be specified by either a :class:`string` or :class:`int`.
When `height`/`max_height` is :class:`str`:
It will set the height to a percentage based on the value provided.
You can optionally add the '%' sign which will be ignored while processing.
Example: "60%" or "60" (60% of the current terminal visible lines)
When `height`/`max_height` is :class:`int`:
It will set the height to exact number of lines based on the value provided.
Example: 20 (20 lines in terminal)
Note:
If `max_height` is not provided or is None, the default `max_height` will be configured to `70%` for
best visual presentation in the terminal.
Args:
height: The desired height in either percentage as string or exact value as int.
max_height: Maximum acceptable height in either percentage as string or exact value as int.
height_offset: Height offset to apply to the height.
Returns:
A :class:`tuple` with the first value being the desired height and the second value being
the maximum height.
Raises:
InvalidArgument: The provided `height`/`max_height` is not able to to be converted to int.
Examples:
>>> calculate_height(height="60%", max_height="100%")
"""
try:
_, term_lines = shutil.get_terminal_size()
term_lines = term_lines
if not height:
dimmension_height = None
else:
if isinstance(height, str):
height = height.replace("%", "")
height = int(height)
dimmension_height = (
math.floor(term_lines * (height / 100)) - height_offset
)
else:
dimmension_height = height
if not max_height:
max_height = "70%" if not height else "100%"
if isinstance(max_height, str):
max_height = max_height.replace("%", "")
max_height = int(max_height)
dimmension_max_height = (
math.floor(term_lines * (max_height / 100)) - height_offset
)
else:
dimmension_max_height = max_height
if dimmension_height and dimmension_height > dimmension_max_height:
dimmension_height = dimmension_max_height
if dimmension_height and dimmension_height <= 0:
dimmension_height = 1
if dimmension_max_height <= 0:
dimmension_max_height = 1
return dimmension_height, dimmension_max_height
except ValueError:
raise InvalidArgument(
"prompt argument height/max_height needs to be type of an int or str"
)
def patched_print(*values) -> None:
"""Patched :func:`print` that can print values without interrupting the prompt.
See Also:
:func:`print`
:func:`~prompt_toolkit.application.run_in_terminal`
Args:
*values: Refer to :func:`print`.
Examples:
>>> patched_print("Hello World")
"""
def _print():
print(*values)
run_in_terminal(_print)
def color_print(
formatted_text: List[Tuple[str, str]], style: Optional[Dict[str, str]] = None
) -> None:
"""Print colored text leveraging :func:`~prompt_toolkit.shortcuts.print_formatted_text`.
This function automatically handles printing the text without interrupting the
current prompt.
Args:
formatted_text: A list of formatted_text.
style: Style to apply to `formatted_text` in :class:`dictionary` form.
Example:
>>> color_print(formatted_text=[("class:aa", "hello "), ("class:bb", "world")], style={"aa": "red", "bb": "blue"})
>>> color_print([("red", "yes"), ("", " "), ("blue", "no")])
"""
def _print():
print_formatted_text(
FormattedText(formatted_text),
style=Style.from_dict(style) if style else None,
)
if get_app().is_running:
run_in_terminal(_print)
else:
_print()
|