display.py
1470 lines
| 48.4 KiB
| text/x-python
|
PythonLexer
Brian Granger
|
r3278 | # -*- coding: utf-8 -*- | ||
Min RK
|
r19557 | """Top-level display functions for displaying object in different formats.""" | ||
Brian Granger
|
r3278 | |||
Min RK
|
r19557 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian Granger
|
r3278 | |||
MinRK
|
r6422 | |||
Marius van Niekerk
|
r23854 | from binascii import b2a_hex, b2a_base64, hexlify | ||
Min RK
|
r19557 | import json | ||
import mimetypes | ||||
MinRK
|
r10050 | import os | ||
MinRK
|
r10803 | import struct | ||
Thomas Kluyver
|
r22192 | import sys | ||
Min RK
|
r19557 | import warnings | ||
M Pacer
|
r23746 | from copy import deepcopy | ||
MinRK
|
r10050 | |||
Srinivas Reddy Thatiparthy
|
r23666 | from IPython.utils.py3compat import cast_unicode | ||
Brian E. Granger
|
r15126 | from IPython.testing.skipdoctest import skip_doctest | ||
MinRK
|
r10449 | |||
Thomas Kluyver
|
r17121 | __all__ = ['display', 'display_pretty', 'display_html', 'display_markdown', | ||
Min RK
|
r23849 | 'display_svg', 'display_png', 'display_jpeg', 'display_latex', 'display_json', | ||
Thomas Kluyver
|
r17121 | 'display_javascript', 'display_pdf', 'DisplayObject', 'TextDisplayObject', | ||
Marius van Niekerk
|
r23859 | 'Pretty', 'HTML', 'Markdown', 'Math', 'Latex', 'SVG', 'ProgressBar', 'JSON', | ||
'GeoJSON', 'Javascript', 'Image', 'clear_output', 'set_matplotlib_formats', | ||||
'set_matplotlib_close', 'publish_display_data', 'update_display', 'DisplayHandle', | ||||
'Video'] | ||||
Thomas Kluyver
|
r17121 | |||
Brian Granger
|
r3278 | #----------------------------------------------------------------------------- | ||
MinRK
|
r10050 | # utility functions | ||
#----------------------------------------------------------------------------- | ||||
def _safe_exists(path): | ||||
Kyle Kelley
|
r10213 | """Check path, but don't let exceptions raise""" | ||
MinRK
|
r10050 | try: | ||
return os.path.exists(path) | ||||
except Exception: | ||||
return False | ||||
MinRK
|
r10448 | def _merge(d1, d2): | ||
"""Like update, but merges sub-dicts instead of clobbering at the top level. | ||||
Dan McDougall
|
r10490 | |||
MinRK
|
r10448 | Updates d1 in-place | ||
""" | ||||
Dan McDougall
|
r10490 | |||
MinRK
|
r10448 | if not isinstance(d2, dict) or not isinstance(d1, dict): | ||
return d2 | ||||
for key, value in d2.items(): | ||||
d1[key] = _merge(d1.get(key), value) | ||||
return d1 | ||||
MinRK
|
r10449 | def _display_mimetype(mimetype, objs, raw=False, metadata=None): | ||
"""internal implementation of all display_foo methods | ||||
Parameters | ||||
---------- | ||||
mimetype : str | ||||
The mimetype to be published (e.g. 'image/png') | ||||
objs : tuple of objects | ||||
The Python objects to display, or if raw=True raw text data to | ||||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
metadata : dict (optional) | ||||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
MinRK
|
r10449 | """ | ||
MinRK
|
r10450 | if metadata: | ||
metadata = {mimetype: metadata} | ||||
MinRK
|
r10449 | if raw: | ||
MinRK
|
r10450 | # turn list of pngdata into list of { 'image/png': pngdata } | ||
objs = [ {mimetype: obj} for obj in objs ] | ||||
display(*objs, raw=raw, metadata=metadata, include=[mimetype]) | ||||
MinRK
|
r10050 | #----------------------------------------------------------------------------- | ||
Brian Granger
|
r3278 | # Main functions | ||
#----------------------------------------------------------------------------- | ||||
Min RK
|
r23015 | # use * to indicate transient is keyword-only | ||
Min RK
|
r23011 | def publish_display_data(data, metadata=None, source=None, *, transient=None, **kwargs): | ||
Thomas Kluyver
|
r17124 | """Publish data and metadata to all frontends. | ||
See the ``display_data`` message in the messaging documentation for | ||||
more details about this message type. | ||||
Min RK
|
r23849 | Keys of data and metadata can be any mime-type. | ||
Thomas Kluyver
|
r17124 | |||
Parameters | ||||
---------- | ||||
data : dict | ||||
A dictionary having keys that are valid MIME types (like | ||||
'text/plain' or 'image/svg+xml') and values that are the data for | ||||
that MIME type. The data itself must be a JSON'able data | ||||
structure. Minimally all data should have the 'text/plain' data, | ||||
which can be displayed by all frontends. If more than the plain | ||||
text is given, it is up to the frontend to decide which | ||||
representation to use. | ||||
metadata : dict | ||||
A dictionary for metadata related to the data. This can contain | ||||
arbitrary key, value pairs that frontends can use to interpret | ||||
the data. mime-type keys matching those in data can be used | ||||
to specify metadata about particular representations. | ||||
source : str, deprecated | ||||
Unused. | ||||
Min RK
|
r23015 | transient : dict, keyword-only | ||
Min RK
|
r23010 | A dictionary of transient data, such as display_id. | ||
Thomas Kluyver
|
r17124 | """ | ||
from IPython.core.interactiveshell import InteractiveShell | ||||
Min RK
|
r23010 | |||
display_pub = InteractiveShell.instance().display_pub | ||||
# only pass transient if supplied, | ||||
# to avoid errors with older ipykernel. | ||||
# TODO: We could check for ipykernel version and provide a detailed upgrade message. | ||||
if transient: | ||||
kwargs['transient'] = transient | ||||
display_pub.publish( | ||||
Thomas Kluyver
|
r17124 | data=data, | ||
metadata=metadata, | ||||
Min RK
|
r22999 | **kwargs | ||
Thomas Kluyver
|
r17124 | ) | ||
Min RK
|
r23012 | |||
def _new_id(): | ||||
"""Generate a new random text id with urandom""" | ||||
return b2a_hex(os.urandom(16)).decode('ascii') | ||||
Min RK
|
r23011 | def display(*objs, include=None, exclude=None, metadata=None, transient=None, display_id=None, **kwargs): | ||
Brian Granger
|
r3278 | """Display a Python object in all frontends. | ||
By default all representations will be computed and sent to the frontends. | ||||
Frontends can decide which representation is used and how. | ||||
Matthias Bussonnier
|
r23712 | In terminal IPython this will be similar to using :func:`print`, for use in richer | ||
Matthias Bussonnier
|
r23711 | frontends see Jupyter notebook examples with rich display logic. | ||
Brian Granger
|
r3278 | Parameters | ||
---------- | ||||
Brian Granger
|
r3288 | objs : tuple of objects | ||
The Python objects to display. | ||||
MinRK
|
r10450 | raw : bool, optional | ||
Are the objects to be displayed already mimetype-keyed dicts of raw display data, | ||||
or Python objects that need to be formatted before display? [default: False] | ||||
Matthias Bussonnier
|
r23712 | include : list, tuple or set, optional | ||
Brian Granger
|
r3278 | A list of format type strings (MIME types) to include in the | ||
format data dict. If this is set *only* the format types included | ||||
in this list will be computed. | ||||
Matthias Bussonnier
|
r23712 | exclude : list, tuple or set, optional | ||
Kyle Kelley
|
r10213 | A list of format type strings (MIME types) to exclude in the format | ||
Brian Granger
|
r3278 | data dict. If this is set all format types will be computed, | ||
except for those included in this argument. | ||||
MinRK
|
r10448 | metadata : dict, optional | ||
A dictionary of metadata to associate with the output. | ||||
mime-type keys in this dictionary will be associated with the individual | ||||
representation formats, if they exist. | ||||
Min RK
|
r22999 | transient : dict, optional | ||
A dictionary of transient data to associate with the output. | ||||
Data in this dict should not be persisted to files (e.g. notebooks). | ||||
Matthias Bussonnier
|
r23712 | display_id : str, bool optional | ||
Min RK
|
r22999 | Set an id for the display. | ||
This id can be used for updating this display area later via update_display. | ||||
Matthias Bussonnier
|
r23712 | If given as `True`, generate a new `display_id` | ||
Min RK
|
r23011 | kwargs: additional keyword-args, optional | ||
Additional keyword-arguments are passed through to the display publisher. | ||||
Matthias Bussonnier
|
r23711 | |||
Min RK
|
r23012 | Returns | ||
------- | ||||
Matthias Bussonnier
|
r23711 | |||
Min RK
|
r23012 | handle: DisplayHandle | ||
Matthias Bussonnier
|
r23712 | Returns a handle on updatable displays for use with :func:`update_display`, | ||
if `display_id` is given. Returns :any:`None` if no `display_id` is given | ||||
(default). | ||||
Matthias Bussonnier
|
r23711 | |||
Examples | ||||
-------- | ||||
>>> class Json(object): | ||||
... def __init__(self, json): | ||||
... self.json = json | ||||
... def _repr_pretty_(self, pp, cycle): | ||||
... import json | ||||
... pp.text(json.dumps(self.json, indent=2)) | ||||
... def __repr__(self): | ||||
... return str(self.json) | ||||
... | ||||
>>> d = Json({1:2, 3: {4:5}}) | ||||
>>> print(d) | ||||
{1: 2, 3: {4: 5}} | ||||
>>> display(d) | ||||
{ | ||||
"1": 2, | ||||
"3": { | ||||
"4": 5 | ||||
} | ||||
} | ||||
>>> def int_formatter(integer, pp, cycle): | ||||
... pp.text('I'*integer) | ||||
>>> plain = get_ipython().display_formatter.formatters['text/plain'] | ||||
>>> plain.for_type(int, int_formatter) | ||||
<function _repr_pprint at 0x...> | ||||
>>> display(7-5) | ||||
II | ||||
>>> del plain.type_printers[int] | ||||
>>> display(7-5) | ||||
2 | ||||
See Also | ||||
-------- | ||||
Matthias Bussonnier
|
r23712 | :func:`update_display` | ||
Matthias Bussonnier
|
r23711 | |||
Notes | ||||
----- | ||||
In Python, objects can declare their textual representation using the | ||||
`__repr__` method. IPython expands on this idea and allows objects to declare | ||||
other, rich representations including: | ||||
- HTML | ||||
- JSON | ||||
- PNG | ||||
- JPEG | ||||
- SVG | ||||
- LaTeX | ||||
A single object can declare some or all of these representations; all are | ||||
handled by IPython's display system. | ||||
The main idea of the first approach is that you have to implement special | ||||
display methods when you define your class, one for each representation you | ||||
want to use. Here is a list of the names of the special methods and the | ||||
values they must return: | ||||
Michael Penkov
|
r24621 | - `_repr_html_`: return raw HTML as a string, or a tuple (see below). | ||
- `_repr_json_`: return a JSONable dict, or a tuple (see below). | ||||
- `_repr_jpeg_`: return raw JPEG data, or a tuple (see below). | ||||
- `_repr_png_`: return raw PNG data, or a tuple (see below). | ||||
- `_repr_svg_`: return raw SVG data as a string, or a tuple (see below). | ||||
- `_repr_latex_`: return LaTeX commands in a string surrounded by "$", | ||||
or a tuple (see below). | ||||
Matthias Bussonnier
|
r23711 | - `_repr_mimebundle_`: return a full mimebundle containing the mapping | ||
Min RK
|
r23849 | from all mimetypes to data. | ||
Use this for any mime-type not listed above. | ||||
Matthias Bussonnier
|
r23711 | |||
Michael Penkov
|
r24621 | The above functions may also return the object's metadata alonside the | ||
data. If the metadata is available, the functions will return a tuple | ||||
containing the data and metadata, in that order. If there is no metadata | ||||
available, then the functions will return the data only. | ||||
Matthias Bussonnier
|
r23711 | When you are directly writing your own classes, you can adapt them for | ||
display in IPython by following the above approach. But in practice, you | ||||
often need to work with existing classes that you can't easily modify. | ||||
Thomas Kluyver
|
r24041 | You can refer to the documentation on integrating with the display system in | ||
order to register custom formatters for already existing types | ||||
(:ref:`integrating_rich_display`). | ||||
Matthias Bussonnier
|
r23711 | |||
Matthias Bussonnier
|
r23712 | .. versionadded:: 5.4 display available without import | ||
.. versionadded:: 6.1 display available without import | ||||
Since IPython 5.4 and 6.1 :func:`display` is automatically made available to | ||||
the user without import. If you are using display in a document that might | ||||
be used in a pure python context or with older version of IPython, use the | ||||
Matthias Bussonnier
|
r23711 | following import at the top of your file:: | ||
from IPython.display import display | ||||
Brian Granger
|
r3278 | """ | ||
Mradul Dubey
|
r23744 | from IPython.core.interactiveshell import InteractiveShell | ||
Mradul Dubey
|
r23745 | if not InteractiveShell.initialized(): | ||
Mradul Dubey
|
r23744 | # Directly print objects. | ||
print(*objs) | ||||
return | ||||
Min RK
|
r22999 | raw = kwargs.pop('raw', False) | ||
Min RK
|
r23011 | if transient is None: | ||
transient = {} | ||||
M Pacer
|
r23746 | if metadata is None: | ||
metadata={} | ||||
Min RK
|
r23011 | if display_id: | ||
Matthias Bussonnier
|
r23711 | if display_id is True: | ||
Min RK
|
r23012 | display_id = _new_id() | ||
Min RK
|
r23011 | transient['display_id'] = display_id | ||
Min RK
|
r22999 | if kwargs.get('update') and 'display_id' not in transient: | ||
raise TypeError('display_id required for update_display') | ||||
Min RK
|
r23011 | if transient: | ||
kwargs['transient'] = transient | ||||
Bernardo B. Marques
|
r4872 | |||
Jonathan Frederic
|
r14233 | if not raw: | ||
format = InteractiveShell.instance().display_formatter.format | ||||
for obj in objs: | ||||
MinRK
|
r14795 | if raw: | ||
Min RK
|
r22999 | publish_display_data(data=obj, metadata=metadata, **kwargs) | ||
MinRK
|
r14795 | else: | ||
format_dict, md_dict = format(obj, include=include, exclude=exclude) | ||||
Min RK
|
r19381 | if not format_dict: | ||
# nothing to display (e.g. _ipython_display_ took over) | ||||
continue | ||||
MinRK
|
r14795 | if metadata: | ||
# kwarg-specified metadata gets precedence | ||||
_merge(md_dict, metadata) | ||||
Min RK
|
r23012 | publish_display_data(data=format_dict, metadata=md_dict, **kwargs) | ||
if display_id: | ||||
return DisplayHandle(display_id) | ||||
Min RK
|
r22999 | |||
Min RK
|
r23015 | # use * for keyword-only display_id arg | ||
Min RK
|
r23012 | def update_display(obj, *, display_id, **kwargs): | ||
"""Update an existing display by id | ||||
Min RK
|
r23011 | |||
Parameters | ||||
---------- | ||||
obj: | ||||
The object with which to update the display | ||||
display_id: keyword-only | ||||
The id of the display to update | ||||
Matthias Bussonnier
|
r23712 | |||
See Also | ||||
-------- | ||||
:func:`display` | ||||
Min RK
|
r23011 | """ | ||
Min RK
|
r22999 | kwargs['update'] = True | ||
Min RK
|
r23012 | display(obj, display_id=display_id, **kwargs) | ||
class DisplayHandle(object): | ||||
"""A handle on an updatable display | ||||
Min RK
|
r23011 | |||
Matthias Bussonnier
|
r23712 | Call `.update(obj)` to display a new object. | ||
Min RK
|
r23012 | |||
Matthias Bussonnier
|
r23712 | Call `.display(obj`) to add a new instance of this display, | ||
Min RK
|
r23012 | and update existing instances. | ||
Matthias Bussonnier
|
r23712 | |||
See Also | ||||
-------- | ||||
:func:`display`, :func:`update_display` | ||||
Min RK
|
r23012 | """ | ||
def __init__(self, display_id=None): | ||||
if display_id is None: | ||||
display_id = _new_id() | ||||
self.display_id = display_id | ||||
def __repr__(self): | ||||
return "<%s display_id=%s>" % (self.__class__.__name__, self.display_id) | ||||
def display(self, obj, **kwargs): | ||||
"""Make a new display with my id, updating existing instances. | ||||
Parameters | ||||
---------- | ||||
obj: | ||||
object to display | ||||
**kwargs: | ||||
additional keyword arguments passed to display | ||||
""" | ||||
display(obj, display_id=self.display_id, **kwargs) | ||||
def update(self, obj, **kwargs): | ||||
"""Update existing displays with my id | ||||
Parameters | ||||
---------- | ||||
obj: | ||||
object to display | ||||
**kwargs: | ||||
additional keyword arguments passed to update_display | ||||
""" | ||||
update_display(obj, display_id=self.display_id, **kwargs) | ||||
Brian Granger
|
r3279 | |||
MinRK
|
r10448 | |||
def display_pretty(*objs, **kwargs): | ||||
"""Display the pretty (default) representation of an object. | ||||
Parameters | ||||
---------- | ||||
objs : tuple of objects | ||||
The Python objects to display, or if raw=True raw text data to | ||||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
metadata : dict (optional) | ||||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
MinRK
|
r10448 | """ | ||
_display_mimetype('text/plain', objs, **kwargs) | ||||
Brian Granger
|
r3279 | |||
Brian E. Granger
|
r4526 | def display_html(*objs, **kwargs): | ||
Brian Granger
|
r3278 | """Display the HTML representation of an object. | ||
Adam Eury
|
r22947 | |||
Peter Waller
|
r21995 | Note: If raw=False and the object does not have a HTML | ||
representation, no HTML will be shown. | ||||
Brian Granger
|
r3278 | |||
Parameters | ||||
---------- | ||||
Brian Granger
|
r3288 | objs : tuple of objects | ||
Brian E. Granger
|
r4528 | The Python objects to display, or if raw=True raw HTML data to | ||
Brian E. Granger
|
r4526 | display. | ||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian E. Granger
|
r4526 | """ | ||
MinRK
|
r10448 | _display_mimetype('text/html', objs, **kwargs) | ||
Brian Granger
|
r3278 | |||
Andrew Jesaitis
|
r16364 | def display_markdown(*objs, **kwargs): | ||
"""Displays the Markdown representation of an object. | ||||
Parameters | ||||
---------- | ||||
objs : tuple of objects | ||||
The Python objects to display, or if raw=True raw markdown data to | ||||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
metadata : dict (optional) | ||||
Metadata to be associated with the specific mimetype output. | ||||
""" | ||||
_display_mimetype('text/markdown', objs, **kwargs) | ||||
Brian E. Granger
|
r4526 | def display_svg(*objs, **kwargs): | ||
Brian Granger
|
r3278 | """Display the SVG representation of an object. | ||
Parameters | ||||
---------- | ||||
Brian Granger
|
r3288 | objs : tuple of objects | ||
Brian E. Granger
|
r4526 | The Python objects to display, or if raw=True raw svg data to | ||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian Granger
|
r3278 | """ | ||
MinRK
|
r10448 | _display_mimetype('image/svg+xml', objs, **kwargs) | ||
Brian Granger
|
r3278 | |||
Brian E. Granger
|
r4526 | def display_png(*objs, **kwargs): | ||
Brian Granger
|
r3278 | """Display the PNG representation of an object. | ||
Parameters | ||||
---------- | ||||
Brian Granger
|
r3288 | objs : tuple of objects | ||
Brian E. Granger
|
r4526 | The Python objects to display, or if raw=True raw png data to | ||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian Granger
|
r3278 | """ | ||
MinRK
|
r10448 | _display_mimetype('image/png', objs, **kwargs) | ||
Brian Granger
|
r3278 | |||
Brian E. Granger
|
r4528 | def display_jpeg(*objs, **kwargs): | ||
"""Display the JPEG representation of an object. | ||||
Parameters | ||||
---------- | ||||
objs : tuple of objects | ||||
The Python objects to display, or if raw=True raw JPEG data to | ||||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian E. Granger
|
r4528 | """ | ||
MinRK
|
r10448 | _display_mimetype('image/jpeg', objs, **kwargs) | ||
Brian E. Granger
|
r4528 | |||
Brian E. Granger
|
r4526 | def display_latex(*objs, **kwargs): | ||
Brian Granger
|
r3278 | """Display the LaTeX representation of an object. | ||
Parameters | ||||
---------- | ||||
Brian Granger
|
r3288 | objs : tuple of objects | ||
Brian E. Granger
|
r4526 | The Python objects to display, or if raw=True raw latex data to | ||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian Granger
|
r3278 | """ | ||
MinRK
|
r10448 | _display_mimetype('text/latex', objs, **kwargs) | ||
Brian Granger
|
r3278 | |||
Brian E. Granger
|
r4526 | def display_json(*objs, **kwargs): | ||
Brian Granger
|
r3278 | """Display the JSON representation of an object. | ||
Kyle Kelley
|
r10212 | |||
Thomas Kluyver
|
r8086 | Note that not many frontends support displaying JSON. | ||
Brian Granger
|
r3278 | |||
Parameters | ||||
---------- | ||||
Brian Granger
|
r3288 | objs : tuple of objects | ||
Brian E. Granger
|
r4526 | The Python objects to display, or if raw=True raw json data to | ||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian Granger
|
r3278 | """ | ||
MinRK
|
r10448 | _display_mimetype('application/json', objs, **kwargs) | ||
Brian Granger
|
r3278 | |||
Brian E. Granger
|
r4526 | def display_javascript(*objs, **kwargs): | ||
Brian Granger
|
r3878 | """Display the Javascript representation of an object. | ||
Brian Granger
|
r3278 | |||
Brian Granger
|
r3878 | Parameters | ||
---------- | ||||
objs : tuple of objects | ||||
Brian E. Granger
|
r4526 | The Python objects to display, or if raw=True raw javascript data to | ||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
MinRK
|
r10448 | metadata : dict (optional) | ||
MinRK
|
r10450 | Metadata to be associated with the specific mimetype output. | ||
Brian Granger
|
r3878 | """ | ||
MinRK
|
r10448 | _display_mimetype('application/javascript', objs, **kwargs) | ||
Brian E. Granger
|
r4526 | |||
Brian E. Granger
|
r15121 | |||
def display_pdf(*objs, **kwargs): | ||||
"""Display the PDF representation of an object. | ||||
Parameters | ||||
---------- | ||||
objs : tuple of objects | ||||
The Python objects to display, or if raw=True raw javascript data to | ||||
display. | ||||
raw : bool | ||||
Are the data objects raw data or Python objects that need to be | ||||
formatted before display? [default: False] | ||||
metadata : dict (optional) | ||||
Metadata to be associated with the specific mimetype output. | ||||
""" | ||||
_display_mimetype('application/pdf', objs, **kwargs) | ||||
Brian E. Granger
|
r4526 | #----------------------------------------------------------------------------- | ||
# Smart classes | ||||
#----------------------------------------------------------------------------- | ||||
class DisplayObject(object): | ||||
"""An object that wraps data to be displayed.""" | ||||
Brian E. Granger
|
r4528 | _read_flags = 'r' | ||
Jessica B. Hamrick
|
r16396 | _show_mem_addr = False | ||
M Pacer
|
r23746 | metadata = None | ||
Brian E. Granger
|
r4528 | |||
M Pacer
|
r23746 | def __init__(self, data=None, url=None, filename=None, metadata=None): | ||
Brian E. Granger
|
r4528 | """Create a display object given raw data. | ||
Brian E. Granger
|
r4526 | |||
When this object is returned by an expression or passed to the | ||||
display function, it will result in the data being displayed | ||||
in the frontend. The MIME type of the data should match the | ||||
subclasses used, so the Png subclass should be used for 'image/png' | ||||
data. If the data is a URL, the data will first be downloaded | ||||
Bernardo B. Marques
|
r4872 | and then displayed. If | ||
Brian E. Granger
|
r4526 | |||
Parameters | ||||
---------- | ||||
data : unicode, str or bytes | ||||
MinRK
|
r10050 | The raw data or a URL or file to load the data from | ||
Brian E. Granger
|
r4528 | url : unicode | ||
A URL to download the data from. | ||||
filename : unicode | ||||
Path to a local file to load the data from. | ||||
M Pacer
|
r23746 | metadata : dict | ||
Dict of metadata associated to be the object when displayed | ||||
Brian E. Granger
|
r4526 | """ | ||
Srinivas Reddy Thatiparthy
|
r23037 | if data is not None and isinstance(data, str): | ||
MinRK
|
r10050 | if data.startswith('http') and url is None: | ||
url = data | ||||
filename = None | ||||
data = None | ||||
elif _safe_exists(data) and filename is None: | ||||
url = None | ||||
filename = data | ||||
data = None | ||||
Kyle Kelley
|
r10212 | |||
MinRK
|
r10050 | self.data = data | ||
self.url = url | ||||
Srinivas Reddy Thatiparthy
|
r23044 | self.filename = filename | ||
Kyle Kelley
|
r10212 | |||
M Pacer
|
r23746 | if metadata is not None: | ||
self.metadata = metadata | ||||
elif self.metadata is None: | ||||
self.metadata = {} | ||||
Brian E. Granger
|
r4528 | self.reload() | ||
MinRK
|
r14153 | self._check_data() | ||
Jessica B. Hamrick
|
r16396 | |||
def __repr__(self): | ||||
if not self._show_mem_addr: | ||||
cls = self.__class__ | ||||
r = "<%s.%s object>" % (cls.__module__, cls.__name__) | ||||
else: | ||||
r = super(DisplayObject, self).__repr__() | ||||
return r | ||||
MinRK
|
r14153 | def _check_data(self): | ||
"""Override in subclasses if there's something to check.""" | ||||
pass | ||||
Brian E. Granger
|
r4528 | |||
M Pacer
|
r23746 | def _data_and_metadata(self): | ||
"""shortcut for returning metadata with shape information, if defined""" | ||||
if self.metadata: | ||||
return self.data, deepcopy(self.metadata) | ||||
else: | ||||
return self.data | ||||
Brian E. Granger
|
r4528 | def reload(self): | ||
"""Reload the raw data from file or URL.""" | ||||
if self.filename is not None: | ||||
with open(self.filename, self._read_flags) as f: | ||||
self.data = f.read() | ||||
elif self.url is not None: | ||||
try: | ||||
Srinivas Reddy Thatiparthy
|
r23075 | # Deferred import | ||
from urllib.request import urlopen | ||||
Sean Vig
|
r13640 | response = urlopen(self.url) | ||
Brian E. Granger
|
r4528 | self.data = response.read() | ||
MinRK
|
r4684 | # extract encoding from header, if there is one: | ||
encoding = None | ||||
for sub in response.headers['content-type'].split(';'): | ||||
sub = sub.strip() | ||||
if sub.startswith('charset'): | ||||
encoding = sub.split('=')[-1].strip() | ||||
break | ||||
# decode data, if an encoding was specified | ||||
if encoding: | ||||
self.data = self.data.decode(encoding, 'replace') | ||||
Brian E. Granger
|
r4528 | except: | ||
self.data = None | ||||
Brian E. Granger
|
r4526 | |||
MinRK
|
r14153 | class TextDisplayObject(DisplayObject): | ||
"""Validate that display data is text""" | ||||
def _check_data(self): | ||||
Srinivas Reddy Thatiparthy
|
r23037 | if self.data is not None and not isinstance(self.data, str): | ||
MinRK
|
r14159 | raise TypeError("%s expects text, not %r" % (self.__class__.__name__, self.data)) | ||
MinRK
|
r14153 | |||
class Pretty(TextDisplayObject): | ||||
Brian E. Granger
|
r4526 | |||
ryan thielke
|
r23729 | def _repr_pretty_(self, pp, cycle): | ||
return pp.text(self.data) | ||||
Brian E. Granger
|
r4526 | |||
MinRK
|
r14153 | class HTML(TextDisplayObject): | ||
Brian E. Granger
|
r4526 | |||
Michael Penkov
|
r24619 | def __init__(self, data=None, url=None, filename=None, metadata=None): | ||
Michael Penkov
|
r24654 | def warn(): | ||
if not data: | ||||
return False | ||||
# | ||||
# Avoid calling lower() on the entire data, because it could be a | ||||
# long string and we're only interested in its beginning and end. | ||||
# | ||||
prefix = data[:10].lower() | ||||
suffix = data[-10:].lower() | ||||
return prefix.startswith("<iframe ") and suffix.endswith("</iframe>") | ||||
if warn(): | ||||
Michael Penkov
|
r24619 | warnings.warn("Consider using IPython.display.IFrame instead") | ||
super(HTML, self).__init__(data=data, url=url, filename=filename, metadata=metadata) | ||||
Brian E. Granger
|
r4526 | def _repr_html_(self): | ||
Thomas Kluyver
|
r24301 | return self._data_and_metadata() | ||
Brian E. Granger
|
r4526 | |||
Dan McDougall
|
r10490 | def __html__(self): | ||
""" | ||||
This method exists to inform other HTML-using modules (e.g. Markupsafe, | ||||
htmltag, etc) that this object is HTML and does not need things like | ||||
special characters (<>&) escaped. | ||||
""" | ||||
return self._repr_html_() | ||||
Brian E. Granger
|
r4526 | |||
Andrew Jesaitis
|
r16364 | class Markdown(TextDisplayObject): | ||
def _repr_markdown_(self): | ||||
Thomas Kluyver
|
r24301 | return self._data_and_metadata() | ||
Andrew Jesaitis
|
r16364 | |||
MinRK
|
r14153 | class Math(TextDisplayObject): | ||
Brian E. Granger
|
r4526 | |||
def _repr_latex_(self): | ||||
Matthias Bussonnier
|
r24780 | s = r"$\displaystyle %s$" % self.data.strip('$') | ||
Thomas Kluyver
|
r24301 | if self.metadata: | ||
return s, deepcopy(self.metadata) | ||||
else: | ||||
return s | ||||
Brian Granger
|
r6065 | |||
MinRK
|
r14153 | class Latex(TextDisplayObject): | ||
Brian Granger
|
r6065 | |||
def _repr_latex_(self): | ||||
Thomas Kluyver
|
r24301 | return self._data_and_metadata() | ||
Brian E. Granger
|
r4526 | |||
Brian E. Granger
|
r4528 | class SVG(DisplayObject): | ||
Kyle Kelley
|
r10212 | |||
Srinivas Reddy Thatiparthy
|
r23026 | _read_flags = 'rb' | ||
MinRK
|
r5671 | # wrap data in a property, which extracts the <svg> tag, discarding | ||
# document headers | ||||
_data = None | ||||
Kyle Kelley
|
r10212 | |||
MinRK
|
r5671 | @property | ||
def data(self): | ||||
return self._data | ||||
Kyle Kelley
|
r10212 | |||
MinRK
|
r5671 | @data.setter | ||
def data(self, svg): | ||||
if svg is None: | ||||
self._data = None | ||||
return | ||||
# parse into dom object | ||||
Thomas Kluyver
|
r9390 | from xml.dom import minidom | ||
MinRK
|
r5671 | x = minidom.parseString(svg) | ||
# get svg tag (should be 1) | ||||
found_svg = x.getElementsByTagName('svg') | ||||
if found_svg: | ||||
svg = found_svg[0].toxml() | ||||
else: | ||||
# fallback on the input, trust the user | ||||
# but this is probably an error. | ||||
pass | ||||
MinRK
|
r11535 | svg = cast_unicode(svg) | ||
MinRK
|
r5671 | self._data = svg | ||
M Pacer
|
r23746 | |||
Brian E. Granger
|
r4526 | def _repr_svg_(self): | ||
M Pacer
|
r23746 | return self._data_and_metadata() | ||
Brian E. Granger
|
r4526 | |||
Marius van Niekerk
|
r23856 | class ProgressBar(DisplayObject): | ||
Marius van Niekerk
|
r23854 | """Progressbar supports displaying a progressbar like element | ||
""" | ||||
def __init__(self, total): | ||||
"""Creates a new progressbar | ||||
Parameters | ||||
---------- | ||||
total : int | ||||
Marius van Niekerk
|
r23859 | maximum size of the progressbar | ||
Marius van Niekerk
|
r23854 | """ | ||
self.total = total | ||||
self._progress = 0 | ||||
Marius van Niekerk
|
r23860 | self.html_width = '60ex' | ||
self.text_width = 60 | ||||
self._display_id = hexlify(os.urandom(8)).decode('ascii') | ||||
def __repr__(self): | ||||
fraction = self.progress / self.total | ||||
filled = '=' * int(fraction * self.text_width) | ||||
rest = ' ' * (self.text_width - len(filled)) | ||||
return '[{}{}] {}/{}'.format( | ||||
filled, rest, | ||||
self.progress, self.total, | ||||
) | ||||
Marius van Niekerk
|
r23854 | |||
def _repr_html_(self): | ||||
Marius van Niekerk
|
r23860 | return "<progress style='width:{}' max='{}' value='{}'></progress>".format( | ||
self.html_width, self.total, self.progress) | ||||
Marius van Niekerk
|
r23854 | |||
def display(self): | ||||
display(self, display_id=self._display_id) | ||||
def update(self): | ||||
display(self, display_id=self._display_id, update=True) | ||||
@property | ||||
def progress(self): | ||||
return self._progress | ||||
@progress.setter | ||||
def progress(self, value): | ||||
self._progress = value | ||||
self.update() | ||||
Brian E. Granger
|
r4526 | |||
Henry Fredrick Schreiner
|
r23943 | def __iter__(self): | ||
self.display() | ||||
self._progress = -1 # First iteration is 0 | ||||
return self | ||||
def __next__(self): | ||||
"""Returns current value and increments display by one.""" | ||||
self.progress += 1 | ||||
if self.progress < self.total: | ||||
return self.progress | ||||
else: | ||||
raise StopIteration() | ||||
Min RK
|
r19557 | class JSON(DisplayObject): | ||
"""JSON expects a JSON-able dict or list | ||||
Adam Eury
|
r22947 | |||
Min RK
|
r19557 | not an already-serialized JSON string. | ||
Adam Eury
|
r22947 | |||
Min RK
|
r19557 | Scalar types (None, number, string) are not allowed, only dict or list containers. | ||
""" | ||||
# wrap data in a property, which warns about passing already-serialized JSON | ||||
_data = None | ||||
Dave Hirschfeld
|
r24412 | def __init__(self, data=None, url=None, filename=None, expanded=False, metadata=None, root='root', **kwargs): | ||
Adam Eury
|
r22947 | """Create a JSON display object given raw data. | ||
Parameters | ||||
---------- | ||||
data : dict or list | ||||
JSON data to display. Not an already-serialized JSON string. | ||||
Scalar types (None, number, string) are not allowed, only dict | ||||
or list containers. | ||||
url : unicode | ||||
A URL to download the data from. | ||||
filename : unicode | ||||
Path to a local file to load the data from. | ||||
expanded : boolean | ||||
Metadata to control whether a JSON display component is expanded. | ||||
metadata: dict | ||||
Specify extra metadata to attach to the json display object. | ||||
Dave Hirschfeld
|
r24412 | root : str | ||
The name of the root element of the JSON tree | ||||
Adam Eury
|
r22947 | """ | ||
Dave Hirschfeld
|
r24412 | self.metadata = { | ||
'expanded': expanded, | ||||
'root': root, | ||||
} | ||||
Grant Nestor
|
r23501 | if metadata: | ||
self.metadata.update(metadata) | ||||
Grant Nestor
|
r23503 | if kwargs: | ||
self.metadata.update(kwargs) | ||||
Adam Eury
|
r22947 | super(JSON, self).__init__(data=data, url=url, filename=filename) | ||
Min RK
|
r19557 | def _check_data(self): | ||
if self.data is not None and not isinstance(self.data, (dict, list)): | ||||
raise TypeError("%s expects JSONable dict or list, not %r" % (self.__class__.__name__, self.data)) | ||||
@property | ||||
def data(self): | ||||
return self._data | ||||
Adam Eury
|
r22947 | |||
Min RK
|
r19557 | @data.setter | ||
def data(self, data): | ||||
Srinivas Reddy Thatiparthy
|
r23037 | if isinstance(data, str): | ||
Matthias Bussonnier
|
r23395 | if getattr(self, 'filename', None) is None: | ||
Grant Nestor
|
r23393 | warnings.warn("JSON expects JSONable dict or list, not JSON strings") | ||
Min RK
|
r19557 | data = json.loads(data) | ||
self._data = data | ||||
Brian E. Granger
|
r4526 | |||
Adam Eury
|
r22947 | def _data_and_metadata(self): | ||
Grant Nestor
|
r23501 | return self.data, self.metadata | ||
Adam Eury
|
r22947 | |||
Brian E. Granger
|
r4526 | def _repr_json_(self): | ||
Adam Eury
|
r22947 | return self._data_and_metadata() | ||
Brian E. Granger
|
r4526 | |||
Grant Nestor
|
r24470 | _css_t = """var link = document.createElement("link"); | ||
link.ref = "stylesheet"; | ||||
link.type = "text/css"; | ||||
link.href = "%s"; | ||||
document.head.appendChild(link); | ||||
Brian Granger
|
r6120 | """ | ||
Grant Nestor
|
r24479 | _lib_t1 = """new Promise(function(resolve, reject) { | ||
var script = document.createElement("script"); | ||||
script.onload = resolve; | ||||
script.onerror = reject; | ||||
Grant Nestor
|
r24470 | script.src = "%s"; | ||
Grant Nestor
|
r24479 | document.head.appendChild(script); | ||
}).then(() => { | ||||
Brian Granger
|
r6120 | """ | ||
Brian E. Granger
|
r4526 | |||
Grant Nestor
|
r24479 | _lib_t2 = """ | ||
});""" | ||||
Grant Nestor
|
r24470 | |||
Grant Nestor
|
r23501 | class GeoJSON(JSON): | ||
Grant Nestor
|
r23404 | """GeoJSON expects JSON-able dict | ||
Grant Nestor
|
r23312 | |||
Grant Nestor
|
r23404 | not an already-serialized JSON string. | ||
Scalar types (None, number, string) are not allowed, only dict containers. | ||||
""" | ||||
Grant Nestor
|
r23501 | def __init__(self, *args, **kwargs): | ||
Grant Nestor
|
r23404 | """Create a GeoJSON display object given raw data. | ||
Parameters | ||||
---------- | ||||
data : dict or list | ||||
VegaLite data. Not an already-serialized JSON string. | ||||
Scalar types (None, number, string) are not allowed, only dict | ||||
or list containers. | ||||
url_template : string | ||||
Leaflet TileLayer URL template: http://leafletjs.com/reference.html#url-template | ||||
layer_options : dict | ||||
Leaflet TileLayer options: http://leafletjs.com/reference.html#tilelayer-options | ||||
url : unicode | ||||
A URL to download the data from. | ||||
filename : unicode | ||||
Path to a local file to load the data from. | ||||
metadata: dict | ||||
Specify extra metadata to attach to the json display object. | ||||
Matthias Bussonnier
|
r23407 | |||
Examples | ||||
-------- | ||||
The following will display an interactive map of Mars with a point of | ||||
interest on frontend that do support GeoJSON display. | ||||
>>> from IPython.display import GeoJSON | ||||
>>> GeoJSON(data={ | ||||
... "type": "Feature", | ||||
... "geometry": { | ||||
... "type": "Point", | ||||
... "coordinates": [-81.327, 296.038] | ||||
... } | ||||
... }, | ||||
... url_template="http://s3-eu-west-1.amazonaws.com/whereonmars.cartodb.net/{basemap_id}/{z}/{x}/{y}.png", | ||||
... layer_options={ | ||||
... "basemap_id": "celestia_mars-shaded-16k_global", | ||||
... "attribution" : "Celestia/praesepe", | ||||
... "minZoom" : 0, | ||||
... "maxZoom" : 18, | ||||
... }) | ||||
<IPython.core.display.GeoJSON object> | ||||
In the terminal IPython, you will only see the text representation of | ||||
the GeoJSON object. | ||||
Grant Nestor
|
r23404 | """ | ||
Grant Nestor
|
r23501 | super(GeoJSON, self).__init__(*args, **kwargs) | ||
Grant Nestor
|
r23312 | |||
def _ipython_display_(self): | ||||
bundle = { | ||||
'application/geo+json': self.data, | ||||
Grant Nestor
|
r23501 | 'text/plain': '<IPython.display.GeoJSON object>' | ||
Grant Nestor
|
r23312 | } | ||
Grant Nestor
|
r23404 | metadata = { | ||
Grant Nestor
|
r23501 | 'application/geo+json': self.metadata | ||
Grant Nestor
|
r23404 | } | ||
display(bundle, metadata=metadata, raw=True) | ||||
Grant Nestor
|
r23312 | |||
MinRK
|
r14153 | class Javascript(TextDisplayObject): | ||
Brian E. Granger
|
r4526 | |||
Brian Granger
|
r6120 | def __init__(self, data=None, url=None, filename=None, lib=None, css=None): | ||
"""Create a Javascript display object given raw data. | ||||
When this object is returned by an expression or passed to the | ||||
display function, it will result in the data being displayed | ||||
in the frontend. If the data is a URL, the data will first be | ||||
Kyle Kelley
|
r10212 | downloaded and then displayed. | ||
MinRK
|
r6425 | In the Notebook, the containing element will be available as `element`, | ||
Andrew Payne
|
r17002 | and jQuery will be available. Content appended to `element` will be | ||
Adam Eury
|
r22947 | visible in the output area. | ||
Brian Granger
|
r6120 | |||
Parameters | ||||
---------- | ||||
data : unicode, str or bytes | ||||
The Javascript source code or a URL to download it from. | ||||
url : unicode | ||||
A URL to download the data from. | ||||
filename : unicode | ||||
Path to a local file to load the data from. | ||||
lib : list or str | ||||
A sequence of Javascript library URLs to load asynchronously before | ||||
running the source code. The full URLs of the libraries should | ||||
be given. A single Javascript library URL can also be given as a | ||||
string. | ||||
css: : list or str | ||||
A sequence of css files to load before running the source code. | ||||
Kyle Kelley
|
r10212 | The full URLs of the css files should be given. A single css URL | ||
Brian Granger
|
r6120 | can also be given as a string. | ||
""" | ||||
Srinivas Reddy Thatiparthy
|
r23037 | if isinstance(lib, str): | ||
Brian Granger
|
r6120 | lib = [lib] | ||
elif lib is None: | ||||
lib = [] | ||||
Srinivas Reddy Thatiparthy
|
r23037 | if isinstance(css, str): | ||
Brian Granger
|
r6120 | css = [css] | ||
elif css is None: | ||||
css = [] | ||||
if not isinstance(lib, (list,tuple)): | ||||
raise TypeError('expected sequence, got: %r' % lib) | ||||
if not isinstance(css, (list,tuple)): | ||||
raise TypeError('expected sequence, got: %r' % css) | ||||
self.lib = lib | ||||
self.css = css | ||||
Grant Nestor
|
r23816 | super(Javascript, self).__init__(data=data, url=url, filename=filename) | ||
Brian Granger
|
r6120 | |||
Brian E. Granger
|
r4526 | def _repr_javascript_(self): | ||
Brian Granger
|
r6120 | r = '' | ||
for c in self.css: | ||||
Matthias Bussonnier
|
r23524 | r += _css_t % c | ||
Brian Granger
|
r6120 | for l in self.lib: | ||
Matthias Bussonnier
|
r23524 | r += _lib_t1 % l | ||
Brian Granger
|
r6120 | r += self.data | ||
Matthias Bussonnier
|
r23524 | r += _lib_t2*len(self.lib) | ||
Brian Granger
|
r6120 | return r | ||
Brian E. Granger
|
r4526 | |||
MinRK
|
r10049 | # constants for identifying png/jpeg data | ||
_PNG = b'\x89PNG\r\n\x1a\n' | ||||
_JPEG = b'\xff\xd8' | ||||
Brian E. Granger
|
r4526 | |||
MinRK
|
r10803 | def _pngxy(data): | ||
"""read the (width, height) from a PNG header""" | ||||
ihdr = data.index(b'IHDR') | ||||
# next 8 bytes are width/height | ||||
Grant Nestor
|
r23821 | return struct.unpack('>ii', data[ihdr+4:ihdr+12]) | ||
MinRK
|
r10803 | |||
def _jpegxy(data): | ||||
"""read the (width, height) from a JPEG header""" | ||||
# adapted from http://www.64lines.com/jpeg-width-height | ||||
Adam Eury
|
r22947 | |||
MinRK
|
r10803 | idx = 4 | ||
while True: | ||||
block_size = struct.unpack('>H', data[idx:idx+2])[0] | ||||
idx = idx + block_size | ||||
if data[idx:idx+2] == b'\xFF\xC0': | ||||
# found Start of Frame | ||||
iSOF = idx | ||||
break | ||||
else: | ||||
# read another block | ||||
idx += 2 | ||||
Min RK
|
r23851 | h, w = struct.unpack('>HH', data[iSOF+5:iSOF+9]) | ||
return w, h | ||||
Min RK
|
r23849 | |||
Grant Nestor
|
r23821 | def _gifxy(data): | ||
"""read the (width, height) from a GIF header""" | ||||
return struct.unpack('<HH', data[6:10]) | ||||
MinRK
|
r10803 | |||
Min RK
|
r23849 | |||
Brian E. Granger
|
r4528 | class Image(DisplayObject): | ||
_read_flags = 'rb' | ||||
Jerry Fowler
|
r8077 | _FMT_JPEG = u'jpeg' | ||
_FMT_PNG = u'png' | ||||
Grant Nestor
|
r23814 | _FMT_GIF = u'gif' | ||
_ACCEPTABLE_EMBEDDINGS = [_FMT_JPEG, _FMT_PNG, _FMT_GIF] | ||||
Min RK
|
r23849 | _MIMETYPES = { | ||
_FMT_PNG: 'image/png', | ||||
_FMT_JPEG: 'image/jpeg', | ||||
_FMT_GIF: 'image/gif', | ||||
} | ||||
Brian E. Granger
|
r4528 | |||
Min RK
|
r21818 | def __init__(self, data=None, url=None, filename=None, format=None, | ||
Min RK
|
r20986 | embed=None, width=None, height=None, retina=False, | ||
unconfined=False, metadata=None): | ||||
Grant Nestor
|
r23814 | """Create a PNG/JPEG/GIF image object given raw data. | ||
Brian E. Granger
|
r4528 | |||
Andrew Mark
|
r12557 | When this object is returned by an input cell or passed to the | ||
Brian E. Granger
|
r4528 | display function, it will result in the image being displayed | ||
in the frontend. | ||||
Parameters | ||||
---------- | ||||
data : unicode, str or bytes | ||||
MinRK
|
r10619 | The raw image data or a URL or filename to load the data from. | ||
This always results in embedded image data. | ||||
Brian E. Granger
|
r4528 | url : unicode | ||
MinRK
|
r10619 | A URL to download the data from. If you specify `url=`, | ||
the image data will not be embedded unless you also specify `embed=True`. | ||||
Brian E. Granger
|
r4528 | filename : unicode | ||
Path to a local file to load the data from. | ||||
MinRK
|
r10619 | Images from a file are always embedded. | ||
Brian E. Granger
|
r4528 | format : unicode | ||
Grant Nestor
|
r23814 | The format of the image data (png/jpeg/jpg/gif). If a filename or URL is given | ||
Brian E. Granger
|
r4528 | for format will be inferred from the filename extension. | ||
embed : bool | ||||
Matthias BUSSONNIER
|
r6529 | Should the image data be embedded using a data URI (True) or be | ||
loaded using an <img> tag. Set this to True if you want the image | ||||
to be viewable later with no internet connection in the notebook. | ||||
Default is `True`, unless the keyword argument `url` is set, then | ||||
default value is `False`. | ||||
Note that QtConsole is not able to display images if `embed` is set to `False` | ||||
Jerry Fowler
|
r8077 | width : int | ||
nvdv
|
r22394 | Width in pixels to which to constrain the image in html | ||
Jerry Fowler
|
r8077 | height : int | ||
nvdv
|
r22394 | Height in pixels to which to constrain the image in html | ||
MinRK
|
r10803 | retina : bool | ||
Automatically set the width and height to half of the measured | ||||
width and height. | ||||
This only works for embedded images because it reads the width/height | ||||
from image data. | ||||
For non-embedded images, you can just set the desired display width | ||||
and height directly. | ||||
Min RK
|
r20986 | unconfined: bool | ||
Set unconfined=True to disable max-width confinement of the image. | ||||
metadata: dict | ||||
Specify extra metadata to attach to the image. | ||||
Matthias BUSSONNIER
|
r6529 | |||
Examples | ||||
-------- | ||||
MinRK
|
r10619 | # embedded image data, works in qtconsole and notebook | ||
# when passed positionally, the first arg can be any of raw image data, | ||||
# a URL, or a filename from which to load image data. | ||||
# The result is always embedding image data for inline images. | ||||
Matthias BUSSONNIER
|
r6529 | Image('http://www.google.fr/images/srpr/logo3w.png') | ||
MinRK
|
r10619 | Image('/path/to/image.jpg') | ||
Image(b'RAW_PNG_DATA...') | ||||
Matthias BUSSONNIER
|
r6529 | |||
MinRK
|
r10619 | # Specifying Image(url=...) does not embed the image data, | ||
# it only generates `<img>` tag with a link to the source. | ||||
# This will not work in the qtconsole or offline. | ||||
Matthias BUSSONNIER
|
r6529 | Image(url='http://www.google.fr/images/srpr/logo3w.png') | ||
Brian E. Granger
|
r4528 | """ | ||
if filename is not None: | ||||
ext = self._find_ext(filename) | ||||
elif url is not None: | ||||
ext = self._find_ext(url) | ||||
Jerry Fowler
|
r8077 | elif data is None: | ||
raise ValueError("No image data found. Expecting filename, url, or data.") | ||||
Srinivas Reddy Thatiparthy
|
r23037 | elif isinstance(data, str) and ( | ||
MinRK
|
r10050 | data.startswith('http') or _safe_exists(data) | ||
): | ||||
Brian E. Granger
|
r4528 | ext = self._find_ext(data) | ||
else: | ||||
ext = None | ||||
Jerry Fowler
|
r8077 | |||
Min RK
|
r21818 | if format is None: | ||
if ext is not None: | ||||
if ext == u'jpg' or ext == u'jpeg': | ||||
format = self._FMT_JPEG | ||||
Thomas Kluyver
|
r23944 | elif ext == u'png': | ||
Min RK
|
r21818 | format = self._FMT_PNG | ||
Thomas Kluyver
|
r23944 | elif ext == u'gif': | ||
Grant Nestor
|
r23818 | format = self._FMT_GIF | ||
Min RK
|
r21818 | else: | ||
format = ext.lower() | ||||
elif isinstance(data, bytes): | ||||
# infer image type from image data header, | ||||
# only if format has not been specified. | ||||
if data[:2] == _JPEG: | ||||
format = self._FMT_JPEG | ||||
# failed to detect format, default png | ||||
if format is None: | ||||
Grant Nestor
|
r23814 | format = self._FMT_PNG | ||
Jerry Fowler
|
r8077 | |||
Matthias Bussonnier
|
r22013 | if format.lower() == 'jpg': | ||
# jpg->jpeg | ||||
format = self._FMT_JPEG | ||||
Min RK
|
r23849 | |||
Srinivas Reddy Thatiparthy
|
r23044 | self.format = format.lower() | ||
Matthias BUSSONNIER
|
r6529 | self.embed = embed if embed is not None else (url is None) | ||
Jerry Fowler
|
r8077 | |||
if self.embed and self.format not in self._ACCEPTABLE_EMBEDDINGS: | ||||
raise ValueError("Cannot embed the '%s' image format" % (self.format)) | ||||
Min RK
|
r23849 | if self.embed: | ||
self._mimetype = self._MIMETYPES.get(self.format) | ||||
Jerry Fowler
|
r8077 | self.width = width | ||
self.height = height | ||||
MinRK
|
r10803 | self.retina = retina | ||
Min RK
|
r20986 | self.unconfined = unconfined | ||
M Pacer
|
r23746 | super(Image, self).__init__(data=data, url=url, filename=filename, | ||
metadata=metadata) | ||||
if self.width is None and self.metadata.get('width', {}): | ||||
self.width = metadata['width'] | ||||
if self.height is None and self.metadata.get('height', {}): | ||||
self.height = metadata['height'] | ||||
Adam Eury
|
r22947 | |||
MinRK
|
r10803 | if retina: | ||
self._retina_shape() | ||||
Adam Eury
|
r22947 | |||
Min RK
|
r23849 | |||
MinRK
|
r10803 | def _retina_shape(self): | ||
"""load pixel-doubled width and height from image data""" | ||||
if not self.embed: | ||||
return | ||||
Grant Nestor
|
r23814 | if self.format == self._FMT_PNG: | ||
MinRK
|
r10803 | w, h = _pngxy(self.data) | ||
Grant Nestor
|
r23814 | elif self.format == self._FMT_JPEG: | ||
MinRK
|
r10803 | w, h = _jpegxy(self.data) | ||
Grant Nestor
|
r23818 | elif self.format == self._FMT_GIF: | ||
Grant Nestor
|
r23821 | w, h = _gifxy(self.data) | ||
MinRK
|
r10803 | else: | ||
# retina only supports png | ||||
return | ||||
self.width = w // 2 | ||||
self.height = h // 2 | ||||
Brian E. Granger
|
r4528 | |||
def reload(self): | ||||
"""Reload the raw data from file or URL.""" | ||||
if self.embed: | ||||
super(Image,self).reload() | ||||
MinRK
|
r10803 | if self.retina: | ||
self._retina_shape() | ||||
Brian E. Granger
|
r4528 | |||
def _repr_html_(self): | ||||
if not self.embed: | ||||
Min RK
|
r20986 | width = height = klass = '' | ||
Jerry Fowler
|
r8077 | if self.width: | ||
width = ' width="%d"' % self.width | ||||
if self.height: | ||||
height = ' height="%d"' % self.height | ||||
Min RK
|
r20986 | if self.unconfined: | ||
klass = ' class="unconfined"' | ||||
return u'<img src="{url}"{width}{height}{klass}/>'.format( | ||||
url=self.url, | ||||
width=width, | ||||
height=height, | ||||
klass=klass, | ||||
) | ||||
Dan McDougall
|
r10490 | |||
Min RK
|
r23849 | def _repr_mimebundle_(self, include=None, exclude=None): | ||
"""Return the image as a mimebundle | ||||
Any new mimetype support should be implemented here. | ||||
""" | ||||
if self.embed: | ||||
mimetype = self._mimetype | ||||
data, metadata = self._data_and_metadata(always_both=True) | ||||
if metadata: | ||||
metadata = {mimetype: metadata} | ||||
return {mimetype: data}, metadata | ||||
else: | ||||
return {'text/html': self._repr_html_()} | ||||
def _data_and_metadata(self, always_both=False): | ||||
MinRK
|
r10444 | """shortcut for returning metadata with shape information, if defined""" | ||
Nick Tallant
|
r24989 | try: | ||
b64_data = b2a_base64(self.data).decode('ascii') | ||||
except TypeError: | ||||
raise FileNotFoundError( | ||||
"No such file or directory: '%s'" % (self.data)) | ||||
MinRK
|
r10444 | md = {} | ||
M Pacer
|
r23746 | if self.metadata: | ||
md.update(self.metadata) | ||||
MinRK
|
r10444 | if self.width: | ||
md['width'] = self.width | ||||
if self.height: | ||||
md['height'] = self.height | ||||
Min RK
|
r20986 | if self.unconfined: | ||
md['unconfined'] = self.unconfined | ||||
Min RK
|
r23849 | if md or always_both: | ||
Min RK
|
r23835 | return b64_data, md | ||
MinRK
|
r10444 | else: | ||
Min RK
|
r23835 | return b64_data | ||
Brian E. Granger
|
r4528 | |||
def _repr_png_(self): | ||||
Nick Tallant
|
r24990 | if self.embed and self.format == self._FMT_PNG: | ||
return self._data_and_metadata() | ||||
Brian E. Granger
|
r4528 | |||
def _repr_jpeg_(self): | ||||
Nick Tallant
|
r24990 | if self.embed and self.format == self._FMT_JPEG: | ||
return self._data_and_metadata() | ||||
Brian E. Granger
|
r4528 | |||
def _find_ext(self, s): | ||||
Srinivas Reddy Thatiparthy
|
r23044 | return s.split('.')[-1].lower() | ||
Brian Granger
|
r5080 | |||
Min RK
|
r23849 | |||
Daniel Wehner
|
r17001 | class Video(DisplayObject): | ||
Dominic Kuang
|
r24623 | def __init__(self, data=None, url=None, filename=None, embed=False, | ||
mimetype=None, width=None, height=None): | ||||
Daniel Wehner
|
r17001 | """Create a video object given raw data or an URL. | ||
When this object is returned by an input cell or passed to the | ||||
display function, it will result in the video being displayed | ||||
in the frontend. | ||||
Parameters | ||||
---------- | ||||
data : unicode, str or bytes | ||||
Min RK
|
r22116 | The raw video data or a URL or filename to load the data from. | ||
Raw data will require passing `embed=True`. | ||||
Daniel Wehner
|
r17001 | url : unicode | ||
Min RK
|
r22116 | A URL for the video. If you specify `url=`, | ||
the image data will not be embedded. | ||||
Daniel Wehner
|
r17001 | filename : unicode | ||
Min RK
|
r22116 | Path to a local file containing the video. | ||
Will be interpreted as a local URL unless `embed=True`. | ||||
Daniel Wehner
|
r17001 | embed : bool | ||
Min RK
|
r22116 | Should the video be embedded using a data URI (True) or be | ||
loaded using a <video> tag (False). | ||||
Daniel Wehner
|
r17001 | |||
Min RK
|
r22116 | Since videos are large, embedding them should be avoided, if possible. | ||
You must confirm embedding as your intention by passing `embed=True`. | ||||
Local files can be displayed with URLs without embedding the content, via:: | ||||
Video('./video.mp4') | ||||
Daniel Wehner
|
r17001 | |||
mimetype: unicode | ||||
Min RK
|
r22116 | Specify the mimetype for embedded videos. | ||
Default will be guessed from file extension, if available. | ||||
Dominic Kuang
|
r24623 | width : int | ||
Width in pixels to which to constrain the video in HTML. | ||||
If not supplied, defaults to the width of the video. | ||||
height : int | ||||
Height in pixels to which to constrain the video in html. | ||||
If not supplied, defaults to the height of the video. | ||||
Min RK
|
r22115 | |||
Daniel Wehner
|
r17001 | Examples | ||
-------- | ||||
Min RK
|
r22115 | |||
Daniel Wehner
|
r17001 | Video('https://archive.org/download/Sita_Sings_the_Blues/Sita_Sings_the_Blues_small.mp4') | ||
Video('path/to/video.mp4') | ||||
Min RK
|
r22116 | Video('path/to/video.mp4', embed=True) | ||
Video(b'raw-videodata', embed=True) | ||||
Daniel Wehner
|
r17001 | """ | ||
Srinivas Reddy Thatiparthy
|
r23037 | if url is None and isinstance(data, str) and data.startswith(('http:', 'https:')): | ||
Daniel Wehner
|
r17001 | url = data | ||
data = None | ||||
elif os.path.exists(data): | ||||
filename = data | ||||
data = None | ||||
Adam Eury
|
r22947 | |||
Min RK
|
r22116 | if data and not embed: | ||
msg = ''.join([ | ||||
"To embed videos, you must pass embed=True ", | ||||
"(this may make your notebook files huge)\n", | ||||
"Consider passing Video(url='...')", | ||||
]) | ||||
raise ValueError(msg) | ||||
Daniel Wehner
|
r17001 | |||
self.mimetype = mimetype | ||||
Min RK
|
r22116 | self.embed = embed | ||
Dominic Kuang
|
r24623 | self.width = width | ||
self.height = height | ||||
Daniel Wehner
|
r17001 | super(Video, self).__init__(data=data, url=url, filename=filename) | ||
def _repr_html_(self): | ||||
Dominic Kuang
|
r24623 | width = height = '' | ||
if self.width: | ||||
width = ' width="%d"' % self.width | ||||
if self.height: | ||||
height = ' height="%d"' % self.height | ||||
Daniel Wehner
|
r17001 | # External URLs and potentially local files are not embedded into the | ||
# notebook output. | ||||
if not self.embed: | ||||
url = self.url if self.url is not None else self.filename | ||||
Dominic Kuang
|
r24623 | output = """<video src="{0}" controls {1} {2}> | ||
Daniel Wehner
|
r17001 | Your browser does not support the <code>video</code> element. | ||
Dominic Kuang
|
r24623 | </video>""".format(url, width, height) | ||
Daniel Wehner
|
r17001 | return output | ||
Adam Eury
|
r22947 | |||
Min RK
|
r22116 | # Embedded videos are base64-encoded. | ||
Min RK
|
r22115 | mimetype = self.mimetype | ||
Daniel Wehner
|
r17001 | if self.filename is not None: | ||
Min RK
|
r22115 | if not mimetype: | ||
mimetype, _ = mimetypes.guess_type(self.filename) | ||||
Adam Eury
|
r22947 | |||
Min RK
|
r22115 | with open(self.filename, 'rb') as f: | ||
video = f.read() | ||||
else: | ||||
video = self.data | ||||
Srinivas Reddy Thatiparthy
|
r23044 | if isinstance(video, str): | ||
Min RK
|
r22115 | # unicode input is already b64-encoded | ||
b64_video = video | ||||
Daniel Wehner
|
r17001 | else: | ||
Min RK
|
r23835 | b64_video = b2a_base64(video).decode('ascii').rstrip() | ||
Adam Eury
|
r22947 | |||
Dominic Kuang
|
r24623 | output = """<video controls {0} {1}> | ||
<source src="data:{2};base64,{3}" type="{2}"> | ||||
Daniel Wehner
|
r17001 | Your browser does not support the video tag. | ||
Dominic Kuang
|
r24623 | </video>""".format(width, height, mimetype, b64_video) | ||
Daniel Wehner
|
r17001 | return output | ||
def reload(self): | ||||
# TODO | ||||
pass | ||||
Brian Granger
|
r5080 | |||
Jonathan Frederic
|
r12592 | def clear_output(wait=False): | ||
"""Clear the output of the current cell receiving output. | ||||
Parameters | ||||
---------- | ||||
wait : bool [default: false] | ||||
Wait to clear the output until new output is available to replace it.""" | ||||
Brian Granger
|
r5080 | from IPython.core.interactiveshell import InteractiveShell | ||
MinRK
|
r6422 | if InteractiveShell.initialized(): | ||
Jonathan Frederic
|
r12592 | InteractiveShell.instance().display_pub.clear_output(wait) | ||
MinRK
|
r6422 | else: | ||
Thomas Kluyver
|
r22192 | print('\033[2K\r', end='') | ||
sys.stdout.flush() | ||||
print('\033[2K\r', end='') | ||||
sys.stderr.flush() | ||||
Brian E. Granger
|
r15124 | |||
Brian E. Granger
|
r15126 | @skip_doctest | ||
def set_matplotlib_formats(*formats, **kwargs): | ||||
"""Select figure formats for the inline backend. Optionally pass quality for JPEG. | ||||
For example, this enables PNG and JPEG output with a JPEG quality of 90%:: | ||||
In [1]: set_matplotlib_formats('png', 'jpeg', quality=90) | ||||
To set this in your config files use the following:: | ||||
Adam Eury
|
r22947 | |||
MinRK
|
r15393 | c.InlineBackend.figure_formats = {'png', 'jpeg'} | ||
c.InlineBackend.print_figure_kwargs.update({'quality' : 90}) | ||||
Brian E. Granger
|
r15124 | |||
Parameters | ||||
Brian E. Granger
|
r15126 | ---------- | ||
MinRK
|
r15393 | *formats : strs | ||
One or more figure formats to enable: 'png', 'retina', 'jpeg', 'svg', 'pdf'. | ||||
**kwargs : | ||||
Keyword args will be relayed to ``figure.canvas.print_figure``. | ||||
Brian E. Granger
|
r15124 | """ | ||
from IPython.core.interactiveshell import InteractiveShell | ||||
from IPython.core.pylabtools import select_figure_formats | ||||
MinRK
|
r15393 | # build kwargs, starting with InlineBackend config | ||
kw = {} | ||||
Min RK
|
r21337 | from ipykernel.pylab.config import InlineBackend | ||
MinRK
|
r15393 | cfg = InlineBackend.instance() | ||
kw.update(cfg.print_figure_kwargs) | ||||
kw.update(**kwargs) | ||||
Brian E. Granger
|
r15124 | shell = InteractiveShell.instance() | ||
MinRK
|
r15393 | select_figure_formats(shell, formats, **kw) | ||
Brian E. Granger
|
r15124 | |||
Brian E. Granger
|
r15126 | @skip_doctest | ||
MinRK
|
r15395 | def set_matplotlib_close(close=True): | ||
Brian E. Granger
|
r15197 | """Set whether the inline backend closes all figures automatically or not. | ||
Adam Eury
|
r22947 | |||
Brian E. Granger
|
r15126 | By default, the inline backend used in the IPython Notebook will close all | ||
matplotlib figures automatically after each cell is run. This means that | ||||
plots in different cells won't interfere. Sometimes, you may want to make | ||||
a plot in one cell and then refine it in later cells. This can be accomplished | ||||
by:: | ||||
Adam Eury
|
r22947 | |||
Brian E. Granger
|
r15126 | In [1]: set_matplotlib_close(False) | ||
Adam Eury
|
r22947 | |||
Brian E. Granger
|
r15126 | To set this in your config files use the following:: | ||
Adam Eury
|
r22947 | |||
Brian E. Granger
|
r15126 | c.InlineBackend.close_figures = False | ||
Adam Eury
|
r22947 | |||
Brian E. Granger
|
r15126 | Parameters | ||
---------- | ||||
close : bool | ||||
Should all matplotlib figures be automatically closed after each cell is | ||||
run? | ||||
""" | ||||
Min RK
|
r21337 | from ipykernel.pylab.config import InlineBackend | ||
MinRK
|
r15395 | cfg = InlineBackend.instance() | ||
cfg.close_figures = close | ||||