coloransi.py
246 lines
| 8.6 KiB
| text/x-python
|
PythonLexer
fperez
|
r0 | """Tools for coloring text in ANSI terminals. | ||
Fernando Perez
|
r1853 | """ | ||
fperez
|
r0 | |||
#***************************************************************************** | ||||
fperez
|
r88 | # Copyright (C) 2002-2006 Fernando Perez. <fperez@colorado.edu> | ||
fperez
|
r0 | # | ||
# Distributed under the terms of the BSD License. The full license is in | ||||
# the file COPYING, distributed as part of this software. | ||||
#***************************************************************************** | ||||
import os | ||||
Matthias Bussonnier
|
r28595 | import warnings | ||
fperez
|
r0 | |||
Brian Granger
|
r2030 | from IPython.utils.ipstruct import Struct | ||
fperez
|
r0 | |||
Matthias Bussonnier
|
r28587 | __all__ = ["TermColors", "InputTermColors", "ColorScheme", "ColorSchemeTable"] | ||
Matthias Bussonnier
|
r28596 | _sentinel = object() | ||
Thomas Kluyver
|
r5495 | color_templates = ( | ||
Fernando Perez
|
r1593 | # Dark colors | ||
fperez
|
r0 | ("Black" , "0;30"), | ||
("Red" , "0;31"), | ||||
("Green" , "0;32"), | ||||
("Brown" , "0;33"), | ||||
("Blue" , "0;34"), | ||||
("Purple" , "0;35"), | ||||
("Cyan" , "0;36"), | ||||
("LightGray" , "0;37"), | ||||
Fernando Perez
|
r1593 | # Light colors | ||
fperez
|
r0 | ("DarkGray" , "1;30"), | ||
("LightRed" , "1;31"), | ||||
("LightGreen" , "1;32"), | ||||
("Yellow" , "1;33"), | ||||
("LightBlue" , "1;34"), | ||||
("LightPurple" , "1;35"), | ||||
("LightCyan" , "1;36"), | ||||
Bernardo B. Marques
|
r4872 | ("White" , "1;37"), | ||
Fernando Perez
|
r1593 | # Blinking colors. Probably should not be used in anything serious. | ||
("BlinkBlack" , "5;30"), | ||||
("BlinkRed" , "5;31"), | ||||
("BlinkGreen" , "5;32"), | ||||
("BlinkYellow" , "5;33"), | ||||
("BlinkBlue" , "5;34"), | ||||
("BlinkPurple" , "5;35"), | ||||
("BlinkCyan" , "5;36"), | ||||
("BlinkLightGray", "5;37"), | ||||
) | ||||
Fernando Perez
|
r1589 | |||
Thomas Kluyver
|
r5495 | def make_color_table(in_class): | ||
"""Build a set of color attributes in a class. | ||||
Thomas Kluyver
|
r13592 | Helper function for building the :class:`TermColors` and | ||
:class`InputTermColors`. | ||||
""" | ||||
Fernando Perez
|
r1594 | for name,value in color_templates: | ||
setattr(in_class,name,in_class._base % value) | ||||
Fernando Perez
|
r1589 | |||
fperez
|
r0 | class TermColors: | ||
"""Color escape sequences. | ||||
Bernardo B. Marques
|
r4872 | This class defines the escape sequences for all the standard (ANSI?) | ||
fperez
|
r0 | colors in terminals. Also defines a NoColor escape which is just the null | ||
string, suitable for defining 'dummy' color schemes in terminals which get | ||||
confused by color escapes. | ||||
This class should be used as a mixin for building color schemes.""" | ||||
Bernardo B. Marques
|
r4872 | |||
fperez
|
r0 | NoColor = '' # for color schemes in color-less terminals. | ||
Normal = '\033[0m' # Reset normal coloring | ||||
Matthias Bussonnier
|
r28723 | |||
Black = "\033[0;30m" | ||||
Red = "\033[0;31m" | ||||
Green = "\033[0;32m" | ||||
Brown = "\033[0;33m" | ||||
Blue = "\033[0;34m" | ||||
Purple = "\033[0;35m" | ||||
Cyan = "\033[0;36m" | ||||
LightGray = "\033[0;37m" | ||||
Matthias Bussonnier
|
r28711 | # Light colors | ||
Matthias Bussonnier
|
r28723 | DarkGray = "\033[1;30m" | ||
LightRed = "\033[1;31m" | ||||
LightGreen = "\033[1;32m" | ||||
Yellow = "\033[1;33m" | ||||
LightBlue = "\033[1;34m" | ||||
LightPurple = "\033[1;35m" | ||||
LightCyan = "\033[1;36m" | ||||
White = "\033[1;37m" | ||||
Matthias Bussonnier
|
r28711 | # Blinking colors. Probably should not be used in anything serious. | ||
Matthias Bussonnier
|
r28723 | BlinkBlack = "\033[5;30m" | ||
BlinkRed = "\033[5;31m" | ||||
BlinkGreen = "\033[5;32m" | ||||
BlinkYellow = "\033[5;33m" | ||||
BlinkBlue = "\033[5;34m" | ||||
BlinkPurple = "\033[5;35m" | ||||
BlinkCyan = "\033[5;36m" | ||||
BlinkLightGray = "\033[5;37m" | ||||
Matthias Bussonnier
|
r28711 | |||
fperez
|
r0 | |||
class InputTermColors: | ||||
"""Color escape sequences for input prompts. | ||||
Maxim Cournoyer
|
r27688 | This class is similar to TermColors, but the escapes are wrapped in \\001 | ||
and \\002 so that readline can properly know the length of each line and | ||||
fperez
|
r0 | can wrap lines accordingly. Use this class for any colored text which | ||
needs to be used in input prompts, such as in calls to raw_input(). | ||||
Bernardo B. Marques
|
r4872 | This class defines the escape sequences for all the standard (ANSI?) | ||
fperez
|
r0 | colors in terminals. Also defines a NoColor escape which is just the null | ||
string, suitable for defining 'dummy' color schemes in terminals which get | ||||
confused by color escapes. | ||||
This class should be used as a mixin for building color schemes.""" | ||||
Bernardo B. Marques
|
r4872 | |||
fperez
|
r0 | NoColor = '' # for color schemes in color-less terminals. | ||
fperez
|
r550 | |||
fperez
|
r564 | if os.name == 'nt' and os.environ.get('TERM','dumb') == 'emacs': | ||
fperez
|
r550 | # (X)emacs on W32 gets confused with \001 and \002 so we remove them | ||
Normal = '\033[0m' # Reset normal coloring | ||||
_base = '\033[%sm' # Template for all other colors | ||||
else: | ||||
Normal = '\001\033[0m\002' # Reset normal coloring | ||||
_base = '\001\033[%sm\002' # Template for all other colors | ||||
fperez
|
r0 | |||
# Build the actual color table as a set of class attributes: | ||||
make_color_table(InputTermColors) | ||||
Thomas Kluyver
|
r5495 | class NoColors: | ||
"""This defines all the same names as the colour classes, but maps them to | ||||
empty strings, so it can easily be substituted to turn off colours.""" | ||||
NoColor = '' | ||||
MinRK
|
r7580 | Normal = '' | ||
Thomas Kluyver
|
r5495 | |||
for name, value in color_templates: | ||||
setattr(NoColors, name, '') | ||||
fperez
|
r0 | class ColorScheme: | ||
"""Generic color scheme class. Just a name and a Struct.""" | ||||
Matthias Bussonnier
|
r28587 | |||
name: str | ||||
colors: Struct | ||||
Matthias Bussonnier
|
r28595 | def __init__(self, __scheme_name_, colordict=None, **colormap): | ||
fperez
|
r0 | self.name = __scheme_name_ | ||
Matthias Bussonnier
|
r28595 | if colormap: | ||
warnings.warn( | ||||
"Passing each colors as a kwarg to ColorScheme is " | ||||
"considered for deprecation. Please pass a " | ||||
M Bussonnier
|
r28605 | "a single dict as second parameter. If you are using this" | ||
"feature, please comment an subscribe to issue " | ||||
Matthias Bussonnier
|
r28595 | "https://github.com/ipython/ipython/issues/14304", | ||
PendingDeprecationWarning, | ||||
stacklevel=2, | ||||
) | ||||
fperez
|
r0 | if colordict is None: | ||
self.colors = Struct(**colormap) | ||||
else: | ||||
self.colors = Struct(colordict) | ||||
fperez
|
r46 | |||
def copy(self,name=None): | ||||
"""Return a full copy of the object, optionally renaming it.""" | ||||
if name is None: | ||||
name = self.name | ||||
Brian Granger
|
r2077 | return ColorScheme(name, self.colors.dict()) | ||
Bernardo B. Marques
|
r4872 | |||
fperez
|
r46 | class ColorSchemeTable(dict): | ||
fperez
|
r0 | """General class to handle tables of color schemes. | ||
It's basically a dict of color schemes with a couple of shorthand | ||||
attributes and some convenient methods. | ||||
Bernardo B. Marques
|
r4872 | |||
fperez
|
r0 | active_scheme_name -> obvious | ||
active_colors -> actual color table of the active scheme""" | ||||
Matthias Bussonnier
|
r21778 | def __init__(self, scheme_list=None, default_scheme=''): | ||
fperez
|
r0 | """Create a table of color schemes. | ||
The table can be created empty and manually filled or it can be | ||||
created with a list of valid color schemes AND the specification for | ||||
the default active scheme. | ||||
""" | ||||
Bernardo B. Marques
|
r4872 | |||
fperez
|
r46 | # create object attributes to be set later | ||
self.active_scheme_name = '' | ||||
self.active_colors = None | ||||
Bernardo B. Marques
|
r4872 | |||
fperez
|
r46 | if scheme_list: | ||
fperez
|
r0 | if default_scheme == '': | ||
Bradley M. Froehle
|
r7843 | raise ValueError('you must specify the default color scheme') | ||
fperez
|
r0 | for scheme in scheme_list: | ||
self.add_scheme(scheme) | ||||
self.set_active_scheme(default_scheme) | ||||
fperez
|
r46 | def copy(self): | ||
"""Return full copy of object""" | ||||
return ColorSchemeTable(self.values(),self.active_scheme_name) | ||||
Matthias Bussonnier
|
r28596 | def __setitem__(self, key: str, value: ColorScheme): | ||
assert isinstance(key, str) | ||||
assert isinstance(value, ColorScheme) | ||||
super().__setitem__(key, value) | ||||
def add_scheme(self, new_scheme): | ||||
fperez
|
r0 | """Add a new color scheme to the table.""" | ||
Matthias Bussonnier
|
r28596 | if not isinstance(new_scheme, ColorScheme): | ||
Bradley M. Froehle
|
r7843 | raise ValueError('ColorSchemeTable only accepts ColorScheme instances') | ||
fperez
|
r0 | self[new_scheme.name] = new_scheme | ||
Bernardo B. Marques
|
r4872 | |||
Matthias Bussonnier
|
r28596 | def set_active_scheme(self, scheme, case_sensitive=_sentinel): | ||
fperez
|
r0 | """Set the currently active scheme. | ||
Names are by default compared in a case-insensitive way, but this can | ||||
be changed by setting the parameter case_sensitive to true.""" | ||||
Matthias Bussonnier
|
r28596 | if case_sensitive is _sentinel: | ||
case_sensitive = False | ||||
else: | ||||
warnings.warn( | ||||
"set_active_scheme(case_sensitive=...) is Pending " | ||||
"deprecation. Please comment on " | ||||
"https://github.com/ipython/ipython/issues/14306 " | ||||
"to let the ipython maintainer that you are affected.", | ||||
PendingDeprecationWarning, | ||||
stacklevel=2, | ||||
) | ||||
Thomas Kluyver
|
r13358 | scheme_names = list(self.keys()) | ||
fperez
|
r0 | if case_sensitive: | ||
fperez
|
r46 | valid_schemes = scheme_names | ||
fperez
|
r0 | scheme_test = scheme | ||
else: | ||||
fperez
|
r46 | valid_schemes = [s.lower() for s in scheme_names] | ||
fperez
|
r0 | scheme_test = scheme.lower() | ||
try: | ||||
scheme_idx = valid_schemes.index(scheme_test) | ||||
Ram Rachum
|
r25833 | except ValueError as e: | ||
Bradley M. Froehle
|
r7843 | raise ValueError('Unrecognized color scheme: ' + scheme + \ | ||
Ram Rachum
|
r25833 | '\nValid schemes: '+str(scheme_names).replace("'', ",'')) from e | ||
fperez
|
r0 | else: | ||
fperez
|
r46 | active = scheme_names[scheme_idx] | ||
fperez
|
r0 | self.active_scheme_name = active | ||
self.active_colors = self[active].colors | ||||
# Now allow using '' as an index for the current active scheme | ||||
self[''] = self[active] | ||||