color.py
241 lines
| 7.7 KiB
| text/x-python
|
PythonLexer
/ hgext / color.py
Jordi Gutiérrez Hermoso
|
r22771 | # color.py color output for Mercurial commands | ||
Kevin Christen
|
r5787 | # | ||
# Copyright (C) 2007 Kevin Christen <kevin.christen@gmail.com> | ||||
# | ||||
Augie Fackler
|
r15771 | # This software may be used and distributed according to the terms of the | ||
# GNU General Public License version 2 or any later version. | ||||
Kevin Christen
|
r5787 | |||
Cédric Duval
|
r8894 | '''colorize output from some commands | ||
Kevin Christen
|
r5787 | |||
Jordi Gutiérrez Hermoso
|
r22771 | The color extension colorizes output from several Mercurial commands. | ||
For example, the diff command shows additions in green and deletions | ||||
in red, while the status command shows modified files in magenta. Many | ||||
other commands have analogous colors. It is possible to customize | ||||
these colors. | ||||
Georg Brandl
|
r7457 | |||
Jordi Gutiérrez Hermoso
|
r22772 | Effects | ||
------- | ||||
Martin Geisler
|
r7988 | Other effects in addition to color, like bold and underlined text, are | ||
Danek Duvall
|
r13987 | also available. By default, the terminfo database is used to find the | ||
terminal codes used to change color and effect. If terminfo is not | ||||
available, then effects are rendered with the ECMA-48 SGR control | ||||
Martin Geisler
|
r13635 | function (aka ANSI escape codes). | ||
Kevin Christen
|
r5787 | |||
Jordi Gutiérrez Hermoso
|
r22772 | The available effects in terminfo mode are 'blink', 'bold', 'dim', | ||
'inverse', 'invisible', 'italic', 'standout', and 'underline'; in | ||||
ECMA-48 mode, the options are 'bold', 'inverse', 'italic', and | ||||
'underline'. How each is rendered depends on the terminal emulator. | ||||
Some may not be available for a given terminal type, and will be | ||||
silently ignored. | ||||
Danek Duvall
|
r30176 | If the terminfo entry for your terminal is missing codes for an effect | ||
or has the wrong codes, you can add or override those codes in your | ||||
configuration:: | ||||
[color] | ||||
terminfo.dim = \E[2m | ||||
where '\E' is substituted with an escape character. | ||||
Jordi Gutiérrez Hermoso
|
r22772 | Labels | ||
------ | ||||
Jordi Gutiérrez Hermoso
|
r22462 | Text receives color effects depending on the labels that it has. Many | ||
default Mercurial commands emit labelled text. You can also define | ||||
your own labels in templates using the label function, see :hg:`help | ||||
templates`. A single portion of text may have more than one label. In | ||||
that case, effects given to the last label will override any other | ||||
effects. This includes the special "none" effect, which nullifies | ||||
other effects. | ||||
Jordi Gutiérrez Hermoso
|
r22463 | Labels are normally invisible. In order to see these labels and their | ||
Jordi Gutiérrez Hermoso
|
r22711 | position in the text, use the global --color=debug option. The same | ||
anchor text may be associated to multiple labels, e.g. | ||||
Jordi Gutiérrez Hermoso
|
r22463 | |||
Jordi Gutiérrez Hermoso
|
r22711 | [log.changeset changeset.secret|changeset: 22611:6f0a53c8f587] | ||
Jordi Gutiérrez Hermoso
|
r22463 | |||
Jordi Gutiérrez Hermoso
|
r22462 | The following are the default effects for some default labels. Default | ||
effects may be overridden from your configuration file:: | ||||
Kevin Christen
|
r5787 | |||
Martin Geisler
|
r9206 | [color] | ||
status.modified = blue bold underline red_background | ||||
status.added = green bold | ||||
status.removed = red bold blue_background | ||||
status.deleted = cyan bold underline | ||||
status.unknown = magenta bold underline | ||||
status.ignored = black bold | ||||
Kevin Christen
|
r5787 | |||
Martin Geisler
|
r9206 | # 'none' turns off all effects | ||
status.clean = none | ||||
status.copied = none | ||||
Kevin Christen
|
r5787 | |||
Martin Geisler
|
r9206 | qseries.applied = blue bold underline | ||
qseries.unapplied = black bold | ||||
qseries.missing = red bold | ||||
Brodie Rao
|
r7456 | |||
Martin Geisler
|
r9206 | diff.diffline = bold | ||
diff.extended = cyan bold | ||||
diff.file_a = red bold | ||||
diff.file_b = green bold | ||||
diff.hunk = magenta | ||||
diff.deleted = red | ||||
diff.inserted = green | ||||
diff.changed = white | ||||
Jordi Gutiérrez Hermoso
|
r22710 | diff.tab = | ||
Martin Geisler
|
r9206 | diff.trailingwhitespace = bold red_background | ||
David Soria Parra
|
r10046 | |||
Jordi Gutiérrez Hermoso
|
r22465 | # Blank so it inherits the style of the surrounding label | ||
changeset.public = | ||||
changeset.draft = | ||||
changeset.secret = | ||||
Georg Brandl
|
r10223 | resolve.unresolved = red bold | ||
resolve.resolved = green bold | ||||
Ryan McElroy
|
r25347 | bookmarks.active = green | ||
Steve Borho
|
r10870 | |||
Jeremy Whitlock
|
r11969 | branches.active = none | ||
branches.closed = black bold | ||||
branches.current = green | ||||
branches.inactive = none | ||||
Marc Simpson
|
r15048 | tags.normal = green | ||
tags.local = black bold | ||||
Bryan O'Sullivan
|
r19214 | rebase.rebased = blue | ||
rebase.remaining = red bold | ||||
David Soria Parra
|
r19854 | shelve.age = cyan | ||
shelve.newest = green bold | ||||
shelve.name = blue bold | ||||
Bryan O'Sullivan
|
r19215 | histedit.remaining = red bold | ||
Jordi Gutiérrez Hermoso
|
r22772 | Custom colors | ||
------------- | ||||
Brodie Rao
|
r14769 | |||
Danek Duvall
|
r13987 | Because there are only eight standard colors, this module allows you | ||
to define color names for other color slots which might be available | ||||
for your terminal type, assuming terminfo mode. For instance:: | ||||
color.brightblue = 12 | ||||
color.pink = 207 | ||||
color.orange = 202 | ||||
to set 'brightblue' to color slot 12 (useful for 16 color terminals | ||||
that have brighter colors defined in the upper eight) and, 'pink' and | ||||
'orange' to colors in 256-color xterm's default color cube. These | ||||
defined colors may then be used as any of the pre-defined eight, | ||||
including appending '_background' to set the background to that color. | ||||
Jordi Gutiérrez Hermoso
|
r22772 | Modes | ||
----- | ||||
Brodie Rao
|
r14769 | By default, the color extension will use ANSI mode (or win32 mode on | ||
Windows) if it detects a terminal. To override auto mode (to enable | ||||
terminfo mode, for example), set the following configuration option:: | ||||
Steve Borho
|
r10870 | |||
[color] | ||||
Brodie Rao
|
r14769 | mode = terminfo | ||
Steve Borho
|
r10870 | |||
Danek Duvall
|
r13987 | Any value other than 'ansi', 'win32', 'terminfo', or 'auto' will | ||
disable color. | ||||
Jordi Gutiérrez Hermoso
|
r22772 | |||
Note that on some systems, terminfo mode may cause problems when using | ||||
color with the pager extension and less -R. less with the -R option | ||||
will only display ECMA-48 color codes, and terminfo mode may sometimes | ||||
emit codes that less doesn't understand. You can work around this by | ||||
either using ansi mode (or auto mode), or by using less -r (which will | ||||
pass through all terminal control codes, not just color control | ||||
codes). | ||||
Gregory Szorc
|
r24068 | |||
On some systems (such as MSYS in Windows), the terminal may support | ||||
a different color mode than the pager (activated via the "pager" | ||||
extension). It is possible to define separate modes depending on whether | ||||
the pager is active:: | ||||
[color] | ||||
mode = auto | ||||
pagermode = ansi | ||||
If ``pagermode`` is not defined, the ``mode`` will be used. | ||||
Kevin Christen
|
r5787 | ''' | ||
Pulkit Goyal
|
r28968 | from __future__ import absolute_import | ||
Kevin Christen
|
r5787 | |||
Pierre-Yves David
|
r30968 | try: | ||
import curses | ||||
curses.COLOR_BLACK # force import | ||||
except ImportError: | ||||
curses = None | ||||
Yuya Nishihara
|
r29205 | from mercurial.i18n import _ | ||
Pulkit Goyal
|
r28968 | from mercurial import ( | ||
cmdutil, | ||||
Pierre-Yves David
|
r30652 | color, | ||
Pulkit Goyal
|
r28968 | commands, | ||
) | ||||
Kevin Christen
|
r5787 | |||
Gregory Szorc
|
r21249 | cmdtable = {} | ||
command = cmdutil.command(cmdtable) | ||||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r29841 | testedwith = 'ships-with-hg-core' | ||
Augie Fackler
|
r16743 | |||
Brodie Rao
|
r12693 | def extsetup(ui): | ||
Pierre-Yves David
|
r31104 | # change default color config | ||
Pierre-Yves David
|
r31110 | color._enabledbydefault = True | ||
Pierre-Yves David
|
r31104 | for idx, entry in enumerate(commands.globalopts): | ||
if entry[1] == 'color': | ||||
Pierre-Yves David
|
r31110 | patch = (entry[3].replace(' (EXPERIMENTAL)', ''),) | ||
new = entry[:3] + patch + entry[4:] | ||||
Pierre-Yves David
|
r31104 | commands.globalopts[idx] = new | ||
break | ||||
Steve Borho
|
r10870 | |||
Pierre-Yves David
|
r30286 | @command('debugcolor', | ||
[('', 'style', None, _('show all configured styles'))], | ||||
'hg debugcolor') | ||||
Olle Lundberg
|
r20455 | def debugcolor(ui, repo, **opts): | ||
Pierre-Yves David
|
r30286 | """show available color, effects or style""" | ||
Pierre-Yves David
|
r30284 | ui.write(('color mode: %s\n') % ui._colormode) | ||
Pierre-Yves David
|
r30286 | if opts.get('style'): | ||
return _debugdisplaystyle(ui) | ||||
else: | ||||
return _debugdisplaycolor(ui) | ||||
Pierre-Yves David
|
r30284 | |||
def _debugdisplaycolor(ui): | ||||
Pierre-Yves David
|
r30652 | oldstyle = color._styles.copy() | ||
Pierre-Yves David
|
r30283 | try: | ||
Pierre-Yves David
|
r30652 | color._styles.clear() | ||
Pierre-Yves David
|
r30967 | for effect in color._effects.keys(): | ||
Pierre-Yves David
|
r30652 | color._styles[effect] = effect | ||
Pierre-Yves David
|
r31113 | if ui._terminfoparams: | ||
Pierre-Yves David
|
r30283 | for k, v in ui.configitems('color'): | ||
if k.startswith('color.'): | ||||
Pierre-Yves David
|
r30652 | color._styles[k] = k[6:] | ||
Pierre-Yves David
|
r30283 | elif k.startswith('terminfo.'): | ||
Pierre-Yves David
|
r30652 | color._styles[k] = k[9:] | ||
Pierre-Yves David
|
r30283 | ui.write(_('available colors:\n')) | ||
Pierre-Yves David
|
r30285 | # sort label with a '_' after the other to group '_background' entry. | ||
Pierre-Yves David
|
r30652 | items = sorted(color._styles.items(), | ||
Pierre-Yves David
|
r30285 | key=lambda i: ('_' in i[0], i[0], i[1])) | ||
for colorname, label in items: | ||||
Pierre-Yves David
|
r30283 | ui.write(('%s\n') % colorname, label=label) | ||
finally: | ||||
Pierre-Yves David
|
r30652 | color._styles.clear() | ||
color._styles.update(oldstyle) | ||||
Olle Lundberg
|
r20455 | |||
Pierre-Yves David
|
r30286 | def _debugdisplaystyle(ui): | ||
ui.write(_('available style:\n')) | ||||
Pierre-Yves David
|
r30652 | width = max(len(s) for s in color._styles) | ||
for label, effects in sorted(color._styles.items()): | ||||
Pierre-Yves David
|
r30286 | ui.write('%s' % label, label=label) | ||
if effects: | ||||
# 50 | ||||
ui.write(': ') | ||||
ui.write(' ' * (max(0, width - len(label)))) | ||||
ui.write(', '.join(ui.label(e, e) for e in effects.split())) | ||||
ui.write('\n') | ||||