2007-12-31 18:15:39 +03:00
|
|
|
# color.py color output for the status and qseries commands
|
|
|
|
#
|
|
|
|
# Copyright (C) 2007 Kevin Christen <kevin.christen@gmail.com>
|
|
|
|
#
|
2012-01-03 05:30:21 +04:00
|
|
|
# This software may be used and distributed according to the terms of the
|
|
|
|
# GNU General Public License version 2 or any later version.
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2009-06-22 17:48:08 +04:00
|
|
|
'''colorize output from some commands
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2011-03-14 17:19:58 +03:00
|
|
|
This extension modifies the status and resolve commands to add color
|
|
|
|
to their output to reflect file status, the qseries command to add
|
|
|
|
color to reflect patch status (applied, unapplied, missing), and to
|
|
|
|
diff-related commands to highlight additions, removals, diff headers,
|
|
|
|
and trailing whitespace.
|
2008-11-27 00:58:07 +03:00
|
|
|
|
2009-07-26 03:42:15 +04:00
|
|
|
Other effects in addition to color, like bold and underlined text, are
|
2011-04-22 00:47:45 +04:00
|
|
|
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
|
2011-03-14 15:54:25 +03:00
|
|
|
function (aka ANSI escape codes).
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2010-08-28 06:36:35 +04:00
|
|
|
Default effects may be overridden from your configuration file::
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2009-07-23 02:22:35 +04:00
|
|
|
[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
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2009-07-23 02:22:35 +04:00
|
|
|
# 'none' turns off all effects
|
|
|
|
status.clean = none
|
|
|
|
status.copied = none
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2009-07-23 02:22:35 +04:00
|
|
|
qseries.applied = blue bold underline
|
|
|
|
qseries.unapplied = black bold
|
|
|
|
qseries.missing = red bold
|
2008-12-02 18:46:45 +03:00
|
|
|
|
2009-07-23 02:22:35 +04:00
|
|
|
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
|
|
|
|
diff.trailingwhitespace = bold red_background
|
2009-12-11 13:04:31 +03:00
|
|
|
|
2010-01-09 12:47:47 +03:00
|
|
|
resolve.unresolved = red bold
|
|
|
|
resolve.resolved = green bold
|
|
|
|
|
2009-12-11 13:04:31 +03:00
|
|
|
bookmarks.current = green
|
2010-04-06 17:49:19 +04:00
|
|
|
|
2010-07-29 09:05:03 +04:00
|
|
|
branches.active = none
|
|
|
|
branches.closed = black bold
|
|
|
|
branches.current = green
|
|
|
|
branches.inactive = none
|
|
|
|
|
2011-08-21 23:40:10 +04:00
|
|
|
tags.normal = green
|
|
|
|
tags.local = black bold
|
|
|
|
|
2013-05-14 22:23:17 +04:00
|
|
|
rebase.rebased = blue
|
|
|
|
rebase.remaining = red bold
|
|
|
|
|
shelve: add a shelve extension to save/restore working changes
This extension saves shelved changes using a temporary draft commit,
and bundles the temporary commit and its draft ancestors, then
strips them.
This strategy makes it possible to use Mercurial's bundle and merge
machinery to resolve conflicts if necessary when unshelving, even
when the destination commit or its ancestors have been amended,
squashed, or evolved. (Once a change has been unshelved, its
associated unbundled commits are either rolled back or stripped.)
Storing the shelved change as a bundle also avoids the difficulty
that hidden commits would cause, of making it impossible to amend
the parent if it is a draft commits (a common scenario).
Although this extension shares its name and some functionality with
the third party hgshelve extension, it has little else in common.
Notably, the hgshelve extension shelves changes as unified diffs,
which makes conflict resolution a matter of finding .rej files and
conflict markers, and cleaning up the mess by hand.
We do not yet allow hunk-level choosing of changes to record.
Compared to the hgshelve extension, this is a small regression in
usability, but we hope to integrate that at a later point, once the
record machinery becomes more reusable and robust.
2013-08-29 20:22:13 +04:00
|
|
|
shelve.age = cyan
|
|
|
|
shelve.newest = green bold
|
|
|
|
shelve.name = blue bold
|
|
|
|
|
2013-05-14 22:23:18 +04:00
|
|
|
histedit.remaining = red bold
|
|
|
|
|
2011-04-22 00:47:45 +04:00
|
|
|
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.
|
|
|
|
|
2011-06-29 21:20:40 +04:00
|
|
|
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).
|
|
|
|
|
2011-04-22 00:47:45 +04:00
|
|
|
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.
|
|
|
|
|
2011-06-29 21:20:40 +04:00
|
|
|
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::
|
2010-04-06 17:49:19 +04:00
|
|
|
|
|
|
|
[color]
|
2011-06-29 21:20:40 +04:00
|
|
|
mode = terminfo
|
2010-04-06 17:49:19 +04:00
|
|
|
|
2011-04-22 00:47:45 +04:00
|
|
|
Any value other than 'ansi', 'win32', 'terminfo', or 'auto' will
|
|
|
|
disable color.
|
2007-12-31 18:15:39 +03:00
|
|
|
'''
|
|
|
|
|
2011-05-01 17:21:57 +04:00
|
|
|
import os
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2010-08-29 06:57:36 +04:00
|
|
|
from mercurial import commands, dispatch, extensions, ui as uimod, util
|
2013-04-13 04:27:09 +04:00
|
|
|
from mercurial import templater, error
|
2007-12-31 18:15:39 +03:00
|
|
|
from mercurial.i18n import _
|
|
|
|
|
2012-05-15 23:37:49 +04:00
|
|
|
testedwith = 'internal'
|
|
|
|
|
2007-12-31 18:15:39 +03:00
|
|
|
# start and stop parameters for effects
|
2010-04-03 00:22:17 +04:00
|
|
|
_effects = {'none': 0, 'black': 30, 'red': 31, 'green': 32, 'yellow': 33,
|
|
|
|
'blue': 34, 'magenta': 35, 'cyan': 36, 'white': 37, 'bold': 1,
|
|
|
|
'italic': 3, 'underline': 4, 'inverse': 7,
|
|
|
|
'black_background': 40, 'red_background': 41,
|
|
|
|
'green_background': 42, 'yellow_background': 43,
|
|
|
|
'blue_background': 44, 'purple_background': 45,
|
|
|
|
'cyan_background': 46, 'white_background': 47}
|
|
|
|
|
2011-05-25 20:48:00 +04:00
|
|
|
def _terminfosetup(ui, mode):
|
2011-04-22 00:47:45 +04:00
|
|
|
'''Initialize terminfo data and the terminal if we're in terminfo mode.'''
|
|
|
|
|
|
|
|
global _terminfo_params
|
|
|
|
# If we failed to load curses, we go ahead and return.
|
|
|
|
if not _terminfo_params:
|
|
|
|
return
|
|
|
|
# Otherwise, see what the config file says.
|
|
|
|
if mode not in ('auto', 'terminfo'):
|
|
|
|
return
|
|
|
|
|
2011-04-25 01:06:19 +04:00
|
|
|
_terminfo_params.update((key[6:], (False, int(val)))
|
2011-04-22 00:47:45 +04:00
|
|
|
for key, val in ui.configitems('color')
|
2011-04-25 01:06:19 +04:00
|
|
|
if key.startswith('color.'))
|
2011-04-22 00:47:45 +04:00
|
|
|
|
|
|
|
try:
|
|
|
|
curses.setupterm()
|
|
|
|
except curses.error, e:
|
|
|
|
_terminfo_params = {}
|
|
|
|
return
|
|
|
|
|
|
|
|
for key, (b, e) in _terminfo_params.items():
|
|
|
|
if not b:
|
|
|
|
continue
|
|
|
|
if not curses.tigetstr(e):
|
|
|
|
# Most terminals don't support dim, invis, etc, so don't be
|
|
|
|
# noisy and use ui.debug().
|
|
|
|
ui.debug("no terminfo entry for %s\n" % e)
|
|
|
|
del _terminfo_params[key]
|
|
|
|
if not curses.tigetstr('setaf') or not curses.tigetstr('setab'):
|
2011-06-27 22:36:42 +04:00
|
|
|
# Only warn about missing terminfo entries if we explicitly asked for
|
|
|
|
# terminfo mode.
|
|
|
|
if mode == "terminfo":
|
|
|
|
ui.warn(_("no terminfo entry for setab/setaf: reverting to "
|
|
|
|
"ECMA-48 color\n"))
|
2011-04-22 00:47:45 +04:00
|
|
|
_terminfo_params = {}
|
|
|
|
|
2013-06-06 00:06:02 +04:00
|
|
|
def _modesetup(ui, coloropt):
|
2011-05-25 20:48:00 +04:00
|
|
|
global _terminfo_params
|
|
|
|
|
|
|
|
auto = coloropt == 'auto'
|
|
|
|
always = not auto and util.parsebool(coloropt)
|
|
|
|
if not always and not auto:
|
|
|
|
return None
|
|
|
|
|
|
|
|
formatted = always or (os.environ.get('TERM') != 'dumb' and ui.formatted())
|
|
|
|
|
|
|
|
mode = ui.config('color', 'mode', 'auto')
|
|
|
|
realmode = mode
|
|
|
|
if mode == 'auto':
|
|
|
|
if os.name == 'nt' and 'TERM' not in os.environ:
|
|
|
|
# looks line a cmd.exe console, use win32 API or nothing
|
|
|
|
realmode = 'win32'
|
|
|
|
else:
|
2011-06-29 21:20:40 +04:00
|
|
|
realmode = 'ansi'
|
2011-05-25 20:48:00 +04:00
|
|
|
|
|
|
|
if realmode == 'win32':
|
2011-07-29 15:36:27 +04:00
|
|
|
_terminfo_params = {}
|
2011-06-29 21:20:39 +04:00
|
|
|
if not w32effects:
|
|
|
|
if mode == 'win32':
|
|
|
|
# only warn if color.mode is explicitly set to win32
|
|
|
|
ui.warn(_('warning: failed to set color mode to %s\n') % mode)
|
2011-05-25 20:48:00 +04:00
|
|
|
return None
|
|
|
|
_effects.update(w32effects)
|
|
|
|
elif realmode == 'ansi':
|
|
|
|
_terminfo_params = {}
|
|
|
|
elif realmode == 'terminfo':
|
|
|
|
_terminfosetup(ui, mode)
|
|
|
|
if not _terminfo_params:
|
|
|
|
if mode == 'terminfo':
|
|
|
|
## FIXME Shouldn't we return None in this case too?
|
|
|
|
# only warn if color.mode is explicitly set to win32
|
|
|
|
ui.warn(_('warning: failed to set color mode to %s\n') % mode)
|
|
|
|
realmode = 'ansi'
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if always or (auto and formatted):
|
|
|
|
return realmode
|
|
|
|
return None
|
|
|
|
|
2011-04-22 00:47:45 +04:00
|
|
|
try:
|
|
|
|
import curses
|
|
|
|
# Mapping from effect name to terminfo attribute name or color number.
|
|
|
|
# This will also force-load the curses module.
|
|
|
|
_terminfo_params = {'none': (True, 'sgr0'),
|
|
|
|
'standout': (True, 'smso'),
|
|
|
|
'underline': (True, 'smul'),
|
|
|
|
'reverse': (True, 'rev'),
|
|
|
|
'inverse': (True, 'rev'),
|
|
|
|
'blink': (True, 'blink'),
|
|
|
|
'dim': (True, 'dim'),
|
|
|
|
'bold': (True, 'bold'),
|
|
|
|
'invisible': (True, 'invis'),
|
|
|
|
'italic': (True, 'sitm'),
|
|
|
|
'black': (False, curses.COLOR_BLACK),
|
|
|
|
'red': (False, curses.COLOR_RED),
|
|
|
|
'green': (False, curses.COLOR_GREEN),
|
|
|
|
'yellow': (False, curses.COLOR_YELLOW),
|
|
|
|
'blue': (False, curses.COLOR_BLUE),
|
|
|
|
'magenta': (False, curses.COLOR_MAGENTA),
|
|
|
|
'cyan': (False, curses.COLOR_CYAN),
|
|
|
|
'white': (False, curses.COLOR_WHITE)}
|
|
|
|
except ImportError:
|
|
|
|
_terminfo_params = False
|
|
|
|
|
2010-04-03 00:22:17 +04:00
|
|
|
_styles = {'grep.match': 'red bold',
|
2012-10-14 22:27:55 +04:00
|
|
|
'grep.linenumber': 'green',
|
|
|
|
'grep.rev': 'green',
|
|
|
|
'grep.change': 'green',
|
|
|
|
'grep.sep': 'cyan',
|
|
|
|
'grep.filename': 'magenta',
|
|
|
|
'grep.user': 'magenta',
|
|
|
|
'grep.date': 'magenta',
|
2011-02-10 22:46:28 +03:00
|
|
|
'bookmarks.current': 'green',
|
2010-07-29 09:05:03 +04:00
|
|
|
'branches.active': 'none',
|
|
|
|
'branches.closed': 'black bold',
|
|
|
|
'branches.current': 'green',
|
|
|
|
'branches.inactive': 'none',
|
2010-04-03 00:22:17 +04:00
|
|
|
'diff.changed': 'white',
|
|
|
|
'diff.deleted': 'red',
|
|
|
|
'diff.diffline': 'bold',
|
|
|
|
'diff.extended': 'cyan bold',
|
|
|
|
'diff.file_a': 'red bold',
|
|
|
|
'diff.file_b': 'green bold',
|
|
|
|
'diff.hunk': 'magenta',
|
|
|
|
'diff.inserted': 'green',
|
|
|
|
'diff.trailingwhitespace': 'bold red_background',
|
|
|
|
'diffstat.deleted': 'red',
|
|
|
|
'diffstat.inserted': 'green',
|
2013-05-14 22:23:18 +04:00
|
|
|
'histedit.remaining': 'red bold',
|
2011-03-27 14:59:25 +04:00
|
|
|
'ui.prompt': 'yellow',
|
2010-04-03 00:22:17 +04:00
|
|
|
'log.changeset': 'yellow',
|
2013-05-14 22:23:17 +04:00
|
|
|
'rebase.rebased': 'blue',
|
|
|
|
'rebase.remaining': 'red bold',
|
2010-04-03 00:22:17 +04:00
|
|
|
'resolve.resolved': 'green bold',
|
|
|
|
'resolve.unresolved': 'red bold',
|
shelve: add a shelve extension to save/restore working changes
This extension saves shelved changes using a temporary draft commit,
and bundles the temporary commit and its draft ancestors, then
strips them.
This strategy makes it possible to use Mercurial's bundle and merge
machinery to resolve conflicts if necessary when unshelving, even
when the destination commit or its ancestors have been amended,
squashed, or evolved. (Once a change has been unshelved, its
associated unbundled commits are either rolled back or stripped.)
Storing the shelved change as a bundle also avoids the difficulty
that hidden commits would cause, of making it impossible to amend
the parent if it is a draft commits (a common scenario).
Although this extension shares its name and some functionality with
the third party hgshelve extension, it has little else in common.
Notably, the hgshelve extension shelves changes as unified diffs,
which makes conflict resolution a matter of finding .rej files and
conflict markers, and cleaning up the mess by hand.
We do not yet allow hunk-level choosing of changes to record.
Compared to the hgshelve extension, this is a small regression in
usability, but we hope to integrate that at a later point, once the
record machinery becomes more reusable and robust.
2013-08-29 20:22:13 +04:00
|
|
|
'shelve.age': 'cyan',
|
|
|
|
'shelve.newest': 'green bold',
|
|
|
|
'shelve.name': 'blue bold',
|
2010-04-03 00:22:17 +04:00
|
|
|
'status.added': 'green bold',
|
|
|
|
'status.clean': 'none',
|
|
|
|
'status.copied': 'none',
|
|
|
|
'status.deleted': 'cyan bold underline',
|
|
|
|
'status.ignored': 'black bold',
|
|
|
|
'status.modified': 'blue bold',
|
|
|
|
'status.removed': 'red bold',
|
2011-08-21 23:40:10 +04:00
|
|
|
'status.unknown': 'magenta bold underline',
|
|
|
|
'tags.normal': 'green',
|
|
|
|
'tags.local': 'black bold'}
|
2010-04-03 00:22:17 +04:00
|
|
|
|
2007-12-31 18:15:39 +03:00
|
|
|
|
2011-04-22 00:47:45 +04:00
|
|
|
def _effect_str(effect):
|
|
|
|
'''Helper function for render_effects().'''
|
|
|
|
|
|
|
|
bg = False
|
|
|
|
if effect.endswith('_background'):
|
|
|
|
bg = True
|
|
|
|
effect = effect[:-11]
|
|
|
|
attr, val = _terminfo_params[effect]
|
|
|
|
if attr:
|
|
|
|
return curses.tigetstr(val)
|
|
|
|
elif bg:
|
|
|
|
return curses.tparm(curses.tigetstr('setab'), val)
|
|
|
|
else:
|
|
|
|
return curses.tparm(curses.tigetstr('setaf'), val)
|
|
|
|
|
2009-05-26 00:44:53 +04:00
|
|
|
def render_effects(text, effects):
|
2007-12-31 18:15:39 +03:00
|
|
|
'Wrap text in commands to turn on each effect.'
|
2010-04-03 00:22:17 +04:00
|
|
|
if not text:
|
|
|
|
return text
|
2011-04-22 00:47:45 +04:00
|
|
|
if not _terminfo_params:
|
|
|
|
start = [str(_effects[e]) for e in ['none'] + effects.split()]
|
|
|
|
start = '\033[' + ';'.join(start) + 'm'
|
|
|
|
stop = '\033[' + str(_effects['none']) + 'm'
|
|
|
|
else:
|
|
|
|
start = ''.join(_effect_str(effect)
|
|
|
|
for effect in ['none'] + effects.split())
|
|
|
|
stop = _effect_str('none')
|
2010-04-03 00:22:19 +04:00
|
|
|
return ''.join([start, text, stop])
|
2010-04-03 00:22:17 +04:00
|
|
|
|
|
|
|
def extstyles():
|
|
|
|
for name, ext in extensions.extensions():
|
|
|
|
_styles.update(getattr(ext, 'colortable', {}))
|
|
|
|
|
2014-04-08 00:33:46 +04:00
|
|
|
def valideffect(effect):
|
|
|
|
'Determine if the effect is valid or not.'
|
|
|
|
good = False
|
|
|
|
if not _terminfo_params and effect in _effects:
|
|
|
|
good = True
|
|
|
|
elif effect in _terminfo_params or effect[:-11] in _terminfo_params:
|
|
|
|
good = True
|
|
|
|
return good
|
|
|
|
|
2010-04-03 00:22:17 +04:00
|
|
|
def configstyles(ui):
|
|
|
|
for status, cfgeffects in ui.configitems('color'):
|
2011-04-22 00:47:45 +04:00
|
|
|
if '.' not in status or status.startswith('color.'):
|
2010-04-03 00:22:17 +04:00
|
|
|
continue
|
|
|
|
cfgeffects = ui.configlist('color', status)
|
|
|
|
if cfgeffects:
|
2009-06-25 17:23:33 +04:00
|
|
|
good = []
|
2010-04-03 00:22:17 +04:00
|
|
|
for e in cfgeffects:
|
2014-04-08 00:33:46 +04:00
|
|
|
if valideffect(e):
|
2009-06-25 17:23:33 +04:00
|
|
|
good.append(e)
|
|
|
|
else:
|
|
|
|
ui.warn(_("ignoring unknown color/effect %r "
|
|
|
|
"(configured in color.%s)\n")
|
2010-04-03 00:22:17 +04:00
|
|
|
% (e, status))
|
|
|
|
_styles[status] = ' '.join(good)
|
|
|
|
|
2010-07-02 04:23:26 +04:00
|
|
|
class colorui(uimod.ui):
|
|
|
|
def popbuffer(self, labeled=False):
|
2013-04-20 03:57:20 +04:00
|
|
|
if self._colormode is None:
|
|
|
|
return super(colorui, self).popbuffer(labeled)
|
|
|
|
|
2010-07-02 04:23:26 +04:00
|
|
|
if labeled:
|
|
|
|
return ''.join(self.label(a, label) for a, label
|
|
|
|
in self._buffers.pop())
|
|
|
|
return ''.join(a for a, label in self._buffers.pop())
|
|
|
|
|
|
|
|
_colormode = 'ansi'
|
|
|
|
def write(self, *args, **opts):
|
2013-04-20 03:57:20 +04:00
|
|
|
if self._colormode is None:
|
|
|
|
return super(colorui, self).write(*args, **opts)
|
|
|
|
|
2010-07-02 04:23:26 +04:00
|
|
|
label = opts.get('label', '')
|
|
|
|
if self._buffers:
|
|
|
|
self._buffers[-1].extend([(str(a), label) for a in args])
|
|
|
|
elif self._colormode == 'win32':
|
|
|
|
for a in args:
|
2010-08-02 00:26:02 +04:00
|
|
|
win32print(a, super(colorui, self).write, **opts)
|
2010-07-02 04:23:26 +04:00
|
|
|
else:
|
|
|
|
return super(colorui, self).write(
|
|
|
|
*[self.label(str(a), label) for a in args], **opts)
|
|
|
|
|
|
|
|
def write_err(self, *args, **opts):
|
2013-04-20 03:57:20 +04:00
|
|
|
if self._colormode is None:
|
|
|
|
return super(colorui, self).write_err(*args, **opts)
|
|
|
|
|
2010-07-02 04:23:26 +04:00
|
|
|
label = opts.get('label', '')
|
|
|
|
if self._colormode == 'win32':
|
|
|
|
for a in args:
|
2010-08-02 00:26:02 +04:00
|
|
|
win32print(a, super(colorui, self).write_err, **opts)
|
2010-07-02 04:23:26 +04:00
|
|
|
else:
|
2010-08-02 18:48:31 +04:00
|
|
|
return super(colorui, self).write_err(
|
2010-07-02 04:23:26 +04:00
|
|
|
*[self.label(str(a), label) for a in args], **opts)
|
|
|
|
|
|
|
|
def label(self, msg, label):
|
2013-04-20 03:57:20 +04:00
|
|
|
if self._colormode is None:
|
|
|
|
return super(colorui, self).label(msg, label)
|
|
|
|
|
2010-07-02 04:23:26 +04:00
|
|
|
effects = []
|
|
|
|
for l in label.split():
|
|
|
|
s = _styles.get(l, '')
|
|
|
|
if s:
|
|
|
|
effects.append(s)
|
2014-04-08 00:37:27 +04:00
|
|
|
elif valideffect(l):
|
|
|
|
effects.append(l)
|
2011-05-01 18:54:48 +04:00
|
|
|
effects = ' '.join(effects)
|
2010-07-02 04:23:26 +04:00
|
|
|
if effects:
|
|
|
|
return '\n'.join([render_effects(s, effects)
|
|
|
|
for s in msg.split('\n')])
|
|
|
|
return msg
|
|
|
|
|
2012-12-29 05:25:10 +04:00
|
|
|
def templatelabel(context, mapping, args):
|
|
|
|
if len(args) != 2:
|
|
|
|
# i18n: "label" is a keyword
|
|
|
|
raise error.ParseError(_("label expects two arguments"))
|
|
|
|
|
2013-11-19 00:37:09 +04:00
|
|
|
thing = templater._evalifliteral(args[1], context, mapping)
|
2012-12-29 05:25:10 +04:00
|
|
|
|
|
|
|
# apparently, repo could be a string that is the favicon?
|
|
|
|
repo = mapping.get('repo', '')
|
|
|
|
if isinstance(repo, str):
|
|
|
|
return thing
|
|
|
|
|
2014-03-09 20:01:42 +04:00
|
|
|
label = templater._evalifliteral(args[0], context, mapping)
|
2012-12-29 05:25:10 +04:00
|
|
|
|
|
|
|
thing = templater.stringify(thing)
|
|
|
|
label = templater.stringify(label)
|
|
|
|
|
|
|
|
return repo.ui.label(thing, label)
|
2010-04-03 00:22:17 +04:00
|
|
|
|
|
|
|
def uisetup(ui):
|
2010-04-06 17:54:18 +04:00
|
|
|
if ui.plain():
|
|
|
|
return
|
2013-06-06 00:06:16 +04:00
|
|
|
if not isinstance(ui, colorui):
|
2013-04-19 01:53:24 +04:00
|
|
|
colorui.__bases__ = (ui.__class__,)
|
|
|
|
ui.__class__ = colorui
|
2010-04-03 00:22:17 +04:00
|
|
|
def colorcmd(orig, ui_, opts, cmd, cmdfunc):
|
2013-06-06 00:06:02 +04:00
|
|
|
mode = _modesetup(ui_, opts['color'])
|
2013-04-20 03:57:10 +04:00
|
|
|
colorui._colormode = mode
|
2011-05-25 20:48:00 +04:00
|
|
|
if mode:
|
2010-04-03 00:22:17 +04:00
|
|
|
extstyles()
|
2010-07-02 04:23:26 +04:00
|
|
|
configstyles(ui_)
|
2010-04-03 00:22:17 +04:00
|
|
|
return orig(ui_, opts, cmd, cmdfunc)
|
|
|
|
extensions.wrapfunction(dispatch, '_runcommand', colorcmd)
|
2012-12-29 05:25:10 +04:00
|
|
|
templater.funcs['label'] = templatelabel
|
2010-04-03 00:22:17 +04:00
|
|
|
|
2010-10-10 21:21:36 +04:00
|
|
|
def extsetup(ui):
|
|
|
|
commands.globalopts.append(
|
|
|
|
('', 'color', 'auto',
|
2010-10-22 18:34:38 +04:00
|
|
|
# i18n: 'always', 'auto', and 'never' are keywords and should
|
|
|
|
# not be translated
|
2010-10-10 21:21:36 +04:00
|
|
|
_("when to colorize (boolean, always, auto, or never)"),
|
|
|
|
_('TYPE')))
|
2010-04-06 17:49:19 +04:00
|
|
|
|
2014-02-12 04:00:51 +04:00
|
|
|
def debugcolor(ui, repo, **opts):
|
|
|
|
global _styles
|
|
|
|
_styles = {}
|
|
|
|
for effect in _effects.keys():
|
|
|
|
_styles[effect] = effect
|
2014-04-13 21:01:00 +04:00
|
|
|
ui.write(('color mode: %s\n') % ui._colormode)
|
2014-02-12 04:00:51 +04:00
|
|
|
ui.write(_('available colors:\n'))
|
|
|
|
for label, colors in _styles.items():
|
|
|
|
ui.write(('%s\n') % colors, label=label)
|
|
|
|
|
2011-03-11 13:14:21 +03:00
|
|
|
if os.name != 'nt':
|
|
|
|
w32effects = None
|
|
|
|
else:
|
|
|
|
import re, ctypes
|
|
|
|
|
|
|
|
_kernel32 = ctypes.windll.kernel32
|
|
|
|
|
|
|
|
_WORD = ctypes.c_ushort
|
|
|
|
|
|
|
|
_INVALID_HANDLE_VALUE = -1
|
|
|
|
|
|
|
|
class _COORD(ctypes.Structure):
|
|
|
|
_fields_ = [('X', ctypes.c_short),
|
|
|
|
('Y', ctypes.c_short)]
|
|
|
|
|
|
|
|
class _SMALL_RECT(ctypes.Structure):
|
|
|
|
_fields_ = [('Left', ctypes.c_short),
|
|
|
|
('Top', ctypes.c_short),
|
|
|
|
('Right', ctypes.c_short),
|
|
|
|
('Bottom', ctypes.c_short)]
|
|
|
|
|
|
|
|
class _CONSOLE_SCREEN_BUFFER_INFO(ctypes.Structure):
|
|
|
|
_fields_ = [('dwSize', _COORD),
|
|
|
|
('dwCursorPosition', _COORD),
|
|
|
|
('wAttributes', _WORD),
|
|
|
|
('srWindow', _SMALL_RECT),
|
|
|
|
('dwMaximumWindowSize', _COORD)]
|
|
|
|
|
|
|
|
_STD_OUTPUT_HANDLE = 0xfffffff5L # (DWORD)-11
|
|
|
|
_STD_ERROR_HANDLE = 0xfffffff4L # (DWORD)-12
|
|
|
|
|
|
|
|
_FOREGROUND_BLUE = 0x0001
|
|
|
|
_FOREGROUND_GREEN = 0x0002
|
|
|
|
_FOREGROUND_RED = 0x0004
|
|
|
|
_FOREGROUND_INTENSITY = 0x0008
|
|
|
|
|
|
|
|
_BACKGROUND_BLUE = 0x0010
|
|
|
|
_BACKGROUND_GREEN = 0x0020
|
|
|
|
_BACKGROUND_RED = 0x0040
|
|
|
|
_BACKGROUND_INTENSITY = 0x0080
|
|
|
|
|
|
|
|
_COMMON_LVB_REVERSE_VIDEO = 0x4000
|
|
|
|
_COMMON_LVB_UNDERSCORE = 0x8000
|
2010-04-06 17:49:19 +04:00
|
|
|
|
|
|
|
# http://msdn.microsoft.com/en-us/library/ms682088%28VS.85%29.aspx
|
|
|
|
w32effects = {
|
2010-09-13 18:12:25 +04:00
|
|
|
'none': -1,
|
2010-04-06 17:49:19 +04:00
|
|
|
'black': 0,
|
2011-03-11 13:14:21 +03:00
|
|
|
'red': _FOREGROUND_RED,
|
|
|
|
'green': _FOREGROUND_GREEN,
|
|
|
|
'yellow': _FOREGROUND_RED | _FOREGROUND_GREEN,
|
|
|
|
'blue': _FOREGROUND_BLUE,
|
|
|
|
'magenta': _FOREGROUND_BLUE | _FOREGROUND_RED,
|
|
|
|
'cyan': _FOREGROUND_BLUE | _FOREGROUND_GREEN,
|
|
|
|
'white': _FOREGROUND_RED | _FOREGROUND_GREEN | _FOREGROUND_BLUE,
|
|
|
|
'bold': _FOREGROUND_INTENSITY,
|
2010-09-13 18:25:13 +04:00
|
|
|
'black_background': 0x100, # unused value > 0x0f
|
2011-03-11 13:14:21 +03:00
|
|
|
'red_background': _BACKGROUND_RED,
|
|
|
|
'green_background': _BACKGROUND_GREEN,
|
|
|
|
'yellow_background': _BACKGROUND_RED | _BACKGROUND_GREEN,
|
|
|
|
'blue_background': _BACKGROUND_BLUE,
|
|
|
|
'purple_background': _BACKGROUND_BLUE | _BACKGROUND_RED,
|
|
|
|
'cyan_background': _BACKGROUND_BLUE | _BACKGROUND_GREEN,
|
|
|
|
'white_background': (_BACKGROUND_RED | _BACKGROUND_GREEN |
|
|
|
|
_BACKGROUND_BLUE),
|
|
|
|
'bold_background': _BACKGROUND_INTENSITY,
|
|
|
|
'underline': _COMMON_LVB_UNDERSCORE, # double-byte charsets only
|
|
|
|
'inverse': _COMMON_LVB_REVERSE_VIDEO, # double-byte charsets only
|
2010-04-06 17:49:19 +04:00
|
|
|
}
|
|
|
|
|
2011-03-11 13:14:21 +03:00
|
|
|
passthrough = set([_FOREGROUND_INTENSITY,
|
|
|
|
_BACKGROUND_INTENSITY,
|
|
|
|
_COMMON_LVB_UNDERSCORE,
|
|
|
|
_COMMON_LVB_REVERSE_VIDEO])
|
|
|
|
|
|
|
|
stdout = _kernel32.GetStdHandle(
|
|
|
|
_STD_OUTPUT_HANDLE) # don't close the handle returned
|
|
|
|
if stdout is None or stdout == _INVALID_HANDLE_VALUE:
|
|
|
|
w32effects = None
|
|
|
|
else:
|
|
|
|
csbi = _CONSOLE_SCREEN_BUFFER_INFO()
|
|
|
|
if not _kernel32.GetConsoleScreenBufferInfo(
|
|
|
|
stdout, ctypes.byref(csbi)):
|
|
|
|
# stdout may not support GetConsoleScreenBufferInfo()
|
|
|
|
# when called from subprocess or redirected
|
|
|
|
w32effects = None
|
|
|
|
else:
|
|
|
|
origattr = csbi.wAttributes
|
|
|
|
ansire = re.compile('\033\[([^m]*)m([^\033]*)(.*)',
|
|
|
|
re.MULTILINE | re.DOTALL)
|
2010-04-06 17:49:19 +04:00
|
|
|
|
|
|
|
def win32print(text, orig, **opts):
|
|
|
|
label = opts.get('label', '')
|
2010-09-13 18:12:25 +04:00
|
|
|
attr = origattr
|
|
|
|
|
|
|
|
def mapcolor(val, attr):
|
|
|
|
if val == -1:
|
|
|
|
return origattr
|
|
|
|
elif val in passthrough:
|
|
|
|
return attr | val
|
|
|
|
elif val > 0x0f:
|
|
|
|
return (val & 0x70) | (attr & 0x8f)
|
|
|
|
else:
|
|
|
|
return (val & 0x07) | (attr & 0xf8)
|
2010-04-06 17:49:19 +04:00
|
|
|
|
|
|
|
# determine console attributes based on labels
|
|
|
|
for l in label.split():
|
|
|
|
style = _styles.get(l, '')
|
|
|
|
for effect in style.split():
|
2010-09-13 18:12:25 +04:00
|
|
|
attr = mapcolor(w32effects[effect], attr)
|
2010-04-06 17:49:19 +04:00
|
|
|
|
|
|
|
# hack to ensure regexp finds data
|
|
|
|
if not text.startswith('\033['):
|
|
|
|
text = '\033[m' + text
|
|
|
|
|
|
|
|
# Look for ANSI-like codes embedded in text
|
|
|
|
m = re.match(ansire, text)
|
2011-04-10 00:53:23 +04:00
|
|
|
|
|
|
|
try:
|
|
|
|
while m:
|
|
|
|
for sattr in m.group(1).split(';'):
|
|
|
|
if sattr:
|
|
|
|
attr = mapcolor(int(sattr), attr)
|
|
|
|
_kernel32.SetConsoleTextAttribute(stdout, attr)
|
|
|
|
orig(m.group(2), **opts)
|
|
|
|
m = re.match(ansire, m.group(3))
|
|
|
|
finally:
|
2012-08-16 00:38:42 +04:00
|
|
|
# Explicitly reset original attributes
|
2011-04-10 00:53:23 +04:00
|
|
|
_kernel32.SetConsoleTextAttribute(stdout, origattr)
|
2014-02-12 04:00:51 +04:00
|
|
|
|
|
|
|
cmdtable = {
|
|
|
|
'debugcolor':
|
|
|
|
(debugcolor, [], ('hg debugcolor'))
|
|
|
|
}
|