##// END OF EJS Templates
making it PEP8 compliant
Toby Gilham -
Show More
@@ -1,561 +1,561
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 import time
16 16 from textwrap import dedent
17 17
18 18 # System library imports
19 19 from IPython.external.qt import QtCore, QtGui
20 20
21 21 # Local imports
22 22 from IPython.core.inputsplitter import IPythonInputSplitter, \
23 23 transform_ipy_prompt
24 24 from IPython.utils.traitlets import Bool, Unicode
25 25 from frontend_widget import FrontendWidget
26 26 import styles
27 27
28 28 #-----------------------------------------------------------------------------
29 29 # Constants
30 30 #-----------------------------------------------------------------------------
31 31
32 32 # Default strings to build and display input and output prompts (and separators
33 33 # in between)
34 34 default_in_prompt = 'In [<span class="in-prompt-number">%i</span>]: '
35 35 default_out_prompt = 'Out[<span class="out-prompt-number">%i</span>]: '
36 36 default_input_sep = '\n'
37 37 default_output_sep = ''
38 38 default_output_sep2 = ''
39 39
40 40 # Base path for most payload sources.
41 41 zmq_shell_source = 'IPython.zmq.zmqshell.ZMQInteractiveShell'
42 42
43 43 if sys.platform.startswith('win'):
44 44 default_editor = 'notepad'
45 45 else:
46 46 default_editor = ''
47 47
48 48 #-----------------------------------------------------------------------------
49 49 # IPythonWidget class
50 50 #-----------------------------------------------------------------------------
51 51
52 52 class IPythonWidget(FrontendWidget):
53 53 """ A FrontendWidget for an IPython kernel.
54 54 """
55 55
56 56 # If set, the 'custom_edit_requested(str, int)' signal will be emitted when
57 57 # an editor is needed for a file. This overrides 'editor' and 'editor_line'
58 58 # settings.
59 59 custom_edit = Bool(False)
60 60 custom_edit_requested = QtCore.Signal(object, object)
61 61
62 62 editor = Unicode(default_editor, config=True,
63 63 help="""
64 64 A command for invoking a system text editor. If the string contains a
65 65 {filename} format specifier, it will be used. Otherwise, the filename
66 66 will be appended to the end the command.
67 67 """)
68 68
69 69 editor_line = Unicode(config=True,
70 70 help="""
71 71 The editor command to use when a specific line number is requested. The
72 72 string should contain two format specifiers: {line} and {filename}. If
73 73 this parameter is not specified, the line number option to the %edit
74 74 magic will be ignored.
75 75 """)
76 76
77 77 style_sheet = Unicode(config=True,
78 78 help="""
79 79 A CSS stylesheet. The stylesheet can contain classes for:
80 80 1. Qt: QPlainTextEdit, QFrame, QWidget, etc
81 81 2. Pygments: .c, .k, .o, etc. (see PygmentsHighlighter)
82 82 3. IPython: .error, .in-prompt, .out-prompt, etc
83 83 """)
84 84
85 85 syntax_style = Unicode(config=True,
86 86 help="""
87 87 If not empty, use this Pygments style for syntax highlighting.
88 88 Otherwise, the style sheet is queried for Pygments style
89 89 information.
90 90 """)
91 91
92 92 # Prompts.
93 93 in_prompt = Unicode(default_in_prompt, config=True)
94 94 out_prompt = Unicode(default_out_prompt, config=True)
95 95 input_sep = Unicode(default_input_sep, config=True)
96 96 output_sep = Unicode(default_output_sep, config=True)
97 97 output_sep2 = Unicode(default_output_sep2, config=True)
98 98
99 99 # FrontendWidget protected class variables.
100 100 _input_splitter_class = IPythonInputSplitter
101 101 _transform_prompt = staticmethod(transform_ipy_prompt)
102 102
103 103 # IPythonWidget protected class variables.
104 104 _PromptBlock = namedtuple('_PromptBlock', ['block', 'length', 'number'])
105 105 _payload_source_edit = zmq_shell_source + '.edit_magic'
106 106 _payload_source_exit = zmq_shell_source + '.ask_exit'
107 107 _payload_source_next_input = zmq_shell_source + '.set_next_input'
108 108 _payload_source_page = 'IPython.zmq.page.page'
109 109 _retrying_history_request = False
110 110
111 111 #---------------------------------------------------------------------------
112 112 # 'object' interface
113 113 #---------------------------------------------------------------------------
114 114
115 115 def __init__(self, *args, **kw):
116 116 super(IPythonWidget, self).__init__(*args, **kw)
117 117
118 118 # IPythonWidget protected variables.
119 119 self._payload_handlers = {
120 120 self._payload_source_edit : self._handle_payload_edit,
121 121 self._payload_source_exit : self._handle_payload_exit,
122 122 self._payload_source_page : self._handle_payload_page,
123 123 self._payload_source_next_input : self._handle_payload_next_input }
124 124 self._previous_prompt_obj = None
125 125 self._keep_kernel_on_exit = None
126 126
127 127 # Initialize widget styling.
128 128 if self.style_sheet:
129 129 self._style_sheet_changed()
130 130 self._syntax_style_changed()
131 131 else:
132 132 self.set_default_style()
133 133
134 134 #---------------------------------------------------------------------------
135 135 # 'BaseFrontendMixin' abstract interface
136 136 #---------------------------------------------------------------------------
137 137
138 138 def _handle_complete_reply(self, rep):
139 139 """ Reimplemented to support IPython's improved completion machinery.
140 140 """
141 141 self.log.debug("complete: %s", rep.get('content', ''))
142 142 cursor = self._get_cursor()
143 143 info = self._request_info.get('complete')
144 144 if info and info.id == rep['parent_header']['msg_id'] and \
145 145 info.pos == cursor.position():
146 146 matches = rep['content']['matches']
147 147 text = rep['content']['matched_text']
148 148 offset = len(text)
149 149
150 150 # Clean up matches with period and path separators if the matched
151 151 # text has not been transformed. This is done by truncating all
152 152 # but the last component and then suitably decreasing the offset
153 153 # between the current cursor position and the start of completion.
154 154 if len(matches) > 1 and matches[0][:offset] == text:
155 155 parts = re.split(r'[./\\]', text)
156 156 sep_count = len(parts) - 1
157 157 if sep_count:
158 158 chop_length = sum(map(len, parts[:sep_count])) + sep_count
159 159 matches = [ match[chop_length:] for match in matches ]
160 160 offset -= chop_length
161 161
162 162 # Move the cursor to the start of the match and complete.
163 163 cursor.movePosition(QtGui.QTextCursor.Left, n=offset)
164 164 self._complete_with_items(cursor, matches)
165 165
166 166 def _handle_execute_reply(self, msg):
167 167 """ Reimplemented to support prompt requests.
168 168 """
169 169 msg_id = msg['parent_header'].get('msg_id')
170 170 info = self._request_info['execute'].get(msg_id)
171 171 if info and info.kind == 'prompt':
172 172 number = msg['content']['execution_count'] + 1
173 173 self._show_interpreter_prompt(number)
174 174 self._request_info['execute'].pop(msg_id)
175 175 else:
176 176 super(IPythonWidget, self)._handle_execute_reply(msg)
177 177
178 178 def _handle_history_reply(self, msg):
179 179 """ Implemented to handle history tail replies, which are only supported
180 180 by the IPython kernel.
181 181 """
182 182 content = msg['content']
183 183 if 'history' not in content:
184 184 self.log.error("History request failed: %r"%content)
185 185 if content.get('status', '') == 'aborted' and \
186 186 not self._retrying_history_request:
187 187 # a *different* action caused this request to be aborted, so
188 188 # we should try again.
189 189 self.log.error("Retrying aborted history request")
190 190 # prevent multiple retries of aborted requests:
191 191 self._retrying_history_request = True
192 192 # wait out the kernel's queue flush, which is currently timed at 0.1s
193 193 time.sleep(0.25)
194 194 self.kernel_manager.shell_channel.history(hist_access_type='tail',n=1000)
195 195 else:
196 196 self._retrying_history_request = False
197 197 return
198 198 # reset retry flag
199 199 self._retrying_history_request = False
200 200 history_items = content['history']
201 201 self.log.debug("Received history reply with %i entries", len(history_items))
202 202 items = []
203 203 last_cell = u""
204 204 for _, _, cell in history_items:
205 205 cell = cell.rstrip()
206 206 if cell != last_cell:
207 207 items.append(cell)
208 208 last_cell = cell
209 209 self._set_history(items)
210 210
211 211 def _handle_pyout(self, msg):
212 212 """ Reimplemented for IPython-style "display hook".
213 213 """
214 214 self.log.debug("pyout: %s", msg.get('content', ''))
215 215 if not self._hidden and self._is_from_this_session(msg):
216 216 content = msg['content']
217 prompt_number = content.get('execution_count',0)
217 prompt_number = content.get('execution_count', 0)
218 218 data = content['data']
219 219 if data.has_key('text/html'):
220 220 self._append_plain_text(self.output_sep, True)
221 221 self._append_html(self._make_out_prompt(prompt_number), True)
222 222 html = data['text/html']
223 223 self._append_plain_text('\n', True)
224 224 self._append_html(html + self.output_sep2, True)
225 225 elif data.has_key('text/plain'):
226 226 self._append_plain_text(self.output_sep, True)
227 227 self._append_html(self._make_out_prompt(prompt_number), True)
228 228 text = data['text/plain']
229 229 # If the repr is multiline, make sure we start on a new line,
230 230 # so that its lines are aligned.
231 231 if "\n" in text and not self.output_sep.endswith("\n"):
232 232 self._append_plain_text('\n', True)
233 233 self._append_plain_text(text + self.output_sep2, True)
234 234
235 235 def _handle_display_data(self, msg):
236 236 """ The base handler for the ``display_data`` message.
237 237 """
238 238 self.log.debug("display: %s", msg.get('content', ''))
239 239 # For now, we don't display data from other frontends, but we
240 240 # eventually will as this allows all frontends to monitor the display
241 241 # data. But we need to figure out how to handle this in the GUI.
242 242 if not self._hidden and self._is_from_this_session(msg):
243 243 source = msg['content']['source']
244 244 data = msg['content']['data']
245 245 metadata = msg['content']['metadata']
246 246 # In the regular IPythonWidget, we simply print the plain text
247 247 # representation.
248 248 if data.has_key('text/html'):
249 249 html = data['text/html']
250 250 self._append_html(html, True)
251 251 elif data.has_key('text/plain'):
252 252 text = data['text/plain']
253 253 self._append_plain_text(text, True)
254 254 # This newline seems to be needed for text and html output.
255 255 self._append_plain_text(u'\n', True)
256 256
257 257 def _started_channels(self):
258 258 """ Reimplemented to make a history request.
259 259 """
260 260 super(IPythonWidget, self)._started_channels()
261 261 self.kernel_manager.shell_channel.history(hist_access_type='tail',
262 262 n=1000)
263 263 #---------------------------------------------------------------------------
264 264 # 'ConsoleWidget' public interface
265 265 #---------------------------------------------------------------------------
266 266
267 267 #---------------------------------------------------------------------------
268 268 # 'FrontendWidget' public interface
269 269 #---------------------------------------------------------------------------
270 270
271 271 def execute_file(self, path, hidden=False):
272 272 """ Reimplemented to use the 'run' magic.
273 273 """
274 274 # Use forward slashes on Windows to avoid escaping each separator.
275 275 if sys.platform == 'win32':
276 276 path = os.path.normpath(path).replace('\\', '/')
277 277
278 278 # Perhaps we should not be using %run directly, but while we
279 279 # are, it is necessary to quote or escape filenames containing spaces
280 280 # or quotes.
281 281
282 282 # In earlier code here, to minimize escaping, we sometimes quoted the
283 283 # filename with single quotes. But to do this, this code must be
284 284 # platform-aware, because run uses shlex rather than python string
285 285 # parsing, so that:
286 286 # * In Win: single quotes can be used in the filename without quoting,
287 287 # and we cannot use single quotes to quote the filename.
288 288 # * In *nix: we can escape double quotes in a double quoted filename,
289 289 # but can't escape single quotes in a single quoted filename.
290 290
291 291 # So to keep this code non-platform-specific and simple, we now only
292 292 # use double quotes to quote filenames, and escape when needed:
293 293 if ' ' in path or "'" in path or '"' in path:
294 294 path = '"%s"' % path.replace('"', '\\"')
295 295 self.execute('%%run %s' % path, hidden=hidden)
296 296
297 297 #---------------------------------------------------------------------------
298 298 # 'FrontendWidget' protected interface
299 299 #---------------------------------------------------------------------------
300 300
301 301 def _complete(self):
302 302 """ Reimplemented to support IPython's improved completion machinery.
303 303 """
304 304 # We let the kernel split the input line, so we *always* send an empty
305 305 # text field. Readline-based frontends do get a real text field which
306 306 # they can use.
307 307 text = ''
308 308
309 309 # Send the completion request to the kernel
310 310 msg_id = self.kernel_manager.shell_channel.complete(
311 311 text, # text
312 312 self._get_input_buffer_cursor_line(), # line
313 313 self._get_input_buffer_cursor_column(), # cursor_pos
314 314 self.input_buffer) # block
315 315 pos = self._get_cursor().position()
316 316 info = self._CompletionRequest(msg_id, pos)
317 317 self._request_info['complete'] = info
318 318
319 319 def _process_execute_error(self, msg):
320 320 """ Reimplemented for IPython-style traceback formatting.
321 321 """
322 322 content = msg['content']
323 323 traceback = '\n'.join(content['traceback']) + '\n'
324 324 if False:
325 325 # FIXME: For now, tracebacks come as plain text, so we can't use
326 326 # the html renderer yet. Once we refactor ultratb to produce
327 327 # properly styled tracebacks, this branch should be the default
328 328 traceback = traceback.replace(' ', '&nbsp;')
329 329 traceback = traceback.replace('\n', '<br/>')
330 330
331 331 ename = content['ename']
332 332 ename_styled = '<span class="error">%s</span>' % ename
333 333 traceback = traceback.replace(ename, ename_styled)
334 334
335 335 self._append_html(traceback)
336 336 else:
337 337 # This is the fallback for now, using plain text with ansi escapes
338 338 self._append_plain_text(traceback)
339 339
340 340 def _process_execute_payload(self, item):
341 341 """ Reimplemented to dispatch payloads to handler methods.
342 342 """
343 343 handler = self._payload_handlers.get(item['source'])
344 344 if handler is None:
345 345 # We have no handler for this type of payload, simply ignore it
346 346 return False
347 347 else:
348 348 handler(item)
349 349 return True
350 350
351 351 def _show_interpreter_prompt(self, number=None):
352 352 """ Reimplemented for IPython-style prompts.
353 353 """
354 354 # If a number was not specified, make a prompt number request.
355 355 if number is None:
356 356 msg_id = self.kernel_manager.shell_channel.execute('', silent=True)
357 357 info = self._ExecutionRequest(msg_id, 'prompt')
358 358 self._request_info['execute'][msg_id] = info
359 359 return
360 360
361 361 # Show a new prompt and save information about it so that it can be
362 362 # updated later if the prompt number turns out to be wrong.
363 363 self._prompt_sep = self.input_sep
364 364 self._show_prompt(self._make_in_prompt(number), html=True)
365 365 block = self._control.document().lastBlock()
366 366 length = len(self._prompt)
367 367 self._previous_prompt_obj = self._PromptBlock(block, length, number)
368 368
369 369 # Update continuation prompt to reflect (possibly) new prompt length.
370 370 self._set_continuation_prompt(
371 371 self._make_continuation_prompt(self._prompt), html=True)
372 372
373 373 def _show_interpreter_prompt_for_reply(self, msg):
374 374 """ Reimplemented for IPython-style prompts.
375 375 """
376 376 # Update the old prompt number if necessary.
377 377 content = msg['content']
378 378 # abort replies do not have any keys:
379 379 if content['status'] == 'aborted':
380 380 if self._previous_prompt_obj:
381 381 previous_prompt_number = self._previous_prompt_obj.number
382 382 else:
383 383 previous_prompt_number = 0
384 384 else:
385 385 previous_prompt_number = content['execution_count']
386 386 if self._previous_prompt_obj and \
387 387 self._previous_prompt_obj.number != previous_prompt_number:
388 388 block = self._previous_prompt_obj.block
389 389
390 390 # Make sure the prompt block has not been erased.
391 391 if block.isValid() and block.text():
392 392
393 393 # Remove the old prompt and insert a new prompt.
394 394 cursor = QtGui.QTextCursor(block)
395 395 cursor.movePosition(QtGui.QTextCursor.Right,
396 396 QtGui.QTextCursor.KeepAnchor,
397 397 self._previous_prompt_obj.length)
398 398 prompt = self._make_in_prompt(previous_prompt_number)
399 399 self._prompt = self._insert_html_fetching_plain_text(
400 400 cursor, prompt)
401 401
402 402 # When the HTML is inserted, Qt blows away the syntax
403 403 # highlighting for the line, so we need to rehighlight it.
404 404 self._highlighter.rehighlightBlock(cursor.block())
405 405
406 406 self._previous_prompt_obj = None
407 407
408 408 # Show a new prompt with the kernel's estimated prompt number.
409 409 self._show_interpreter_prompt(previous_prompt_number + 1)
410 410
411 411 #---------------------------------------------------------------------------
412 412 # 'IPythonWidget' interface
413 413 #---------------------------------------------------------------------------
414 414
415 415 def set_default_style(self, colors='lightbg'):
416 416 """ Sets the widget style to the class defaults.
417 417
418 418 Parameters:
419 419 -----------
420 420 colors : str, optional (default lightbg)
421 421 Whether to use the default IPython light background or dark
422 422 background or B&W style.
423 423 """
424 424 colors = colors.lower()
425 425 if colors=='lightbg':
426 426 self.style_sheet = styles.default_light_style_sheet
427 427 self.syntax_style = styles.default_light_syntax_style
428 428 elif colors=='linux':
429 429 self.style_sheet = styles.default_dark_style_sheet
430 430 self.syntax_style = styles.default_dark_syntax_style
431 431 elif colors=='nocolor':
432 432 self.style_sheet = styles.default_bw_style_sheet
433 433 self.syntax_style = styles.default_bw_syntax_style
434 434 else:
435 435 raise KeyError("No such color scheme: %s"%colors)
436 436
437 437 #---------------------------------------------------------------------------
438 438 # 'IPythonWidget' protected interface
439 439 #---------------------------------------------------------------------------
440 440
441 441 def _edit(self, filename, line=None):
442 442 """ Opens a Python script for editing.
443 443
444 444 Parameters:
445 445 -----------
446 446 filename : str
447 447 A path to a local system file.
448 448
449 449 line : int, optional
450 450 A line of interest in the file.
451 451 """
452 452 if self.custom_edit:
453 453 self.custom_edit_requested.emit(filename, line)
454 454 elif not self.editor:
455 455 self._append_plain_text('No default editor available.\n'
456 456 'Specify a GUI text editor in the `IPythonWidget.editor` '
457 457 'configurable to enable the %edit magic')
458 458 else:
459 459 try:
460 460 filename = '"%s"' % filename
461 461 if line and self.editor_line:
462 462 command = self.editor_line.format(filename=filename,
463 463 line=line)
464 464 else:
465 465 try:
466 466 command = self.editor.format()
467 467 except KeyError:
468 468 command = self.editor.format(filename=filename)
469 469 else:
470 470 command += ' ' + filename
471 471 except KeyError:
472 472 self._append_plain_text('Invalid editor command.\n')
473 473 else:
474 474 try:
475 475 Popen(command, shell=True)
476 476 except OSError:
477 477 msg = 'Opening editor with command "%s" failed.\n'
478 478 self._append_plain_text(msg % command)
479 479
480 480 def _make_in_prompt(self, number):
481 481 """ Given a prompt number, returns an HTML In prompt.
482 482 """
483 483 try:
484 484 body = self.in_prompt % number
485 485 except TypeError:
486 486 # allow in_prompt to leave out number, e.g. '>>> '
487 487 body = self.in_prompt
488 488 return '<span class="in-prompt">%s</span>' % body
489 489
490 490 def _make_continuation_prompt(self, prompt):
491 491 """ Given a plain text version of an In prompt, returns an HTML
492 492 continuation prompt.
493 493 """
494 494 end_chars = '...: '
495 495 space_count = len(prompt.lstrip('\n')) - len(end_chars)
496 496 body = '&nbsp;' * space_count + end_chars
497 497 return '<span class="in-prompt">%s</span>' % body
498 498
499 499 def _make_out_prompt(self, number):
500 500 """ Given a prompt number, returns an HTML Out prompt.
501 501 """
502 502 body = self.out_prompt % number
503 503 return '<span class="out-prompt">%s</span>' % body
504 504
505 505 #------ Payload handlers --------------------------------------------------
506 506
507 507 # Payload handlers with a generic interface: each takes the opaque payload
508 508 # dict, unpacks it and calls the underlying functions with the necessary
509 509 # arguments.
510 510
511 511 def _handle_payload_edit(self, item):
512 512 self._edit(item['filename'], item['line_number'])
513 513
514 514 def _handle_payload_exit(self, item):
515 515 self._keep_kernel_on_exit = item['keepkernel']
516 516 self.exit_requested.emit(self)
517 517
518 518 def _handle_payload_next_input(self, item):
519 519 self.input_buffer = dedent(item['text'].rstrip())
520 520
521 521 def _handle_payload_page(self, item):
522 522 # Since the plain text widget supports only a very small subset of HTML
523 523 # and we have no control over the HTML source, we only page HTML
524 524 # payloads in the rich text widget.
525 525 if item['html'] and self.kind == 'rich':
526 526 self._page(item['html'], html=True)
527 527 else:
528 528 self._page(item['text'], html=False)
529 529
530 530 #------ Trait change handlers --------------------------------------------
531 531
532 532 def _style_sheet_changed(self):
533 533 """ Set the style sheets of the underlying widgets.
534 534 """
535 535 self.setStyleSheet(self.style_sheet)
536 536 if self._control is not None:
537 537 self._control.document().setDefaultStyleSheet(self.style_sheet)
538 538 bg_color = self._control.palette().window().color()
539 539 self._ansi_processor.set_background_color(bg_color)
540 540
541 541 if self._page_control is not None:
542 542 self._page_control.document().setDefaultStyleSheet(self.style_sheet)
543 543
544 544
545 545
546 546 def _syntax_style_changed(self):
547 547 """ Set the style for the syntax highlighter.
548 548 """
549 549 if self._highlighter is None:
550 550 # ignore premature calls
551 551 return
552 552 if self.syntax_style:
553 553 self._highlighter.set_style(self.syntax_style)
554 554 else:
555 555 self._highlighter.set_style_sheet(self.style_sheet)
556 556
557 557 #------ Trait default initializers -----------------------------------------
558 558
559 559 def _banner_default(self):
560 560 from IPython.core.usage import default_gui_banner
561 561 return default_gui_banner
@@ -1,256 +1,256
1 1 # Standard libary imports.
2 2 from base64 import decodestring
3 3 import os
4 4 import re
5 5
6 6 # System libary imports.
7 7 from IPython.external.qt import QtCore, QtGui
8 8
9 9 # Local imports
10 10 from IPython.frontend.qt.svg import save_svg, svg_to_clipboard, svg_to_image
11 11 from ipython_widget import IPythonWidget
12 12
13 13
14 14 class RichIPythonWidget(IPythonWidget):
15 15 """ An IPythonWidget that supports rich text, including lists, images, and
16 16 tables. Note that raw performance will be reduced compared to the plain
17 17 text version.
18 18 """
19 19
20 20 # RichIPythonWidget protected class variables.
21 21 _payload_source_plot = 'IPython.zmq.pylab.backend_payload.add_plot_payload'
22 22
23 23 #---------------------------------------------------------------------------
24 24 # 'object' interface
25 25 #---------------------------------------------------------------------------
26 26
27 27 def __init__(self, *args, **kw):
28 28 """ Create a RichIPythonWidget.
29 29 """
30 30 kw['kind'] = 'rich'
31 31 super(RichIPythonWidget, self).__init__(*args, **kw)
32 32
33 33 # Configure the ConsoleWidget HTML exporter for our formats.
34 34 self._html_exporter.image_tag = self._get_image_tag
35 35
36 36 # Dictionary for resolving document resource names to SVG data.
37 37 self._name_to_svg_map = {}
38 38
39 39 #---------------------------------------------------------------------------
40 40 # 'ConsoleWidget' protected interface
41 41 #---------------------------------------------------------------------------
42 42
43 43 def _context_menu_make(self, pos):
44 44 """ Reimplemented to return a custom context menu for images.
45 45 """
46 46 format = self._control.cursorForPosition(pos).charFormat()
47 47 name = format.stringProperty(QtGui.QTextFormat.ImageName)
48 48 if name:
49 49 menu = QtGui.QMenu()
50 50
51 51 menu.addAction('Copy Image', lambda: self._copy_image(name))
52 52 menu.addAction('Save Image As...', lambda: self._save_image(name))
53 53 menu.addSeparator()
54 54
55 55 svg = self._name_to_svg_map.get(name, None)
56 56 if svg is not None:
57 57 menu.addSeparator()
58 58 menu.addAction('Copy SVG', lambda: svg_to_clipboard(svg))
59 59 menu.addAction('Save SVG As...',
60 60 lambda: save_svg(svg, self._control))
61 61 else:
62 62 menu = super(RichIPythonWidget, self)._context_menu_make(pos)
63 63 return menu
64 64
65 65 #---------------------------------------------------------------------------
66 66 # 'BaseFrontendMixin' abstract interface
67 67 #---------------------------------------------------------------------------
68 68
69 69 def _handle_pyout(self, msg):
70 70 """ Overridden to handle rich data types, like SVG.
71 71 """
72 72 if not self._hidden and self._is_from_this_session(msg):
73 73 content = msg['content']
74 prompt_number = content.get('execution_count',0)
74 prompt_number = content.get('execution_count', 0)
75 75 data = content['data']
76 76 if data.has_key('image/svg+xml'):
77 77 self.log.debug("pyout: %s", msg.get('content', ''))
78 78 self._append_plain_text(self.output_sep, True)
79 79 self._append_html(self._make_out_prompt(prompt_number), True)
80 80 self._append_svg(data['image/svg+xml'], True)
81 81 self._append_html(self.output_sep2, True)
82 82 elif data.has_key('image/png'):
83 83 self.log.debug("pyout: %s", msg.get('content', ''))
84 84 self._append_plain_text(self.output_sep, True)
85 85 self._append_html(self._make_out_prompt(prompt_number), True)
86 86 # This helps the output to look nice.
87 87 self._append_plain_text('\n', True)
88 88 self._append_png(decodestring(data['image/png'].encode('ascii')), True)
89 89 self._append_html(self.output_sep2, True)
90 90 else:
91 91 # Default back to the plain text representation.
92 92 return super(RichIPythonWidget, self)._handle_pyout(msg)
93 93
94 94 def _handle_display_data(self, msg):
95 95 """ Overridden to handle rich data types, like SVG.
96 96 """
97 97 if not self._hidden and self._is_from_this_session(msg):
98 98 source = msg['content']['source']
99 99 data = msg['content']['data']
100 100 metadata = msg['content']['metadata']
101 101 # Try to use the svg or html representations.
102 102 # FIXME: Is this the right ordering of things to try?
103 103 if data.has_key('image/svg+xml'):
104 104 self.log.debug("display: %s", msg.get('content', ''))
105 105 svg = data['image/svg+xml']
106 106 self._append_svg(svg, True)
107 107 elif data.has_key('image/png'):
108 108 self.log.debug("display: %s", msg.get('content', ''))
109 109 # PNG data is base64 encoded as it passes over the network
110 110 # in a JSON structure so we decode it.
111 111 png = decodestring(data['image/png'].encode('ascii'))
112 112 self._append_png(png, True)
113 113 else:
114 114 # Default back to the plain text representation.
115 115 return super(RichIPythonWidget, self)._handle_display_data(msg)
116 116
117 117 #---------------------------------------------------------------------------
118 118 # 'RichIPythonWidget' protected interface
119 119 #---------------------------------------------------------------------------
120 120
121 121 def _append_png(self, png, before_prompt=False):
122 122 """ Append raw PNG data to the widget.
123 123 """
124 124 self._append_custom(self._insert_png, png, before_prompt)
125 125
126 126 def _append_svg(self, svg, before_prompt=False):
127 127 """ Append raw SVG data to the widget.
128 128 """
129 129 self._append_custom(self._insert_svg, svg, before_prompt)
130 130
131 131 def _add_image(self, image):
132 132 """ Adds the specified QImage to the document and returns a
133 133 QTextImageFormat that references it.
134 134 """
135 135 document = self._control.document()
136 136 name = str(image.cacheKey())
137 137 document.addResource(QtGui.QTextDocument.ImageResource,
138 138 QtCore.QUrl(name), image)
139 139 format = QtGui.QTextImageFormat()
140 140 format.setName(name)
141 141 return format
142 142
143 143 def _copy_image(self, name):
144 144 """ Copies the ImageResource with 'name' to the clipboard.
145 145 """
146 146 image = self._get_image(name)
147 147 QtGui.QApplication.clipboard().setImage(image)
148 148
149 149 def _get_image(self, name):
150 150 """ Returns the QImage stored as the ImageResource with 'name'.
151 151 """
152 152 document = self._control.document()
153 153 image = document.resource(QtGui.QTextDocument.ImageResource,
154 154 QtCore.QUrl(name))
155 155 return image
156 156
157 157 def _get_image_tag(self, match, path = None, format = "png"):
158 158 """ Return (X)HTML mark-up for the image-tag given by match.
159 159
160 160 Parameters
161 161 ----------
162 162 match : re.SRE_Match
163 163 A match to an HTML image tag as exported by Qt, with
164 164 match.group("Name") containing the matched image ID.
165 165
166 166 path : string|None, optional [default None]
167 167 If not None, specifies a path to which supporting files may be
168 168 written (e.g., for linked images). If None, all images are to be
169 169 included inline.
170 170
171 171 format : "png"|"svg", optional [default "png"]
172 172 Format for returned or referenced images.
173 173 """
174 174 if format == "png":
175 175 try:
176 176 image = self._get_image(match.group("name"))
177 177 except KeyError:
178 178 return "<b>Couldn't find image %s</b>" % match.group("name")
179 179
180 180 if path is not None:
181 181 if not os.path.exists(path):
182 182 os.mkdir(path)
183 183 relpath = os.path.basename(path)
184 184 if image.save("%s/qt_img%s.png" % (path,match.group("name")),
185 185 "PNG"):
186 186 return '<img src="%s/qt_img%s.png">' % (relpath,
187 187 match.group("name"))
188 188 else:
189 189 return "<b>Couldn't save image!</b>"
190 190 else:
191 191 ba = QtCore.QByteArray()
192 192 buffer_ = QtCore.QBuffer(ba)
193 193 buffer_.open(QtCore.QIODevice.WriteOnly)
194 194 image.save(buffer_, "PNG")
195 195 buffer_.close()
196 196 return '<img src="data:image/png;base64,\n%s\n" />' % (
197 197 re.sub(r'(.{60})',r'\1\n',str(ba.toBase64())))
198 198
199 199 elif format == "svg":
200 200 try:
201 201 svg = str(self._name_to_svg_map[match.group("name")])
202 202 except KeyError:
203 203 return "<b>Couldn't find image %s</b>" % match.group("name")
204 204
205 205 # Not currently checking path, because it's tricky to find a
206 206 # cross-browser way to embed external SVG images (e.g., via
207 207 # object or embed tags).
208 208
209 209 # Chop stand-alone header from matplotlib SVG
210 210 offset = svg.find("<svg")
211 211 assert(offset > -1)
212 212
213 213 return svg[offset:]
214 214
215 215 else:
216 216 return '<b>Unrecognized image format</b>'
217 217
218 218 def _insert_png(self, cursor, png):
219 219 """ Insert raw PNG data into the widget.
220 220 """
221 221 try:
222 222 image = QtGui.QImage()
223 223 image.loadFromData(png, 'PNG')
224 224 except ValueError:
225 225 self._insert_plain_text(cursor, 'Received invalid PNG data.')
226 226 else:
227 227 format = self._add_image(image)
228 228 cursor.insertBlock()
229 229 cursor.insertImage(format)
230 230 cursor.insertBlock()
231 231
232 232 def _insert_svg(self, cursor, svg):
233 233 """ Insert raw SVG data into the widet.
234 234 """
235 235 try:
236 236 image = svg_to_image(svg)
237 237 except ValueError:
238 238 self._insert_plain_text(cursor, 'Received invalid SVG data.')
239 239 else:
240 240 format = self._add_image(image)
241 241 self._name_to_svg_map[format.name()] = svg
242 242 cursor.insertBlock()
243 243 cursor.insertImage(format)
244 244 cursor.insertBlock()
245 245
246 246 def _save_image(self, name, format='PNG'):
247 247 """ Shows a save dialog for the ImageResource with 'name'.
248 248 """
249 249 dialog = QtGui.QFileDialog(self._control, 'Save Image')
250 250 dialog.setAcceptMode(QtGui.QFileDialog.AcceptSave)
251 251 dialog.setDefaultSuffix(format.lower())
252 252 dialog.setNameFilter('%s file (*.%s)' % (format, format.lower()))
253 253 if dialog.exec_():
254 254 filename = dialog.selectedFiles()[0]
255 255 image = self._get_image(name)
256 256 image.save(filename, format)
General Comments 0
You need to be logged in to leave comments. Login now