Source code for lookatme.render.pygments

"""Pygments related rendering
"""


import urwid
import pygments
import pygments.util
from pygments.formatter import Formatter
import time
import urwid


import lookatme.config as config


LEXER_CACHE = {}
STYLE_CACHE = {}
FORMATTER_CACHE = {}


[docs]def get_formatter(style_name): style = get_style(style_name) formatter, style_bg = FORMATTER_CACHE.get(style_name, (None, None)) if formatter is None: style_bg = UrwidFormatter.findclosest(style.background_color.replace("#", "")) formatter = UrwidFormatter( style=style, usebg=(style_bg is not None), ) FORMATTER_CACHE[style_name] = (formatter, style_bg) return formatter, style_bg
[docs]def get_lexer(lang, default="text"): lexer = LEXER_CACHE.get(lang, None) if lexer is None: try: lexer = pygments.lexers.get_lexer_by_name(lang) except pygments.util.ClassNotFound: lexer = pygments.lexers.get_lexer_by_name(default) LEXER_CACHE[lang] = lexer return lexer
[docs]def get_style(style_name): style = STYLE_CACHE.get(style_name, None) if style is None: style = pygments.styles.get_style_by_name(style_name) STYLE_CACHE[style_name] = style return style
[docs]def render_text(text, lang="text", style_name=None, plain=False): """Render the provided text with the pygments renderer """ if style_name is None: style_name = config.STYLE["style"] lexer = get_lexer(lang) formatter, style_bg = get_formatter(style_name) start = time.time() code_tokens = lexer.get_tokens(text) config.LOG.debug(f"Took {time.time()-start}s to render {len(text)} bytes") markup = [] for x in formatter.formatgenerator(code_tokens): if style_bg: x[0].background = style_bg markup.append(x) if markup[-1][1] == "\n": markup = markup[:-1] if len(markup) == 0: markup = [(None, "")] elif markup[-1][1].endswith("\n"): markup[-1] = (markup[-1][0], markup[-1][1][:-1]) if plain: return markup else: return urwid.AttrMap(urwid.Text(markup), urwid.AttrSpec("default", style_bg))
[docs]class UrwidFormatter(Formatter): """Formatter that returns [(text,attrspec), ...], where text is a piece of text, and attrspec is an urwid.AttrSpec""" def __init__(self, **options): """Extra arguments: usebold: if false, bold will be ignored and always off default: True usebg: if false, background color will always be 'default' default: True colors: number of colors to use (16, 88, or 256) default: 256""" self.usebold = options.get('usebold',True) self.usebg = options.get('usebg', True) colors = options.get('colors', 256) self.style_attrs = {} Formatter.__init__(self, **options) @property def style(self): return self._style @style.setter def style(self, newstyle): self._style = newstyle self._setup_styles() @staticmethod def _distance(col1, col2): r1, g1, b1 = col1 r2, g2, b2 = col2 rd = r1 - r2 gd = g1 - g2 bd = b1 - b2 return rd*rd + gd*gd + bd*bd
[docs] @classmethod def findclosest(cls, colstr, colors=256): """Takes a hex string and finds the nearest color to it. Returns a string urwid will recognize.""" rgb = int(colstr, 16) r = (rgb >> 16) & 0xff g = (rgb >> 8) & 0xff b = rgb & 0xff dist = 257 * 257 * 3 bestcol = urwid.AttrSpec('h0','default') for i in range(colors): curcol = urwid.AttrSpec('h%d' % i,'default', colors=colors) currgb = curcol.get_rgb_values()[:3] curdist = cls._distance((r,g,b), currgb) if curdist < dist: dist = curdist bestcol = curcol return bestcol.foreground
[docs] def findclosestattr(self, fgcolstr=None, bgcolstr=None, othersettings='', colors = 256): """Takes two hex colstring (e.g. 'ff00dd') and returns the nearest urwid style.""" fg = bg = 'default' if fgcolstr: fg = self.findclosest(fgcolstr, colors) if bgcolstr: bg = self.findclosest(bgcolstr, colors) if othersettings: fg = fg + ',' + othersettings return urwid.AttrSpec(fg, bg, colors)
def _setup_styles(self, colors = 256): """Fills self.style_attrs with urwid.AttrSpec attributes corresponding to the closest equivalents to the given style.""" for ttype, ndef in self.style: fgcolstr = bgcolstr = None othersettings = '' if ndef['color']: fgcolstr = ndef['color'] if self.usebg and ndef['bgcolor']: bgcolstr = ndef['bgcolor'] if self.usebold and ndef['bold']: othersettings = 'bold' self.style_attrs[str(ttype)] = self.findclosestattr( fgcolstr, bgcolstr, othersettings, colors)
[docs] def formatgenerator(self, tokensource): """Takes a token source, and generates (tokenstring, urwid.AttrSpec) pairs""" for (ttype, tstring) in tokensource: parts = str(ttype).split(".") while str(ttype) not in self.style_attrs: parts = parts[:-1] ttype = ".".join(parts) attr = self.style_attrs[str(ttype)] yield attr, tstring
[docs] def format(self, tokensource, outfile): for (attr, tstring) in self.formatgenerator(tokensource): outfile.write(attr, tstring)