blob: 1d494690a1c8e5eb9e5f36fbb7263f3c890607db [file] [log] [blame]
José Fonsecab96ab8e2011-09-06 10:22:56 +01001##########################################################################
2#
3# Copyright 2011 Jose Fonseca
4# Copyright 2008-2009 VMware, Inc.
5# All Rights Reserved.
6#
7# Permission is hereby granted, free of charge, to any person obtaining a copy
8# of this software and associated documentation files (the "Software"), to deal
9# in the Software without restriction, including without limitation the rights
10# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11# copies of the Software, and to permit persons to whom the Software is
12# furnished to do so, subject to the following conditions:
13#
14# The above copyright notice and this permission notice shall be included in
15# all copies or substantial portions of the Software.
16#
17# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
23# THE SOFTWARE.
24#
25##########################################################################/
26
27
28import sys
29import platform
30
31
32class PlainHighlighter:
33 '''Plain formatter'''
34
35 black = None
36 red = None
37 green = None
38 yellow = None
39 blue = None
40 magenta = None
41 cyan = None
42 white = None
43
44 def __init__(self, stream):
45 self.stream = stream
46
47 def write(self, text):
48 self.stream.write(text)
49
50 def flush(self):
51 self.stream.flush()
52
53 def normal(self):
54 pass
55
56 def color(self, color):
57 pass
58
59 def bold(self):
60 pass
61
62 def italic(self):
63 pass
64
65
66class AnsiHighlighter(PlainHighlighter):
67 '''Highlighter for plain-text files which outputs ANSI escape codes. See
68 http://en.wikipedia.org/wiki/ANSI_escape_code for more information
69 concerning ANSI escape codes.
70 '''
71
72 _csi = '\33['
73
74 _normal = '0m'
75 _bold = '1m'
76 _italic = '3m'
77
78 black = 0
79 red = 1
80 green = 2
81 yellow = 3
82 blue = 4
83 magenta = 5
84 cyan = 6
85 white = 7
86
87 def __init__(self, stream):
88 PlainHighlighter.__init__(self, stream)
José Fonsecab96ab8e2011-09-06 10:22:56 +010089
90 def _escape(self, code):
José Fonsecab412d242012-03-15 23:43:52 +000091 self.stream.write(self._csi + code)
José Fonsecab96ab8e2011-09-06 10:22:56 +010092
93 def normal(self):
94 self._escape(self._normal)
95
96 def color(self, color):
97 self._escape(str(30 + color) + 'm')
98
José Fonseca587ca792012-03-16 07:09:04 +000099 def bold(self, enable = True):
100 if enable:
101 self._escape('1m')
102 else:
103 self._escape('21m')
José Fonsecab96ab8e2011-09-06 10:22:56 +0100104
105 def italic(self):
106 self._escape(self._italic)
107
108
109class WindowsConsoleHighlighter(PlainHighlighter):
110 '''Highlighter for the Windows Console. See
111 http://code.activestate.com/recipes/496901/ for more information.
112 '''
113
114 INVALID_HANDLE_VALUE = -1
115 STD_INPUT_HANDLE = -10
116 STD_OUTPUT_HANDLE = -11
117 STD_ERROR_HANDLE = -12
118
119 FOREGROUND_BLUE = 0x01
120 FOREGROUND_GREEN = 0x02
121 FOREGROUND_RED = 0x04
122 FOREGROUND_INTENSITY = 0x08
123 BACKGROUND_BLUE = 0x10
124 BACKGROUND_GREEN = 0x20
125 BACKGROUND_RED = 0x40
126 BACKGROUND_INTENSITY = 0x80
127
128 COMMON_LVB_LEADING_BYTE = 0x0100
129 COMMON_LVB_TRAILING_BYTE = 0x0200
130 COMMON_LVB_GRID_HORIZONTAL = 0x0400
131 COMMON_LVB_GRID_LVERTICAL = 0x0800
132 COMMON_LVB_GRID_RVERTICAL = 0x1000
133 COMMON_LVB_REVERSE_VIDEO = 0x4000
134 COMMON_LVB_UNDERSCORE = 0x8000
135
136 _normal = FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED
137 _bold = FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_INTENSITY
138 _italic = FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED
139
140 black = 0
141 red = FOREGROUND_RED
142 green = FOREGROUND_GREEN
143 blue = FOREGROUND_BLUE
144 white = FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED
145
146 def __init__(self, stream):
147 PlainHighlighter.__init__(self, stream)
148
149 if stream is sys.stdin:
150 nStdHandle = self.STD_INPUT_HANDLE
151 elif stream is sys.stdout:
152 nStdHandle = self.STD_OUTPUT_HANDLE
153 elif stream is sys.stderr:
154 nStdHandle = self.STD_ERROR_HANDLE
155 else:
156 nStdHandle = None
157
158 if nStdHandle is not None:
159 import ctypes
160 self._handle = ctypes.windll.kernel32.GetStdHandle(nStdHandle)
161 else:
José Fonseca4d73f852012-02-09 14:04:17 +0000162 self._handle = self.INVALID_HANDLE_VALUE
José Fonsecab96ab8e2011-09-06 10:22:56 +0100163
164 self._attribute = self.white
165
166 def _setAttribute(self, attr):
José Fonseca4d73f852012-02-09 14:04:17 +0000167 if self._handle != self.INVALID_HANDLE_VALUE:
José Fonsecab96ab8e2011-09-06 10:22:56 +0100168 import ctypes
169 ctypes.windll.kernel32.SetConsoleTextAttribute(self._handle, attr)
170 self._attribute = attr
171
172 def normal(self):
173 self._setAttribute(self._normal)
174
175 def color(self, color):
José Fonseca4d73f852012-02-09 14:04:17 +0000176 intensity = self._attribute & self.FOREGROUND_INTENSITY
José Fonsecab96ab8e2011-09-06 10:22:56 +0100177 self._setAttribute(color | intensity)
178
179 def bold(self):
José Fonseca4d73f852012-02-09 14:04:17 +0000180 self._setAttribute(self._attribute | self.FOREGROUND_INTENSITY)
José Fonsecab96ab8e2011-09-06 10:22:56 +0100181
182 def italic(self):
183 pass
184
185
José Fonsecab412d242012-03-15 23:43:52 +0000186def Highlighter(stream = sys.stdout, force = False):
187 if force or stream.isatty():
188 if platform.system() == 'Windows':
189 return WindowsConsoleHighlighter(stream)
190 else:
191 return AnsiHighlighter(stream)
José Fonsecab96ab8e2011-09-06 10:22:56 +0100192 else:
José Fonsecab412d242012-03-15 23:43:52 +0000193 return PlainHighlighter(stream)
José Fonsecab96ab8e2011-09-06 10:22:56 +0100194
195
196__all__ = [
197 'Highlighter',
198]