Coverage for /pythoncovmergedfiles/medio/medio/usr/local/lib/python3.8/site-packages/rich/jupyter.py: 25%
63 statements
« prev ^ index » next coverage.py v7.0.1, created at 2022-12-25 06:11 +0000
« prev ^ index » next coverage.py v7.0.1, created at 2022-12-25 06:11 +0000
1from typing import TYPE_CHECKING, Any, Dict, Iterable, List, Sequence
3if TYPE_CHECKING:
4 from rich.console import ConsoleRenderable
6from . import get_console
7from .segment import Segment
8from .terminal_theme import DEFAULT_TERMINAL_THEME
10if TYPE_CHECKING:
11 from rich.console import ConsoleRenderable
13JUPYTER_HTML_FORMAT = """\
14<pre style="white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace">{code}</pre>
15"""
18class JupyterRenderable:
19 """A shim to write html to Jupyter notebook."""
21 def __init__(self, html: str, text: str) -> None:
22 self.html = html
23 self.text = text
25 def _repr_mimebundle_(
26 self, include: Sequence[str], exclude: Sequence[str], **kwargs: Any
27 ) -> Dict[str, str]:
28 data = {"text/plain": self.text, "text/html": self.html}
29 if include:
30 data = {k: v for (k, v) in data.items() if k in include}
31 if exclude:
32 data = {k: v for (k, v) in data.items() if k not in exclude}
33 return data
36class JupyterMixin:
37 """Add to an Rich renderable to make it render in Jupyter notebook."""
39 __slots__ = ()
41 def _repr_mimebundle_(
42 self: "ConsoleRenderable",
43 include: Sequence[str],
44 exclude: Sequence[str],
45 **kwargs: Any,
46 ) -> Dict[str, str]:
47 console = get_console()
48 segments = list(console.render(self, console.options))
49 html = _render_segments(segments)
50 text = console._render_buffer(segments)
51 data = {"text/plain": text, "text/html": html}
52 if include:
53 data = {k: v for (k, v) in data.items() if k in include}
54 if exclude:
55 data = {k: v for (k, v) in data.items() if k not in exclude}
56 return data
59def _render_segments(segments: Iterable[Segment]) -> str:
60 def escape(text: str) -> str:
61 """Escape html."""
62 return text.replace("&", "&").replace("<", "<").replace(">", ">")
64 fragments: List[str] = []
65 append_fragment = fragments.append
66 theme = DEFAULT_TERMINAL_THEME
67 for text, style, control in Segment.simplify(segments):
68 if control:
69 continue
70 text = escape(text)
71 if style:
72 rule = style.get_html_style(theme)
73 text = f'<span style="{rule}">{text}</span>' if rule else text
74 if style.link:
75 text = f'<a href="{style.link}" target="_blank">{text}</a>'
76 append_fragment(text)
78 code = "".join(fragments)
79 html = JUPYTER_HTML_FORMAT.format(code=code)
81 return html
84def display(segments: Iterable[Segment], text: str) -> None:
85 """Render segments to Jupyter."""
86 html = _render_segments(segments)
87 jupyter_renderable = JupyterRenderable(html, text)
88 try:
89 from IPython.display import display as ipython_display
91 ipython_display(jupyter_renderable)
92 except ModuleNotFoundError:
93 # Handle the case where the Console has force_jupyter=True,
94 # but IPython is not installed.
95 pass
98def print(*args: Any, **kwargs: Any) -> None:
99 """Proxy for Console print."""
100 console = get_console()
101 return console.print(*args, **kwargs)