##// END OF EJS Templates
Merge pull request #762 from jdmarch/jdmarch-755-execute-file-hack...
Evan Patterson -
r4775:af8cee08 merge
parent child Browse files
Show More
@@ -1,513 +1,525
1 1 """ A FrontendWidget that emulates the interface of the console IPython and
2 2 supports the additional functionality provided by the IPython kernel.
3 3 """
4 4
5 5 #-----------------------------------------------------------------------------
6 6 # Imports
7 7 #-----------------------------------------------------------------------------
8 8
9 9 # Standard library imports
10 10 from collections import namedtuple
11 11 import os.path
12 12 import re
13 13 from subprocess import Popen
14 14 import sys
15 15 from textwrap import dedent
16 16
17 17 # System library imports
18 18 from IPython.external.qt import QtCore, QtGui
19 19
20 20 # Local imports
21 21 from IPython.core.inputsplitter import IPythonInputSplitter, \
22 22 transform_ipy_prompt
23 23 from IPython.utils.traitlets import Bool, Unicode
24 24 from frontend_widget import FrontendWidget
25 25 import styles
26 26
27 27 #-----------------------------------------------------------------------------
28 28 # Constants
29 29 #-----------------------------------------------------------------------------
30 30
31 31 # Default strings to build and display input and output prompts (and separators
32 32 # in between)
33 33 default_in_prompt = 'In [<span class="in-prompt-number">%i</span>]: '
34 34 default_out_prompt = 'Out[<span class="out-prompt-number">%i</span>]: '
35 35 default_input_sep = '\n'
36 36 default_output_sep = ''
37 37 default_output_sep2 = ''
38 38
39 39 # Base path for most payload sources.
40 40 zmq_shell_source = 'IPython.zmq.zmqshell.ZMQInteractiveShell'
41 41
42 42 if sys.platform.startswith('win'):
43 43 default_editor = 'notepad'
44 44 else:
45 45 default_editor = ''
46 46
47 47 #-----------------------------------------------------------------------------
48 48 # IPythonWidget class
49 49 #-----------------------------------------------------------------------------
50 50
51 51 class IPythonWidget(FrontendWidget):
52 52 """ A FrontendWidget for an IPython kernel.
53 53 """
54 54
55 55 # If set, the 'custom_edit_requested(str, int)' signal will be emitted when
56 56 # an editor is needed for a file. This overrides 'editor' and 'editor_line'
57 57 # settings.
58 58 custom_edit = Bool(False)
59 59 custom_edit_requested = QtCore.Signal(object, object)
60 60
61 61 editor = Unicode(default_editor, config=True,
62 62 help="""
63 63 A command for invoking a system text editor. If the string contains a
64 64 {filename} format specifier, it will be used. Otherwise, the filename
65 65 will be appended to the end the command.
66 66 """)
67 67
68 68 editor_line = Unicode(config=True,
69 69 help="""
70 70 The editor command to use when a specific line number is requested. The
71 71 string should contain two format specifiers: {line} and {filename}. If
72 72 this parameter is not specified, the line number option to the %edit
73 73 magic will be ignored.
74 74 """)
75 75
76 76 style_sheet = Unicode(config=True,
77 77 help="""
78 78 A CSS stylesheet. The stylesheet can contain classes for:
79 79 1. Qt: QPlainTextEdit, QFrame, QWidget, etc
80 80 2. Pygments: .c, .k, .o, etc. (see PygmentsHighlighter)
81 81 3. IPython: .error, .in-prompt, .out-prompt, etc
82 82 """)
83 83
84 84 syntax_style = Unicode(config=True,
85 85 help="""
86 86 If not empty, use this Pygments style for syntax highlighting.
87 87 Otherwise, the style sheet is queried for Pygments style
88 88 information.
89 89 """)
90 90
91 91 # Prompts.
92 92 in_prompt = Unicode(default_in_prompt, config=True)
93 93 out_prompt = Unicode(default_out_prompt, config=True)
94 94 input_sep = Unicode(default_input_sep, config=True)
95 95 output_sep = Unicode(default_output_sep, config=True)
96 96 output_sep2 = Unicode(default_output_sep2, config=True)
97 97
98 98 # FrontendWidget protected class variables.
99 99 _input_splitter_class = IPythonInputSplitter
100 100
101 101 # IPythonWidget protected class variables.
102 102 _PromptBlock = namedtuple('_PromptBlock', ['block', 'length', 'number'])
103 103 _payload_source_edit = zmq_shell_source + '.edit_magic'
104 104 _payload_source_exit = zmq_shell_source + '.ask_exit'
105 105 _payload_source_next_input = zmq_shell_source + '.set_next_input'
106 106 _payload_source_page = 'IPython.zmq.page.page'
107 107
108 108 #---------------------------------------------------------------------------
109 109 # 'object' interface
110 110 #---------------------------------------------------------------------------
111 111
112 112 def __init__(self, *args, **kw):
113 113 super(IPythonWidget, self).__init__(*args, **kw)
114 114
115 115 # IPythonWidget protected variables.
116 116 self._payload_handlers = {
117 117 self._payload_source_edit : self._handle_payload_edit,
118 118 self._payload_source_exit : self._handle_payload_exit,
119 119 self._payload_source_page : self._handle_payload_page,
120 120 self._payload_source_next_input : self._handle_payload_next_input }
121 121 self._previous_prompt_obj = None
122 122 self._keep_kernel_on_exit = None
123 123
124 124 # Initialize widget styling.
125 125 if self.style_sheet:
126 126 self._style_sheet_changed()
127 127 self._syntax_style_changed()
128 128 else:
129 129 self.set_default_style()
130 130
131 131 #---------------------------------------------------------------------------
132 132 # 'BaseFrontendMixin' abstract interface
133 133 #---------------------------------------------------------------------------
134 134
135 135 def _handle_complete_reply(self, rep):
136 136 """ Reimplemented to support IPython's improved completion machinery.
137 137 """
138 138 cursor = self._get_cursor()
139 139 info = self._request_info.get('complete')
140 140 if info and info.id == rep['parent_header']['msg_id'] and \
141 141 info.pos == cursor.position():
142 142 matches = rep['content']['matches']
143 143 text = rep['content']['matched_text']
144 144 offset = len(text)
145 145
146 146 # Clean up matches with period and path separators if the matched
147 147 # text has not been transformed. This is done by truncating all
148 148 # but the last component and then suitably decreasing the offset
149 149 # between the current cursor position and the start of completion.
150 150 if len(matches) > 1 and matches[0][:offset] == text:
151 151 parts = re.split(r'[./\\]', text)
152 152 sep_count = len(parts) - 1
153 153 if sep_count:
154 154 chop_length = sum(map(len, parts[:sep_count])) + sep_count
155 155 matches = [ match[chop_length:] for match in matches ]
156 156 offset -= chop_length
157 157
158 158 # Move the cursor to the start of the match and complete.
159 159 cursor.movePosition(QtGui.QTextCursor.Left, n=offset)
160 160 self._complete_with_items(cursor, matches)
161 161
162 162 def _handle_execute_reply(self, msg):
163 163 """ Reimplemented to support prompt requests.
164 164 """
165 165 info = self._request_info.get('execute')
166 166 if info and info.id == msg['parent_header']['msg_id']:
167 167 if info.kind == 'prompt':
168 168 number = msg['content']['execution_count'] + 1
169 169 self._show_interpreter_prompt(number)
170 170 else:
171 171 super(IPythonWidget, self)._handle_execute_reply(msg)
172 172
173 173 def _handle_history_reply(self, msg):
174 174 """ Implemented to handle history tail replies, which are only supported
175 175 by the IPython kernel.
176 176 """
177 177 history_items = msg['content']['history']
178 178 items = [ line.rstrip() for _, _, line in history_items ]
179 179 self._set_history(items)
180 180
181 181 def _handle_pyout(self, msg):
182 182 """ Reimplemented for IPython-style "display hook".
183 183 """
184 184 if not self._hidden and self._is_from_this_session(msg):
185 185 content = msg['content']
186 186 prompt_number = content['execution_count']
187 187 data = content['data']
188 188 if data.has_key('text/html'):
189 189 self._append_plain_text(self.output_sep, True)
190 190 self._append_html(self._make_out_prompt(prompt_number), True)
191 191 html = data['text/html']
192 192 self._append_plain_text('\n', True)
193 193 self._append_html(html + self.output_sep2, True)
194 194 elif data.has_key('text/plain'):
195 195 self._append_plain_text(self.output_sep, True)
196 196 self._append_html(self._make_out_prompt(prompt_number), True)
197 197 text = data['text/plain']
198 198 # If the repr is multiline, make sure we start on a new line,
199 199 # so that its lines are aligned.
200 200 if "\n" in text and not self.output_sep.endswith("\n"):
201 201 self._append_plain_text('\n', True)
202 202 self._append_plain_text(text + self.output_sep2, True)
203 203
204 204 def _handle_display_data(self, msg):
205 205 """ The base handler for the ``display_data`` message.
206 206 """
207 207 # For now, we don't display data from other frontends, but we
208 208 # eventually will as this allows all frontends to monitor the display
209 209 # data. But we need to figure out how to handle this in the GUI.
210 210 if not self._hidden and self._is_from_this_session(msg):
211 211 source = msg['content']['source']
212 212 data = msg['content']['data']
213 213 metadata = msg['content']['metadata']
214 214 # In the regular IPythonWidget, we simply print the plain text
215 215 # representation.
216 216 if data.has_key('text/html'):
217 217 html = data['text/html']
218 218 self._append_html(html, True)
219 219 elif data.has_key('text/plain'):
220 220 text = data['text/plain']
221 221 self._append_plain_text(text, True)
222 222 # This newline seems to be needed for text and html output.
223 223 self._append_plain_text(u'\n', True)
224 224
225 225 def _started_channels(self):
226 226 """ Reimplemented to make a history request.
227 227 """
228 228 super(IPythonWidget, self)._started_channels()
229 229 self.kernel_manager.shell_channel.history(hist_access_type='tail',
230 230 n=1000)
231 231 #---------------------------------------------------------------------------
232 232 # 'ConsoleWidget' public interface
233 233 #---------------------------------------------------------------------------
234 234
235 235 def copy(self):
236 236 """ Copy the currently selected text to the clipboard, removing prompts
237 237 if possible.
238 238 """
239 239 text = self._control.textCursor().selection().toPlainText()
240 240 if text:
241 241 lines = map(transform_ipy_prompt, text.splitlines())
242 242 text = '\n'.join(lines)
243 243 QtGui.QApplication.clipboard().setText(text)
244 244
245 245 #---------------------------------------------------------------------------
246 246 # 'FrontendWidget' public interface
247 247 #---------------------------------------------------------------------------
248 248
249 249 def execute_file(self, path, hidden=False):
250 250 """ Reimplemented to use the 'run' magic.
251 251 """
252 252 # Use forward slashes on Windows to avoid escaping each separator.
253 253 if sys.platform == 'win32':
254 254 path = os.path.normpath(path).replace('\\', '/')
255 255
256 # Perhaps we should not be using %run directly, but while we
257 # are, it is necessary to quote filenames containing spaces or quotes.
258 # Escaping quotes in filename in %run seems tricky and inconsistent,
259 # so not trying it at present.
260 if '"' in path:
261 if "'" in path:
262 raise ValueError("Can't run filename containing both single "
263 "and double quotes: %s" % path)
264 path = "'%s'" % path
265 elif ' ' in path or "'" in path:
266 path = '"%s"' % path
267
256 268 self.execute('%%run %s' % path, hidden=hidden)
257 269
258 270 #---------------------------------------------------------------------------
259 271 # 'FrontendWidget' protected interface
260 272 #---------------------------------------------------------------------------
261 273
262 274 def _complete(self):
263 275 """ Reimplemented to support IPython's improved completion machinery.
264 276 """
265 277 # We let the kernel split the input line, so we *always* send an empty
266 278 # text field. Readline-based frontends do get a real text field which
267 279 # they can use.
268 280 text = ''
269 281
270 282 # Send the completion request to the kernel
271 283 msg_id = self.kernel_manager.shell_channel.complete(
272 284 text, # text
273 285 self._get_input_buffer_cursor_line(), # line
274 286 self._get_input_buffer_cursor_column(), # cursor_pos
275 287 self.input_buffer) # block
276 288 pos = self._get_cursor().position()
277 289 info = self._CompletionRequest(msg_id, pos)
278 290 self._request_info['complete'] = info
279 291
280 292 def _process_execute_error(self, msg):
281 293 """ Reimplemented for IPython-style traceback formatting.
282 294 """
283 295 content = msg['content']
284 296 traceback = '\n'.join(content['traceback']) + '\n'
285 297 if False:
286 298 # FIXME: For now, tracebacks come as plain text, so we can't use
287 299 # the html renderer yet. Once we refactor ultratb to produce
288 300 # properly styled tracebacks, this branch should be the default
289 301 traceback = traceback.replace(' ', '&nbsp;')
290 302 traceback = traceback.replace('\n', '<br/>')
291 303
292 304 ename = content['ename']
293 305 ename_styled = '<span class="error">%s</span>' % ename
294 306 traceback = traceback.replace(ename, ename_styled)
295 307
296 308 self._append_html(traceback)
297 309 else:
298 310 # This is the fallback for now, using plain text with ansi escapes
299 311 self._append_plain_text(traceback)
300 312
301 313 def _process_execute_payload(self, item):
302 314 """ Reimplemented to dispatch payloads to handler methods.
303 315 """
304 316 handler = self._payload_handlers.get(item['source'])
305 317 if handler is None:
306 318 # We have no handler for this type of payload, simply ignore it
307 319 return False
308 320 else:
309 321 handler(item)
310 322 return True
311 323
312 324 def _show_interpreter_prompt(self, number=None):
313 325 """ Reimplemented for IPython-style prompts.
314 326 """
315 327 # If a number was not specified, make a prompt number request.
316 328 if number is None:
317 329 msg_id = self.kernel_manager.shell_channel.execute('', silent=True)
318 330 info = self._ExecutionRequest(msg_id, 'prompt')
319 331 self._request_info['execute'] = info
320 332 return
321 333
322 334 # Show a new prompt and save information about it so that it can be
323 335 # updated later if the prompt number turns out to be wrong.
324 336 self._prompt_sep = self.input_sep
325 337 self._show_prompt(self._make_in_prompt(number), html=True)
326 338 block = self._control.document().lastBlock()
327 339 length = len(self._prompt)
328 340 self._previous_prompt_obj = self._PromptBlock(block, length, number)
329 341
330 342 # Update continuation prompt to reflect (possibly) new prompt length.
331 343 self._set_continuation_prompt(
332 344 self._make_continuation_prompt(self._prompt), html=True)
333 345
334 346 def _show_interpreter_prompt_for_reply(self, msg):
335 347 """ Reimplemented for IPython-style prompts.
336 348 """
337 349 # Update the old prompt number if necessary.
338 350 content = msg['content']
339 351 previous_prompt_number = content['execution_count']
340 352 if self._previous_prompt_obj and \
341 353 self._previous_prompt_obj.number != previous_prompt_number:
342 354 block = self._previous_prompt_obj.block
343 355
344 356 # Make sure the prompt block has not been erased.
345 357 if block.isValid() and block.text():
346 358
347 359 # Remove the old prompt and insert a new prompt.
348 360 cursor = QtGui.QTextCursor(block)
349 361 cursor.movePosition(QtGui.QTextCursor.Right,
350 362 QtGui.QTextCursor.KeepAnchor,
351 363 self._previous_prompt_obj.length)
352 364 prompt = self._make_in_prompt(previous_prompt_number)
353 365 self._prompt = self._insert_html_fetching_plain_text(
354 366 cursor, prompt)
355 367
356 368 # When the HTML is inserted, Qt blows away the syntax
357 369 # highlighting for the line, so we need to rehighlight it.
358 370 self._highlighter.rehighlightBlock(cursor.block())
359 371
360 372 self._previous_prompt_obj = None
361 373
362 374 # Show a new prompt with the kernel's estimated prompt number.
363 375 self._show_interpreter_prompt(previous_prompt_number + 1)
364 376
365 377 #---------------------------------------------------------------------------
366 378 # 'IPythonWidget' interface
367 379 #---------------------------------------------------------------------------
368 380
369 381 def set_default_style(self, colors='lightbg'):
370 382 """ Sets the widget style to the class defaults.
371 383
372 384 Parameters:
373 385 -----------
374 386 colors : str, optional (default lightbg)
375 387 Whether to use the default IPython light background or dark
376 388 background or B&W style.
377 389 """
378 390 colors = colors.lower()
379 391 if colors=='lightbg':
380 392 self.style_sheet = styles.default_light_style_sheet
381 393 self.syntax_style = styles.default_light_syntax_style
382 394 elif colors=='linux':
383 395 self.style_sheet = styles.default_dark_style_sheet
384 396 self.syntax_style = styles.default_dark_syntax_style
385 397 elif colors=='nocolor':
386 398 self.style_sheet = styles.default_bw_style_sheet
387 399 self.syntax_style = styles.default_bw_syntax_style
388 400 else:
389 401 raise KeyError("No such color scheme: %s"%colors)
390 402
391 403 #---------------------------------------------------------------------------
392 404 # 'IPythonWidget' protected interface
393 405 #---------------------------------------------------------------------------
394 406
395 407 def _edit(self, filename, line=None):
396 408 """ Opens a Python script for editing.
397 409
398 410 Parameters:
399 411 -----------
400 412 filename : str
401 413 A path to a local system file.
402 414
403 415 line : int, optional
404 416 A line of interest in the file.
405 417 """
406 418 if self.custom_edit:
407 419 self.custom_edit_requested.emit(filename, line)
408 420 elif not self.editor:
409 421 self._append_plain_text('No default editor available.\n'
410 422 'Specify a GUI text editor in the `IPythonWidget.editor` '
411 423 'configurable to enable the %edit magic')
412 424 else:
413 425 try:
414 426 filename = '"%s"' % filename
415 427 if line and self.editor_line:
416 428 command = self.editor_line.format(filename=filename,
417 429 line=line)
418 430 else:
419 431 try:
420 432 command = self.editor.format()
421 433 except KeyError:
422 434 command = self.editor.format(filename=filename)
423 435 else:
424 436 command += ' ' + filename
425 437 except KeyError:
426 438 self._append_plain_text('Invalid editor command.\n')
427 439 else:
428 440 try:
429 441 Popen(command, shell=True)
430 442 except OSError:
431 443 msg = 'Opening editor with command "%s" failed.\n'
432 444 self._append_plain_text(msg % command)
433 445
434 446 def _make_in_prompt(self, number):
435 447 """ Given a prompt number, returns an HTML In prompt.
436 448 """
437 449 try:
438 450 body = self.in_prompt % number
439 451 except TypeError:
440 452 # allow in_prompt to leave out number, e.g. '>>> '
441 453 body = self.in_prompt
442 454 return '<span class="in-prompt">%s</span>' % body
443 455
444 456 def _make_continuation_prompt(self, prompt):
445 457 """ Given a plain text version of an In prompt, returns an HTML
446 458 continuation prompt.
447 459 """
448 460 end_chars = '...: '
449 461 space_count = len(prompt.lstrip('\n')) - len(end_chars)
450 462 body = '&nbsp;' * space_count + end_chars
451 463 return '<span class="in-prompt">%s</span>' % body
452 464
453 465 def _make_out_prompt(self, number):
454 466 """ Given a prompt number, returns an HTML Out prompt.
455 467 """
456 468 body = self.out_prompt % number
457 469 return '<span class="out-prompt">%s</span>' % body
458 470
459 471 #------ Payload handlers --------------------------------------------------
460 472
461 473 # Payload handlers with a generic interface: each takes the opaque payload
462 474 # dict, unpacks it and calls the underlying functions with the necessary
463 475 # arguments.
464 476
465 477 def _handle_payload_edit(self, item):
466 478 self._edit(item['filename'], item['line_number'])
467 479
468 480 def _handle_payload_exit(self, item):
469 481 self._keep_kernel_on_exit = item['keepkernel']
470 482 self.exit_requested.emit()
471 483
472 484 def _handle_payload_next_input(self, item):
473 485 self.input_buffer = dedent(item['text'].rstrip())
474 486
475 487 def _handle_payload_page(self, item):
476 488 # Since the plain text widget supports only a very small subset of HTML
477 489 # and we have no control over the HTML source, we only page HTML
478 490 # payloads in the rich text widget.
479 491 if item['html'] and self.kind == 'rich':
480 492 self._page(item['html'], html=True)
481 493 else:
482 494 self._page(item['text'], html=False)
483 495
484 496 #------ Trait change handlers --------------------------------------------
485 497
486 498 def _style_sheet_changed(self):
487 499 """ Set the style sheets of the underlying widgets.
488 500 """
489 501 self.setStyleSheet(self.style_sheet)
490 502 self._control.document().setDefaultStyleSheet(self.style_sheet)
491 503 if self._page_control:
492 504 self._page_control.document().setDefaultStyleSheet(self.style_sheet)
493 505
494 506 bg_color = self._control.palette().window().color()
495 507 self._ansi_processor.set_background_color(bg_color)
496 508
497 509
498 510 def _syntax_style_changed(self):
499 511 """ Set the style for the syntax highlighter.
500 512 """
501 513 if self._highlighter is None:
502 514 # ignore premature calls
503 515 return
504 516 if self.syntax_style:
505 517 self._highlighter.set_style(self.syntax_style)
506 518 else:
507 519 self._highlighter.set_style_sheet(self.style_sheet)
508 520
509 521 #------ Trait default initializers -----------------------------------------
510 522
511 523 def _banner_default(self):
512 524 from IPython.core.usage import default_gui_banner
513 525 return default_gui_banner
General Comments 0
You need to be logged in to leave comments. Login now