##// END OF EJS Templates
Refining the matplotlib APIs in IPython.display.
Brian E. Granger -
Show More
@@ -1,736 +1,771 b''
1 1 # -*- coding: utf-8 -*-
2 2 """Top-level display functions for displaying object in different formats.
3 3
4 4 Authors:
5 5
6 6 * Brian Granger
7 7 """
8 8
9 9 #-----------------------------------------------------------------------------
10 10 # Copyright (C) 2013 The IPython Development Team
11 11 #
12 12 # Distributed under the terms of the BSD License. The full license is in
13 13 # the file COPYING, distributed as part of this software.
14 14 #-----------------------------------------------------------------------------
15 15
16 16 #-----------------------------------------------------------------------------
17 17 # Imports
18 18 #-----------------------------------------------------------------------------
19 19
20 20 from __future__ import print_function
21 21
22 22 import os
23 23 import struct
24 24
25 25 from IPython.utils.py3compat import (string_types, cast_bytes_py2, cast_unicode,
26 26 unicode_type)
27
27 from IPython.testing.skipdoctest import skip_doctest
28 28 from .displaypub import publish_display_data
29 29
30 30 #-----------------------------------------------------------------------------
31 31 # utility functions
32 32 #-----------------------------------------------------------------------------
33 33
34 34 def _safe_exists(path):
35 35 """Check path, but don't let exceptions raise"""
36 36 try:
37 37 return os.path.exists(path)
38 38 except Exception:
39 39 return False
40 40
41 41 def _merge(d1, d2):
42 42 """Like update, but merges sub-dicts instead of clobbering at the top level.
43 43
44 44 Updates d1 in-place
45 45 """
46 46
47 47 if not isinstance(d2, dict) or not isinstance(d1, dict):
48 48 return d2
49 49 for key, value in d2.items():
50 50 d1[key] = _merge(d1.get(key), value)
51 51 return d1
52 52
53 53 def _display_mimetype(mimetype, objs, raw=False, metadata=None):
54 54 """internal implementation of all display_foo methods
55 55
56 56 Parameters
57 57 ----------
58 58 mimetype : str
59 59 The mimetype to be published (e.g. 'image/png')
60 60 objs : tuple of objects
61 61 The Python objects to display, or if raw=True raw text data to
62 62 display.
63 63 raw : bool
64 64 Are the data objects raw data or Python objects that need to be
65 65 formatted before display? [default: False]
66 66 metadata : dict (optional)
67 67 Metadata to be associated with the specific mimetype output.
68 68 """
69 69 if metadata:
70 70 metadata = {mimetype: metadata}
71 71 if raw:
72 72 # turn list of pngdata into list of { 'image/png': pngdata }
73 73 objs = [ {mimetype: obj} for obj in objs ]
74 74 display(*objs, raw=raw, metadata=metadata, include=[mimetype])
75 75
76 76 #-----------------------------------------------------------------------------
77 77 # Main functions
78 78 #-----------------------------------------------------------------------------
79 79
80 80 def display(*objs, **kwargs):
81 81 """Display a Python object in all frontends.
82 82
83 83 By default all representations will be computed and sent to the frontends.
84 84 Frontends can decide which representation is used and how.
85 85
86 86 Parameters
87 87 ----------
88 88 objs : tuple of objects
89 89 The Python objects to display.
90 90 raw : bool, optional
91 91 Are the objects to be displayed already mimetype-keyed dicts of raw display data,
92 92 or Python objects that need to be formatted before display? [default: False]
93 93 include : list or tuple, optional
94 94 A list of format type strings (MIME types) to include in the
95 95 format data dict. If this is set *only* the format types included
96 96 in this list will be computed.
97 97 exclude : list or tuple, optional
98 98 A list of format type strings (MIME types) to exclude in the format
99 99 data dict. If this is set all format types will be computed,
100 100 except for those included in this argument.
101 101 metadata : dict, optional
102 102 A dictionary of metadata to associate with the output.
103 103 mime-type keys in this dictionary will be associated with the individual
104 104 representation formats, if they exist.
105 105 """
106 106 raw = kwargs.get('raw', False)
107 107 include = kwargs.get('include')
108 108 exclude = kwargs.get('exclude')
109 109 metadata = kwargs.get('metadata')
110 110
111 111 from IPython.core.interactiveshell import InteractiveShell
112 112
113 113 if not raw:
114 114 format = InteractiveShell.instance().display_formatter.format
115 115
116 116 for obj in objs:
117 117
118 118 # If _ipython_display_ is defined, use that to display this object.
119 119 display_method = getattr(obj, '_ipython_display_', None)
120 120 if display_method is not None:
121 121 try:
122 122 display_method(**kwargs)
123 123 except NotImplementedError:
124 124 pass
125 125 else:
126 126 continue
127 127 if raw:
128 128 publish_display_data('display', obj, metadata)
129 129 else:
130 130 format_dict, md_dict = format(obj, include=include, exclude=exclude)
131 131 if metadata:
132 132 # kwarg-specified metadata gets precedence
133 133 _merge(md_dict, metadata)
134 134 publish_display_data('display', format_dict, md_dict)
135 135
136 136
137 137 def display_pretty(*objs, **kwargs):
138 138 """Display the pretty (default) representation of an object.
139 139
140 140 Parameters
141 141 ----------
142 142 objs : tuple of objects
143 143 The Python objects to display, or if raw=True raw text data to
144 144 display.
145 145 raw : bool
146 146 Are the data objects raw data or Python objects that need to be
147 147 formatted before display? [default: False]
148 148 metadata : dict (optional)
149 149 Metadata to be associated with the specific mimetype output.
150 150 """
151 151 _display_mimetype('text/plain', objs, **kwargs)
152 152
153 153
154 154 def display_html(*objs, **kwargs):
155 155 """Display the HTML representation of an object.
156 156
157 157 Parameters
158 158 ----------
159 159 objs : tuple of objects
160 160 The Python objects to display, or if raw=True raw HTML data to
161 161 display.
162 162 raw : bool
163 163 Are the data objects raw data or Python objects that need to be
164 164 formatted before display? [default: False]
165 165 metadata : dict (optional)
166 166 Metadata to be associated with the specific mimetype output.
167 167 """
168 168 _display_mimetype('text/html', objs, **kwargs)
169 169
170 170
171 171 def display_svg(*objs, **kwargs):
172 172 """Display the SVG representation of an object.
173 173
174 174 Parameters
175 175 ----------
176 176 objs : tuple of objects
177 177 The Python objects to display, or if raw=True raw svg data to
178 178 display.
179 179 raw : bool
180 180 Are the data objects raw data or Python objects that need to be
181 181 formatted before display? [default: False]
182 182 metadata : dict (optional)
183 183 Metadata to be associated with the specific mimetype output.
184 184 """
185 185 _display_mimetype('image/svg+xml', objs, **kwargs)
186 186
187 187
188 188 def display_png(*objs, **kwargs):
189 189 """Display the PNG representation of an object.
190 190
191 191 Parameters
192 192 ----------
193 193 objs : tuple of objects
194 194 The Python objects to display, or if raw=True raw png data to
195 195 display.
196 196 raw : bool
197 197 Are the data objects raw data or Python objects that need to be
198 198 formatted before display? [default: False]
199 199 metadata : dict (optional)
200 200 Metadata to be associated with the specific mimetype output.
201 201 """
202 202 _display_mimetype('image/png', objs, **kwargs)
203 203
204 204
205 205 def display_jpeg(*objs, **kwargs):
206 206 """Display the JPEG representation of an object.
207 207
208 208 Parameters
209 209 ----------
210 210 objs : tuple of objects
211 211 The Python objects to display, or if raw=True raw JPEG data to
212 212 display.
213 213 raw : bool
214 214 Are the data objects raw data or Python objects that need to be
215 215 formatted before display? [default: False]
216 216 metadata : dict (optional)
217 217 Metadata to be associated with the specific mimetype output.
218 218 """
219 219 _display_mimetype('image/jpeg', objs, **kwargs)
220 220
221 221
222 222 def display_latex(*objs, **kwargs):
223 223 """Display the LaTeX representation of an object.
224 224
225 225 Parameters
226 226 ----------
227 227 objs : tuple of objects
228 228 The Python objects to display, or if raw=True raw latex data to
229 229 display.
230 230 raw : bool
231 231 Are the data objects raw data or Python objects that need to be
232 232 formatted before display? [default: False]
233 233 metadata : dict (optional)
234 234 Metadata to be associated with the specific mimetype output.
235 235 """
236 236 _display_mimetype('text/latex', objs, **kwargs)
237 237
238 238
239 239 def display_json(*objs, **kwargs):
240 240 """Display the JSON representation of an object.
241 241
242 242 Note that not many frontends support displaying JSON.
243 243
244 244 Parameters
245 245 ----------
246 246 objs : tuple of objects
247 247 The Python objects to display, or if raw=True raw json data to
248 248 display.
249 249 raw : bool
250 250 Are the data objects raw data or Python objects that need to be
251 251 formatted before display? [default: False]
252 252 metadata : dict (optional)
253 253 Metadata to be associated with the specific mimetype output.
254 254 """
255 255 _display_mimetype('application/json', objs, **kwargs)
256 256
257 257
258 258 def display_javascript(*objs, **kwargs):
259 259 """Display the Javascript representation of an object.
260 260
261 261 Parameters
262 262 ----------
263 263 objs : tuple of objects
264 264 The Python objects to display, or if raw=True raw javascript data to
265 265 display.
266 266 raw : bool
267 267 Are the data objects raw data or Python objects that need to be
268 268 formatted before display? [default: False]
269 269 metadata : dict (optional)
270 270 Metadata to be associated with the specific mimetype output.
271 271 """
272 272 _display_mimetype('application/javascript', objs, **kwargs)
273 273
274 274
275 275 def display_pdf(*objs, **kwargs):
276 276 """Display the PDF representation of an object.
277 277
278 278 Parameters
279 279 ----------
280 280 objs : tuple of objects
281 281 The Python objects to display, or if raw=True raw javascript data to
282 282 display.
283 283 raw : bool
284 284 Are the data objects raw data or Python objects that need to be
285 285 formatted before display? [default: False]
286 286 metadata : dict (optional)
287 287 Metadata to be associated with the specific mimetype output.
288 288 """
289 289 _display_mimetype('application/pdf', objs, **kwargs)
290 290
291 291
292 292 #-----------------------------------------------------------------------------
293 293 # Smart classes
294 294 #-----------------------------------------------------------------------------
295 295
296 296
297 297 class DisplayObject(object):
298 298 """An object that wraps data to be displayed."""
299 299
300 300 _read_flags = 'r'
301 301
302 302 def __init__(self, data=None, url=None, filename=None):
303 303 """Create a display object given raw data.
304 304
305 305 When this object is returned by an expression or passed to the
306 306 display function, it will result in the data being displayed
307 307 in the frontend. The MIME type of the data should match the
308 308 subclasses used, so the Png subclass should be used for 'image/png'
309 309 data. If the data is a URL, the data will first be downloaded
310 310 and then displayed. If
311 311
312 312 Parameters
313 313 ----------
314 314 data : unicode, str or bytes
315 315 The raw data or a URL or file to load the data from
316 316 url : unicode
317 317 A URL to download the data from.
318 318 filename : unicode
319 319 Path to a local file to load the data from.
320 320 """
321 321 if data is not None and isinstance(data, string_types):
322 322 if data.startswith('http') and url is None:
323 323 url = data
324 324 filename = None
325 325 data = None
326 326 elif _safe_exists(data) and filename is None:
327 327 url = None
328 328 filename = data
329 329 data = None
330 330
331 331 self.data = data
332 332 self.url = url
333 333 self.filename = None if filename is None else unicode_type(filename)
334 334
335 335 self.reload()
336 336 self._check_data()
337 337
338 338 def _check_data(self):
339 339 """Override in subclasses if there's something to check."""
340 340 pass
341 341
342 342 def reload(self):
343 343 """Reload the raw data from file or URL."""
344 344 if self.filename is not None:
345 345 with open(self.filename, self._read_flags) as f:
346 346 self.data = f.read()
347 347 elif self.url is not None:
348 348 try:
349 349 try:
350 350 from urllib.request import urlopen # Py3
351 351 except ImportError:
352 352 from urllib2 import urlopen
353 353 response = urlopen(self.url)
354 354 self.data = response.read()
355 355 # extract encoding from header, if there is one:
356 356 encoding = None
357 357 for sub in response.headers['content-type'].split(';'):
358 358 sub = sub.strip()
359 359 if sub.startswith('charset'):
360 360 encoding = sub.split('=')[-1].strip()
361 361 break
362 362 # decode data, if an encoding was specified
363 363 if encoding:
364 364 self.data = self.data.decode(encoding, 'replace')
365 365 except:
366 366 self.data = None
367 367
368 368 class TextDisplayObject(DisplayObject):
369 369 """Validate that display data is text"""
370 370 def _check_data(self):
371 371 if self.data is not None and not isinstance(self.data, string_types):
372 372 raise TypeError("%s expects text, not %r" % (self.__class__.__name__, self.data))
373 373
374 374 class Pretty(TextDisplayObject):
375 375
376 376 def _repr_pretty_(self):
377 377 return self.data
378 378
379 379
380 380 class HTML(TextDisplayObject):
381 381
382 382 def _repr_html_(self):
383 383 return self.data
384 384
385 385 def __html__(self):
386 386 """
387 387 This method exists to inform other HTML-using modules (e.g. Markupsafe,
388 388 htmltag, etc) that this object is HTML and does not need things like
389 389 special characters (<>&) escaped.
390 390 """
391 391 return self._repr_html_()
392 392
393 393
394 394 class Math(TextDisplayObject):
395 395
396 396 def _repr_latex_(self):
397 397 s = self.data.strip('$')
398 398 return "$$%s$$" % s
399 399
400 400
401 401 class Latex(TextDisplayObject):
402 402
403 403 def _repr_latex_(self):
404 404 return self.data
405 405
406 406
407 407 class SVG(DisplayObject):
408 408
409 409 # wrap data in a property, which extracts the <svg> tag, discarding
410 410 # document headers
411 411 _data = None
412 412
413 413 @property
414 414 def data(self):
415 415 return self._data
416 416
417 417 @data.setter
418 418 def data(self, svg):
419 419 if svg is None:
420 420 self._data = None
421 421 return
422 422 # parse into dom object
423 423 from xml.dom import minidom
424 424 svg = cast_bytes_py2(svg)
425 425 x = minidom.parseString(svg)
426 426 # get svg tag (should be 1)
427 427 found_svg = x.getElementsByTagName('svg')
428 428 if found_svg:
429 429 svg = found_svg[0].toxml()
430 430 else:
431 431 # fallback on the input, trust the user
432 432 # but this is probably an error.
433 433 pass
434 434 svg = cast_unicode(svg)
435 435 self._data = svg
436 436
437 437 def _repr_svg_(self):
438 438 return self.data
439 439
440 440
441 441 class JSON(TextDisplayObject):
442 442
443 443 def _repr_json_(self):
444 444 return self.data
445 445
446 446 css_t = """$("head").append($("<link/>").attr({
447 447 rel: "stylesheet",
448 448 type: "text/css",
449 449 href: "%s"
450 450 }));
451 451 """
452 452
453 453 lib_t1 = """$.getScript("%s", function () {
454 454 """
455 455 lib_t2 = """});
456 456 """
457 457
458 458 class Javascript(TextDisplayObject):
459 459
460 460 def __init__(self, data=None, url=None, filename=None, lib=None, css=None):
461 461 """Create a Javascript display object given raw data.
462 462
463 463 When this object is returned by an expression or passed to the
464 464 display function, it will result in the data being displayed
465 465 in the frontend. If the data is a URL, the data will first be
466 466 downloaded and then displayed.
467 467
468 468 In the Notebook, the containing element will be available as `element`,
469 469 and jQuery will be available. The output area starts hidden, so if
470 470 the js appends content to `element` that should be visible, then
471 471 it must call `container.show()` to unhide the area.
472 472
473 473 Parameters
474 474 ----------
475 475 data : unicode, str or bytes
476 476 The Javascript source code or a URL to download it from.
477 477 url : unicode
478 478 A URL to download the data from.
479 479 filename : unicode
480 480 Path to a local file to load the data from.
481 481 lib : list or str
482 482 A sequence of Javascript library URLs to load asynchronously before
483 483 running the source code. The full URLs of the libraries should
484 484 be given. A single Javascript library URL can also be given as a
485 485 string.
486 486 css: : list or str
487 487 A sequence of css files to load before running the source code.
488 488 The full URLs of the css files should be given. A single css URL
489 489 can also be given as a string.
490 490 """
491 491 if isinstance(lib, string_types):
492 492 lib = [lib]
493 493 elif lib is None:
494 494 lib = []
495 495 if isinstance(css, string_types):
496 496 css = [css]
497 497 elif css is None:
498 498 css = []
499 499 if not isinstance(lib, (list,tuple)):
500 500 raise TypeError('expected sequence, got: %r' % lib)
501 501 if not isinstance(css, (list,tuple)):
502 502 raise TypeError('expected sequence, got: %r' % css)
503 503 self.lib = lib
504 504 self.css = css
505 505 super(Javascript, self).__init__(data=data, url=url, filename=filename)
506 506
507 507 def _repr_javascript_(self):
508 508 r = ''
509 509 for c in self.css:
510 510 r += css_t % c
511 511 for l in self.lib:
512 512 r += lib_t1 % l
513 513 r += self.data
514 514 r += lib_t2*len(self.lib)
515 515 return r
516 516
517 517 # constants for identifying png/jpeg data
518 518 _PNG = b'\x89PNG\r\n\x1a\n'
519 519 _JPEG = b'\xff\xd8'
520 520
521 521 def _pngxy(data):
522 522 """read the (width, height) from a PNG header"""
523 523 ihdr = data.index(b'IHDR')
524 524 # next 8 bytes are width/height
525 525 w4h4 = data[ihdr+4:ihdr+12]
526 526 return struct.unpack('>ii', w4h4)
527 527
528 528 def _jpegxy(data):
529 529 """read the (width, height) from a JPEG header"""
530 530 # adapted from http://www.64lines.com/jpeg-width-height
531 531
532 532 idx = 4
533 533 while True:
534 534 block_size = struct.unpack('>H', data[idx:idx+2])[0]
535 535 idx = idx + block_size
536 536 if data[idx:idx+2] == b'\xFF\xC0':
537 537 # found Start of Frame
538 538 iSOF = idx
539 539 break
540 540 else:
541 541 # read another block
542 542 idx += 2
543 543
544 544 h, w = struct.unpack('>HH', data[iSOF+5:iSOF+9])
545 545 return w, h
546 546
547 547 class Image(DisplayObject):
548 548
549 549 _read_flags = 'rb'
550 550 _FMT_JPEG = u'jpeg'
551 551 _FMT_PNG = u'png'
552 552 _ACCEPTABLE_EMBEDDINGS = [_FMT_JPEG, _FMT_PNG]
553 553
554 554 def __init__(self, data=None, url=None, filename=None, format=u'png', embed=None, width=None, height=None, retina=False):
555 555 """Create a PNG/JPEG image object given raw data.
556 556
557 557 When this object is returned by an input cell or passed to the
558 558 display function, it will result in the image being displayed
559 559 in the frontend.
560 560
561 561 Parameters
562 562 ----------
563 563 data : unicode, str or bytes
564 564 The raw image data or a URL or filename to load the data from.
565 565 This always results in embedded image data.
566 566 url : unicode
567 567 A URL to download the data from. If you specify `url=`,
568 568 the image data will not be embedded unless you also specify `embed=True`.
569 569 filename : unicode
570 570 Path to a local file to load the data from.
571 571 Images from a file are always embedded.
572 572 format : unicode
573 573 The format of the image data (png/jpeg/jpg). If a filename or URL is given
574 574 for format will be inferred from the filename extension.
575 575 embed : bool
576 576 Should the image data be embedded using a data URI (True) or be
577 577 loaded using an <img> tag. Set this to True if you want the image
578 578 to be viewable later with no internet connection in the notebook.
579 579
580 580 Default is `True`, unless the keyword argument `url` is set, then
581 581 default value is `False`.
582 582
583 583 Note that QtConsole is not able to display images if `embed` is set to `False`
584 584 width : int
585 585 Width to which to constrain the image in html
586 586 height : int
587 587 Height to which to constrain the image in html
588 588 retina : bool
589 589 Automatically set the width and height to half of the measured
590 590 width and height.
591 591 This only works for embedded images because it reads the width/height
592 592 from image data.
593 593 For non-embedded images, you can just set the desired display width
594 594 and height directly.
595 595
596 596 Examples
597 597 --------
598 598 # embedded image data, works in qtconsole and notebook
599 599 # when passed positionally, the first arg can be any of raw image data,
600 600 # a URL, or a filename from which to load image data.
601 601 # The result is always embedding image data for inline images.
602 602 Image('http://www.google.fr/images/srpr/logo3w.png')
603 603 Image('/path/to/image.jpg')
604 604 Image(b'RAW_PNG_DATA...')
605 605
606 606 # Specifying Image(url=...) does not embed the image data,
607 607 # it only generates `<img>` tag with a link to the source.
608 608 # This will not work in the qtconsole or offline.
609 609 Image(url='http://www.google.fr/images/srpr/logo3w.png')
610 610
611 611 """
612 612 if filename is not None:
613 613 ext = self._find_ext(filename)
614 614 elif url is not None:
615 615 ext = self._find_ext(url)
616 616 elif data is None:
617 617 raise ValueError("No image data found. Expecting filename, url, or data.")
618 618 elif isinstance(data, string_types) and (
619 619 data.startswith('http') or _safe_exists(data)
620 620 ):
621 621 ext = self._find_ext(data)
622 622 else:
623 623 ext = None
624 624
625 625 if ext is not None:
626 626 format = ext.lower()
627 627 if ext == u'jpg' or ext == u'jpeg':
628 628 format = self._FMT_JPEG
629 629 if ext == u'png':
630 630 format = self._FMT_PNG
631 631 elif isinstance(data, bytes) and format == 'png':
632 632 # infer image type from image data header,
633 633 # only if format might not have been specified.
634 634 if data[:2] == _JPEG:
635 635 format = 'jpeg'
636 636
637 637 self.format = unicode_type(format).lower()
638 638 self.embed = embed if embed is not None else (url is None)
639 639
640 640 if self.embed and self.format not in self._ACCEPTABLE_EMBEDDINGS:
641 641 raise ValueError("Cannot embed the '%s' image format" % (self.format))
642 642 self.width = width
643 643 self.height = height
644 644 self.retina = retina
645 645 super(Image, self).__init__(data=data, url=url, filename=filename)
646 646
647 647 if retina:
648 648 self._retina_shape()
649 649
650 650 def _retina_shape(self):
651 651 """load pixel-doubled width and height from image data"""
652 652 if not self.embed:
653 653 return
654 654 if self.format == 'png':
655 655 w, h = _pngxy(self.data)
656 656 elif self.format == 'jpeg':
657 657 w, h = _jpegxy(self.data)
658 658 else:
659 659 # retina only supports png
660 660 return
661 661 self.width = w // 2
662 662 self.height = h // 2
663 663
664 664 def reload(self):
665 665 """Reload the raw data from file or URL."""
666 666 if self.embed:
667 667 super(Image,self).reload()
668 668 if self.retina:
669 669 self._retina_shape()
670 670
671 671 def _repr_html_(self):
672 672 if not self.embed:
673 673 width = height = ''
674 674 if self.width:
675 675 width = ' width="%d"' % self.width
676 676 if self.height:
677 677 height = ' height="%d"' % self.height
678 678 return u'<img src="%s"%s%s/>' % (self.url, width, height)
679 679
680 680 def _data_and_metadata(self):
681 681 """shortcut for returning metadata with shape information, if defined"""
682 682 md = {}
683 683 if self.width:
684 684 md['width'] = self.width
685 685 if self.height:
686 686 md['height'] = self.height
687 687 if md:
688 688 return self.data, md
689 689 else:
690 690 return self.data
691 691
692 692 def _repr_png_(self):
693 693 if self.embed and self.format == u'png':
694 694 return self._data_and_metadata()
695 695
696 696 def _repr_jpeg_(self):
697 697 if self.embed and (self.format == u'jpeg' or self.format == u'jpg'):
698 698 return self._data_and_metadata()
699 699
700 700 def _find_ext(self, s):
701 701 return unicode_type(s.split('.')[-1].lower())
702 702
703 703
704 704 def clear_output(wait=False):
705 705 """Clear the output of the current cell receiving output.
706 706
707 707 Parameters
708 708 ----------
709 709 wait : bool [default: false]
710 710 Wait to clear the output until new output is available to replace it."""
711 711 from IPython.core.interactiveshell import InteractiveShell
712 712 if InteractiveShell.initialized():
713 713 InteractiveShell.instance().display_pub.clear_output(wait)
714 714 else:
715 715 from IPython.utils import io
716 716 print('\033[2K\r', file=io.stdout, end='')
717 717 io.stdout.flush()
718 718 print('\033[2K\r', file=io.stderr, end='')
719 719 io.stderr.flush()
720 720
721 721
722 def select_matplotlib_formats(formats, quality=90):
723 """Select figure formats for the inline backend.
722 @skip_doctest
723 def set_matplotlib_formats(*formats, **kwargs):
724 """Select figure formats for the inline backend. Optionally pass quality for JPEG.
725
726 For example, this enables PNG and JPEG output with a JPEG quality of 90%::
727
728 In [1]: set_matplotlib_formats('png', 'jpeg', quality=90)
729
730 To set this in your config files use the following::
731
732 c.InlineBackend.figure_formats = {'pdf', 'png', 'svg'}
724 733
725 734 Parameters
726 ==========
727 formats : list
735 ----------
736 *formats : list, tuple
728 737 One or a set of figure formats to enable: 'png', 'retina', 'jpeg', 'svg', 'pdf'.
729 738 quality : int
730 A percentage for the quality of JPEG figures.
739 A percentage for the quality of JPEG figures. Defaults to 90.
731 740 """
732 741 from IPython.core.interactiveshell import InteractiveShell
733 742 from IPython.core.pylabtools import select_figure_formats
734 743 shell = InteractiveShell.instance()
735 744 select_figure_formats(shell, formats, quality=90)
736 745
746 @skip_doctest
747 def set_matplotlib_close(close):
748 """Should the inline backend close all figures automatically.
749
750 By default, the inline backend used in the IPython Notebook will close all
751 matplotlib figures automatically after each cell is run. This means that
752 plots in different cells won't interfere. Sometimes, you may want to make
753 a plot in one cell and then refine it in later cells. This can be accomplished
754 by::
755
756 In [1]: set_matplotlib_close(False)
757
758 To set this in your config files use the following::
759
760 c.InlineBackend.close_figures = False
761
762 Parameters
763 ----------
764 close : bool
765 Should all matplotlib figures be automatically closed after each cell is
766 run?
767 """
768 from IPython.kernel.zmq.pylab.backend_inline import InlineBackend
769 ilbe = InlineBackend.instance()
770 ilbe.close_figures = close
771
@@ -1,143 +1,147 b''
1 1 """Implementation of magic functions for matplotlib/pylab support.
2 2 """
3 3 from __future__ import print_function
4 4 #-----------------------------------------------------------------------------
5 5 # Copyright (c) 2012 The IPython Development Team.
6 6 #
7 7 # Distributed under the terms of the Modified BSD License.
8 8 #
9 9 # The full license is in the file COPYING.txt, distributed with this software.
10 10 #-----------------------------------------------------------------------------
11 11
12 12 #-----------------------------------------------------------------------------
13 13 # Imports
14 14 #-----------------------------------------------------------------------------
15 15
16 16 # Our own packages
17 17 from IPython.config.application import Application
18 18 from IPython.core import magic_arguments
19 19 from IPython.core.magic import Magics, magics_class, line_magic
20 20 from IPython.testing.skipdoctest import skip_doctest
21 21 from IPython.utils.warn import warn
22 22 from IPython.core.pylabtools import backends
23 23
24 24 #-----------------------------------------------------------------------------
25 25 # Magic implementation classes
26 26 #-----------------------------------------------------------------------------
27 27
28 28 magic_gui_arg = magic_arguments.argument(
29 29 'gui', nargs='?',
30 30 help="""Name of the matplotlib backend to use %s.
31 31 If given, the corresponding matplotlib backend is used,
32 32 otherwise it will be matplotlib's default
33 33 (which you can set in your matplotlib config file).
34 34 """ % str(tuple(sorted(backends.keys())))
35 35 )
36 36
37 37
38 38 @magics_class
39 39 class PylabMagics(Magics):
40 40 """Magics related to matplotlib's pylab support"""
41 41
42 42 @skip_doctest
43 43 @line_magic
44 44 @magic_arguments.magic_arguments()
45 45 @magic_gui_arg
46 46 def matplotlib(self, line=''):
47 47 """Set up matplotlib to work interactively.
48 48
49 49 This function lets you activate matplotlib interactive support
50 at any point during an IPython session.
51 It does not import anything into the interactive namespace.
50 at any point during an IPython session. It does not import anything
51 into the interactive namespace.
52 52
53 If you are using the inline matplotlib backend for embedded figures,
54 you can adjust its behavior via the %config magic::
53 If you are using the inline matplotlib backend in the IPython Notebook
54 you can set which figure formats are enabled using the following::
55 55
56 # enable SVG figures, necessary for SVG+XHTML export in the qtconsole
57 In [1]: %config InlineBackend.figure_format = 'svg'
56 In [1]: from IPython.display import set_matplotlib_formats
58 57
59 # change the behavior of closing all figures at the end of each
60 # execution (cell), or allowing reuse of active figures across
61 # cells:
62 In [2]: %config InlineBackend.close_figures = False
58 In [2]: set_matplotlib_formats('pdf', 'svg')
59
60 See the docstring of `IPython.display.set_matplotlib_formats` and
61 `IPython.display.set_matplotlib_close` for more information on
62 changing the behavior of the inline backend.
63 63
64 64 Examples
65 65 --------
66 In this case, where the MPL default is TkAgg::
66 To enable the inline backend for usage with the IPython Notebook::
67
68 In [1]: %matplotlib inline
69
70 In this case, where the matplotlib default is TkAgg::
67 71
68 72 In [2]: %matplotlib
69 73 Using matplotlib backend: TkAgg
70 74
71 But you can explicitly request a different backend::
75 But you can explicitly request a different GUI backend::
72 76
73 77 In [3]: %matplotlib qt
74 78 """
75 79 args = magic_arguments.parse_argstring(self.matplotlib, line)
76 80 gui, backend = self.shell.enable_matplotlib(args.gui)
77 81 self._show_matplotlib_backend(args.gui, backend)
78 82
79 83 @skip_doctest
80 84 @line_magic
81 85 @magic_arguments.magic_arguments()
82 86 @magic_arguments.argument(
83 87 '--no-import-all', action='store_true', default=None,
84 88 help="""Prevent IPython from performing ``import *`` into the interactive namespace.
85 89
86 90 You can govern the default behavior of this flag with the
87 91 InteractiveShellApp.pylab_import_all configurable.
88 92 """
89 93 )
90 94 @magic_gui_arg
91 95 def pylab(self, line=''):
92 96 """Load numpy and matplotlib to work interactively.
93 97
94 98 This function lets you activate pylab (matplotlib, numpy and
95 99 interactive support) at any point during an IPython session.
96 100
97 101 %pylab makes the following imports::
98 102
99 103 import numpy
100 104 import matplotlib
101 105 from matplotlib import pylab, mlab, pyplot
102 106 np = numpy
103 107 plt = pyplot
104 108
105 109 from IPython.display import display
106 110 from IPython.core.pylabtools import figsize, getfigs
107 111
108 112 from pylab import *
109 113 from numpy import *
110 114
111 115 If you pass `--no-import-all`, the last two `*` imports will be excluded.
112 116
113 117 See the %matplotlib magic for more details about activating matplotlib
114 118 without affecting the interactive namespace.
115 119 """
116 120 args = magic_arguments.parse_argstring(self.pylab, line)
117 121 if args.no_import_all is None:
118 122 # get default from Application
119 123 if Application.initialized():
120 124 app = Application.instance()
121 125 try:
122 126 import_all = app.pylab_import_all
123 127 except AttributeError:
124 128 import_all = True
125 129 else:
126 130 # nothing specified, no app - default True
127 131 import_all = True
128 132 else:
129 133 # invert no-import flag
130 134 import_all = not args.no_import_all
131 135
132 136 gui, backend, clobbered = self.shell.enable_pylab(args.gui, import_all=import_all)
133 137 self._show_matplotlib_backend(args.gui, backend)
134 138 print ("Populating the interactive namespace from numpy and matplotlib")
135 139 if clobbered:
136 140 warn("pylab import has clobbered these variables: %s" % clobbered +
137 141 "\n`%pylab --no-import-all` prevents importing * from pylab and numpy"
138 142 )
139 143
140 144 def _show_matplotlib_backend(self, gui, backend):
141 145 """show matplotlib message backend message"""
142 146 if not gui or gui == 'auto':
143 147 print("Using matplotlib backend: %s" % backend)
General Comments 0
You need to be logged in to leave comments. Login now