1{
2 "cells": [
3  {
4   "cell_type": "markdown",
5   "metadata": {},
6   "source": [
7    "# Using SPICE kernels to study HERA trajectory\n",
8    "In this tutorial we will use the spice kernels available for download at ftp://spiftp.esac.esa.int/data/SPICE/ to study the interplanetary trajectory of the HERA mission\n"
9   ]
10  },
11  {
12   "cell_type": "code",
13   "execution_count": 1,
14   "metadata": {},
15   "outputs": [],
16   "source": [
17    "# We need to import some toolboxes \n",
18    "import pykep as pk\n",
19    "from matplotlib import pyplot as plt\n",
20    "import numpy as np\n",
21    "%matplotlib notebook"
22   ]
23  },
24  {
25   "cell_type": "markdown",
26   "metadata": {},
27   "source": [
28    "## Loading the SPICE kernels"
29   ]
30  },
31  {
32   "cell_type": "code",
33   "execution_count": 2,
34   "metadata": {},
35   "outputs": [],
36   "source": [
37    "# Assuming the kernels have been downloaded and\n",
38    "# placed in the same directory of this notebook\n",
39    "# 1 - We load the solar system ephemerides\n",
40    "pk.util.load_spice_kernel(\"de432s.bsp\")\n",
41    "# 2 - We load the ephemerides for HERA trajectory\n",
42    "pk.util.load_spice_kernel(\"HERA_sc_LPC_EMA_2024c.bsp\")\n",
43    "# 3 - We load the ephemerides for the Didymos position\n",
44    "pk.util.load_spice_kernel(\"didymos_hor_200101_300101_v01.bsp\")"
45   ]
46  },
47  {
48   "cell_type": "markdown",
49   "metadata": {},
50   "source": [
51    "## A simple use"
52   ]
53  },
54  {
55   "cell_type": "code",
56   "execution_count": 3,
57   "metadata": {},
58   "outputs": [
59    {
60     "name": "stdout",
61     "output_type": "stream",
62     "text": [
63      "2026-Jan-04 12:00:00\n"
64     ]
65    }
66   ],
67   "source": [
68    "# We now need to define an epoch (date) to query for positions of the various bodies\n",
69    "# Keep in mind that the kernels do not contain the information for all epochs, only for some\n",
70    "# ranges. You will get an error if you query outside of these bounds\n",
71    "ep = pk.epoch_from_string(\"2026-Jan-04 12:00:00\")\n",
72    "print(ep)"
73   ]
74  },
75  {
76   "cell_type": "code",
77   "execution_count": 4,
78   "metadata": {},
79   "outputs": [],
80   "source": [
81    "# The code assigned to HERA seems to be -999, so we stick to it. Here we create a fake \"planet\"\n",
82    "# describing the position of HERA w.r.t. the SUN in the J2000 ecliptic frame and no aberration.\n",
83    "# (ignore the parameters 100 .. 100 and 100 they describe [in pykep syntax] the mass, dimensions and safe\n",
84    "# distance of planets which in this case makes no sense)\n",
85    "hera = pk.planet.spice('-999', 'SUN', 'ECLIPJ2000', 'NONE', pk.MU_SUN, 100, 100, 100) "
86   ]
87  },
88  {
89   "cell_type": "code",
90   "execution_count": 5,
91   "metadata": {},
92   "outputs": [
93    {
94     "name": "stdout",
95     "output_type": "stream",
96     "text": [
97      "Position vector at epoch:  (-270759976042.35202, -224408729981.53754, 11161707384.613232)\n",
98      "Velocity vector at epoch:  (10603.550394243548, -10862.635786874001, -424.0057060846829)\n"
99     ]
100    }
101   ],
102   "source": [
103    "r, v = hera.eph(ep)\n",
104    "print(\"Position vector at epoch: \", r)\n",
105    "print(\"Velocity vector at epoch: \", v)"
106   ]
107  },
108  {
109   "cell_type": "code",
110   "execution_count": 6,
111   "metadata": {},
112   "outputs": [],
113   "source": [
114    "# We want also the position with respect to didymos? The code assigned to Didymos seems \n",
115    "# to be 2065803 (apparently using names is not fashionable). So we define a new fake planet object\n",
116    "hera_wrt_didymos = pk.planet.spice('-999', '2065803', 'ECLIPJ2000', 'NONE', pk.MU_SUN, 100, 100, 100) "
117   ]
118  },
119  {
120   "cell_type": "code",
121   "execution_count": 7,
122   "metadata": {},
123   "outputs": [
124    {
125     "name": "stdout",
126     "output_type": "stream",
127     "text": [
128      "Position vector at epoch:  (1847403369.0042496, -22574976660.978043, -865934834.397764)\n",
129      "Velocity vector at epoch:  (861.9916442662294, 1239.5955464098067, 342.97888719149586)\n"
130     ]
131    }
132   ],
133   "source": [
134    "r, v = hera_wrt_didymos.eph(ep)\n",
135    "print(\"Position vector at epoch: \", r)\n",
136    "print(\"Velocity vector at epoch: \", v)"
137   ]
138  },
139  {
140   "cell_type": "code",
141   "execution_count": 8,
142   "metadata": {},
143   "outputs": [],
144   "source": [
145    "# We want also the position with respect to the Earth? We define a new fake planet object\n",
146    "hera_wrt_earth = pk.planet.spice('-999', 'EARTH', 'ECLIPJ2000', 'NONE', pk.MU_SUN, 100, 100, 100) "
147   ]
148  },
149  {
150   "cell_type": "code",
151   "execution_count": 9,
152   "metadata": {},
153   "outputs": [
154    {
155     "name": "stdout",
156     "output_type": "stream",
157     "text": [
158      "Position vector at epoch:  (-235735253271.4893, -367278379081.661, 11170314994.444817)\n",
159      "Velocity vector at epoch:  (40007.68764017193, -3663.4004080195828, -425.42168268912485)\n"
160     ]
161    }
162   ],
163   "source": [
164    "r, v = hera_wrt_earth.eph(ep)\n",
165    "print(\"Position vector at epoch: \", r)\n",
166    "print(\"Velocity vector at epoch: \", v)"
167   ]
168  },
169  {
170   "cell_type": "markdown",
171   "metadata": {},
172   "source": [
173    "## Visualizing the whole trajectory"
174   ]
175  },
176  {
177   "cell_type": "code",
178   "execution_count": 10,
179   "metadata": {},
180   "outputs": [],
181   "source": [
182    "# The documentation of HERA_sc_LPC_EMA_2024c.bsp kernel reveals that the kernel contains the following data:\n",
183    "#\n",
184    "# HERA_sc_LPC_EMA_2024c.bsp\n",
185    "# Bodies                                    Start of Interval (UTC)         End of Interval (UTC)\n",
186    "# -------                                   -----------------------------   -----------------------------\n",
187    "# -999 HERA w.r.t. 4 MARS BARYCENTER        2025-MAR-17 10:05:00.284        2025-MAR-18 22:29:36.525\n",
188    "# -999 HERA w.r.t. 10 SUN                   2024-OCT-27 11:26:16.873        2025-MAR-17 10:05:00.284\n",
189    "#                                           2025-MAR-18 22:29:36.525        2026-DEC-28 11:58:50.816\n",
190    "# -999 HERA w.r.t. 399 EARTH                2024-OCT-25 11:58:50.817        2024-OCT-27 11:26:16.873\n",
191    "#\n",
192    "# In reality it seems the data starts at 2024-OCT-25 12:00:00. So we use that as launch date (anything before seems\n",
193    "# to be absent in the data)\n",
194    "\n",
195    "launch = pk.epoch_from_string(\"2024-OCT-25 12:00:00\")\n",
196    "arrival = pk.epoch_from_string(\"2026-DEC-28 11:58:50.816\")\n",
197    "flyby = pk.epoch_from_string(\"2025-MAR-17 10:05:00.284\")\n"
198   ]
199  },
200  {
201   "cell_type": "code",
202   "execution_count": 11,
203   "metadata": {},
204   "outputs": [
205    {
206     "data": {
207      "application/javascript": [
208       "/* Put everything inside the global mpl namespace */\n",
209       "window.mpl = {};\n",
210       "\n",
211       "\n",
212       "mpl.get_websocket_type = function() {\n",
213       "    if (typeof(WebSocket) !== 'undefined') {\n",
214       "        return WebSocket;\n",
215       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
216       "        return MozWebSocket;\n",
217       "    } else {\n",
218       "        alert('Your browser does not have WebSocket support. ' +\n",
219       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
220       "              'Firefox 4 and 5 are also supported but you ' +\n",
221       "              'have to enable WebSockets in about:config.');\n",
222       "    };\n",
223       "}\n",
224       "\n",
225       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
226       "    this.id = figure_id;\n",
227       "\n",
228       "    this.ws = websocket;\n",
229       "\n",
230       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
231       "\n",
232       "    if (!this.supports_binary) {\n",
233       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
234       "        if (warnings) {\n",
235       "            warnings.style.display = 'block';\n",
236       "            warnings.textContent = (\n",
237       "                \"This browser does not support binary websocket messages. \" +\n",
238       "                    \"Performance may be slow.\");\n",
239       "        }\n",
240       "    }\n",
241       "\n",
242       "    this.imageObj = new Image();\n",
243       "\n",
244       "    this.context = undefined;\n",
245       "    this.message = undefined;\n",
246       "    this.canvas = undefined;\n",
247       "    this.rubberband_canvas = undefined;\n",
248       "    this.rubberband_context = undefined;\n",
249       "    this.format_dropdown = undefined;\n",
250       "\n",
251       "    this.image_mode = 'full';\n",
252       "\n",
253       "    this.root = $('<div/>');\n",
254       "    this._root_extra_style(this.root)\n",
255       "    this.root.attr('style', 'display: inline-block');\n",
256       "\n",
257       "    $(parent_element).append(this.root);\n",
258       "\n",
259       "    this._init_header(this);\n",
260       "    this._init_canvas(this);\n",
261       "    this._init_toolbar(this);\n",
262       "\n",
263       "    var fig = this;\n",
264       "\n",
265       "    this.waiting = false;\n",
266       "\n",
267       "    this.ws.onopen =  function () {\n",
268       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
269       "            fig.send_message(\"send_image_mode\", {});\n",
270       "            if (mpl.ratio != 1) {\n",
271       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
272       "            }\n",
273       "            fig.send_message(\"refresh\", {});\n",
274       "        }\n",
275       "\n",
276       "    this.imageObj.onload = function() {\n",
277       "            if (fig.image_mode == 'full') {\n",
278       "                // Full images could contain transparency (where diff images\n",
279       "                // almost always do), so we need to clear the canvas so that\n",
280       "                // there is no ghosting.\n",
281       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
282       "            }\n",
283       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
284       "        };\n",
285       "\n",
286       "    this.imageObj.onunload = function() {\n",
287       "        fig.ws.close();\n",
288       "    }\n",
289       "\n",
290       "    this.ws.onmessage = this._make_on_message_function(this);\n",
291       "\n",
292       "    this.ondownload = ondownload;\n",
293       "}\n",
294       "\n",
295       "mpl.figure.prototype._init_header = function() {\n",
296       "    var titlebar = $(\n",
297       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
298       "        'ui-helper-clearfix\"/>');\n",
299       "    var titletext = $(\n",
300       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
301       "        'text-align: center; padding: 3px;\"/>');\n",
302       "    titlebar.append(titletext)\n",
303       "    this.root.append(titlebar);\n",
304       "    this.header = titletext[0];\n",
305       "}\n",
306       "\n",
307       "\n",
308       "\n",
309       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
310       "\n",
311       "}\n",
312       "\n",
313       "\n",
314       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
315       "\n",
316       "}\n",
317       "\n",
318       "mpl.figure.prototype._init_canvas = function() {\n",
319       "    var fig = this;\n",
320       "\n",
321       "    var canvas_div = $('<div/>');\n",
322       "\n",
323       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
324       "\n",
325       "    function canvas_keyboard_event(event) {\n",
326       "        return fig.key_event(event, event['data']);\n",
327       "    }\n",
328       "\n",
329       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
330       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
331       "    this.canvas_div = canvas_div\n",
332       "    this._canvas_extra_style(canvas_div)\n",
333       "    this.root.append(canvas_div);\n",
334       "\n",
335       "    var canvas = $('<canvas/>');\n",
336       "    canvas.addClass('mpl-canvas');\n",
337       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
338       "\n",
339       "    this.canvas = canvas[0];\n",
340       "    this.context = canvas[0].getContext(\"2d\");\n",
341       "\n",
342       "    var backingStore = this.context.backingStorePixelRatio ||\n",
343       "\tthis.context.webkitBackingStorePixelRatio ||\n",
344       "\tthis.context.mozBackingStorePixelRatio ||\n",
345       "\tthis.context.msBackingStorePixelRatio ||\n",
346       "\tthis.context.oBackingStorePixelRatio ||\n",
347       "\tthis.context.backingStorePixelRatio || 1;\n",
348       "\n",
349       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
350       "\n",
351       "    var rubberband = $('<canvas/>');\n",
352       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
353       "\n",
354       "    var pass_mouse_events = true;\n",
355       "\n",
356       "    canvas_div.resizable({\n",
357       "        start: function(event, ui) {\n",
358       "            pass_mouse_events = false;\n",
359       "        },\n",
360       "        resize: function(event, ui) {\n",
361       "            fig.request_resize(ui.size.width, ui.size.height);\n",
362       "        },\n",
363       "        stop: function(event, ui) {\n",
364       "            pass_mouse_events = true;\n",
365       "            fig.request_resize(ui.size.width, ui.size.height);\n",
366       "        },\n",
367       "    });\n",
368       "\n",
369       "    function mouse_event_fn(event) {\n",
370       "        if (pass_mouse_events)\n",
371       "            return fig.mouse_event(event, event['data']);\n",
372       "    }\n",
373       "\n",
374       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
375       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
376       "    // Throttle sequential mouse events to 1 every 20ms.\n",
377       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
378       "\n",
379       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
380       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
381       "\n",
382       "    canvas_div.on(\"wheel\", function (event) {\n",
383       "        event = event.originalEvent;\n",
384       "        event['data'] = 'scroll'\n",
385       "        if (event.deltaY < 0) {\n",
386       "            event.step = 1;\n",
387       "        } else {\n",
388       "            event.step = -1;\n",
389       "        }\n",
390       "        mouse_event_fn(event);\n",
391       "    });\n",
392       "\n",
393       "    canvas_div.append(canvas);\n",
394       "    canvas_div.append(rubberband);\n",
395       "\n",
396       "    this.rubberband = rubberband;\n",
397       "    this.rubberband_canvas = rubberband[0];\n",
398       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
399       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
400       "\n",
401       "    this._resize_canvas = function(width, height) {\n",
402       "        // Keep the size of the canvas, canvas container, and rubber band\n",
403       "        // canvas in sync.\n",
404       "        canvas_div.css('width', width)\n",
405       "        canvas_div.css('height', height)\n",
406       "\n",
407       "        canvas.attr('width', width * mpl.ratio);\n",
408       "        canvas.attr('height', height * mpl.ratio);\n",
409       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
410       "\n",
411       "        rubberband.attr('width', width);\n",
412       "        rubberband.attr('height', height);\n",
413       "    }\n",
414       "\n",
415       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
416       "    // upon first draw.\n",
417       "    this._resize_canvas(600, 600);\n",
418       "\n",
419       "    // Disable right mouse context menu.\n",
420       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
421       "        return false;\n",
422       "    });\n",
423       "\n",
424       "    function set_focus () {\n",
425       "        canvas.focus();\n",
426       "        canvas_div.focus();\n",
427       "    }\n",
428       "\n",
429       "    window.setTimeout(set_focus, 100);\n",
430       "}\n",
431       "\n",
432       "mpl.figure.prototype._init_toolbar = function() {\n",
433       "    var fig = this;\n",
434       "\n",
435       "    var nav_element = $('<div/>');\n",
436       "    nav_element.attr('style', 'width: 100%');\n",
437       "    this.root.append(nav_element);\n",
438       "\n",
439       "    // Define a callback function for later on.\n",
440       "    function toolbar_event(event) {\n",
441       "        return fig.toolbar_button_onclick(event['data']);\n",
442       "    }\n",
443       "    function toolbar_mouse_event(event) {\n",
444       "        return fig.toolbar_button_onmouseover(event['data']);\n",
445       "    }\n",
446       "\n",
447       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
448       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
449       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
450       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
451       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
452       "\n",
453       "        if (!name) {\n",
454       "            // put a spacer in here.\n",
455       "            continue;\n",
456       "        }\n",
457       "        var button = $('<button/>');\n",
458       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
459       "                        'ui-button-icon-only');\n",
460       "        button.attr('role', 'button');\n",
461       "        button.attr('aria-disabled', 'false');\n",
462       "        button.click(method_name, toolbar_event);\n",
463       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
464       "\n",
465       "        var icon_img = $('<span/>');\n",
466       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
467       "        icon_img.addClass(image);\n",
468       "        icon_img.addClass('ui-corner-all');\n",
469       "\n",
470       "        var tooltip_span = $('<span/>');\n",
471       "        tooltip_span.addClass('ui-button-text');\n",
472       "        tooltip_span.html(tooltip);\n",
473       "\n",
474       "        button.append(icon_img);\n",
475       "        button.append(tooltip_span);\n",
476       "\n",
477       "        nav_element.append(button);\n",
478       "    }\n",
479       "\n",
480       "    var fmt_picker_span = $('<span/>');\n",
481       "\n",
482       "    var fmt_picker = $('<select/>');\n",
483       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
484       "    fmt_picker_span.append(fmt_picker);\n",
485       "    nav_element.append(fmt_picker_span);\n",
486       "    this.format_dropdown = fmt_picker[0];\n",
487       "\n",
488       "    for (var ind in mpl.extensions) {\n",
489       "        var fmt = mpl.extensions[ind];\n",
490       "        var option = $(\n",
491       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
492       "        fmt_picker.append(option);\n",
493       "    }\n",
494       "\n",
495       "    // Add hover states to the ui-buttons\n",
496       "    $( \".ui-button\" ).hover(\n",
497       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
498       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
499       "    );\n",
500       "\n",
501       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
502       "    nav_element.append(status_bar);\n",
503       "    this.message = status_bar[0];\n",
504       "}\n",
505       "\n",
506       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
507       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
508       "    // which will in turn request a refresh of the image.\n",
509       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
510       "}\n",
511       "\n",
512       "mpl.figure.prototype.send_message = function(type, properties) {\n",
513       "    properties['type'] = type;\n",
514       "    properties['figure_id'] = this.id;\n",
515       "    this.ws.send(JSON.stringify(properties));\n",
516       "}\n",
517       "\n",
518       "mpl.figure.prototype.send_draw_message = function() {\n",
519       "    if (!this.waiting) {\n",
520       "        this.waiting = true;\n",
521       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
522       "    }\n",
523       "}\n",
524       "\n",
525       "\n",
526       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
527       "    var format_dropdown = fig.format_dropdown;\n",
528       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
529       "    fig.ondownload(fig, format);\n",
530       "}\n",
531       "\n",
532       "\n",
533       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
534       "    var size = msg['size'];\n",
535       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
536       "        fig._resize_canvas(size[0], size[1]);\n",
537       "        fig.send_message(\"refresh\", {});\n",
538       "    };\n",
539       "}\n",
540       "\n",
541       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
542       "    var x0 = msg['x0'] / mpl.ratio;\n",
543       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
544       "    var x1 = msg['x1'] / mpl.ratio;\n",
545       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
546       "    x0 = Math.floor(x0) + 0.5;\n",
547       "    y0 = Math.floor(y0) + 0.5;\n",
548       "    x1 = Math.floor(x1) + 0.5;\n",
549       "    y1 = Math.floor(y1) + 0.5;\n",
550       "    var min_x = Math.min(x0, x1);\n",
551       "    var min_y = Math.min(y0, y1);\n",
552       "    var width = Math.abs(x1 - x0);\n",
553       "    var height = Math.abs(y1 - y0);\n",
554       "\n",
555       "    fig.rubberband_context.clearRect(\n",
556       "        0, 0, fig.canvas.width / mpl.ratio, fig.canvas.height / mpl.ratio);\n",
557       "\n",
558       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
559       "}\n",
560       "\n",
561       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
562       "    // Updates the figure title.\n",
563       "    fig.header.textContent = msg['label'];\n",
564       "}\n",
565       "\n",
566       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
567       "    var cursor = msg['cursor'];\n",
568       "    switch(cursor)\n",
569       "    {\n",
570       "    case 0:\n",
571       "        cursor = 'pointer';\n",
572       "        break;\n",
573       "    case 1:\n",
574       "        cursor = 'default';\n",
575       "        break;\n",
576       "    case 2:\n",
577       "        cursor = 'crosshair';\n",
578       "        break;\n",
579       "    case 3:\n",
580       "        cursor = 'move';\n",
581       "        break;\n",
582       "    }\n",
583       "    fig.rubberband_canvas.style.cursor = cursor;\n",
584       "}\n",
585       "\n",
586       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
587       "    fig.message.textContent = msg['message'];\n",
588       "}\n",
589       "\n",
590       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
591       "    // Request the server to send over a new figure.\n",
592       "    fig.send_draw_message();\n",
593       "}\n",
594       "\n",
595       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
596       "    fig.image_mode = msg['mode'];\n",
597       "}\n",
598       "\n",
599       "mpl.figure.prototype.updated_canvas_event = function() {\n",
600       "    // Called whenever the canvas gets updated.\n",
601       "    this.send_message(\"ack\", {});\n",
602       "}\n",
603       "\n",
604       "// A function to construct a web socket function for onmessage handling.\n",
605       "// Called in the figure constructor.\n",
606       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
607       "    return function socket_on_message(evt) {\n",
608       "        if (evt.data instanceof Blob) {\n",
609       "            /* FIXME: We get \"Resource interpreted as Image but\n",
610       "             * transferred with MIME type text/plain:\" errors on\n",
611       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
612       "             * to be part of the websocket stream */\n",
613       "            evt.data.type = \"image/png\";\n",
614       "\n",
615       "            /* Free the memory for the previous frames */\n",
616       "            if (fig.imageObj.src) {\n",
617       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
618       "                    fig.imageObj.src);\n",
619       "            }\n",
620       "\n",
621       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
622       "                evt.data);\n",
623       "            fig.updated_canvas_event();\n",
624       "            fig.waiting = false;\n",
625       "            return;\n",
626       "        }\n",
627       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
628       "            fig.imageObj.src = evt.data;\n",
629       "            fig.updated_canvas_event();\n",
630       "            fig.waiting = false;\n",
631       "            return;\n",
632       "        }\n",
633       "\n",
634       "        var msg = JSON.parse(evt.data);\n",
635       "        var msg_type = msg['type'];\n",
636       "\n",
637       "        // Call the  \"handle_{type}\" callback, which takes\n",
638       "        // the figure and JSON message as its only arguments.\n",
639       "        try {\n",
640       "            var callback = fig[\"handle_\" + msg_type];\n",
641       "        } catch (e) {\n",
642       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
643       "            return;\n",
644       "        }\n",
645       "\n",
646       "        if (callback) {\n",
647       "            try {\n",
648       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
649       "                callback(fig, msg);\n",
650       "            } catch (e) {\n",
651       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
652       "            }\n",
653       "        }\n",
654       "    };\n",
655       "}\n",
656       "\n",
657       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
658       "mpl.findpos = function(e) {\n",
659       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
660       "    var targ;\n",
661       "    if (!e)\n",
662       "        e = window.event;\n",
663       "    if (e.target)\n",
664       "        targ = e.target;\n",
665       "    else if (e.srcElement)\n",
666       "        targ = e.srcElement;\n",
667       "    if (targ.nodeType == 3) // defeat Safari bug\n",
668       "        targ = targ.parentNode;\n",
669       "\n",
670       "    // jQuery normalizes the pageX and pageY\n",
671       "    // pageX,Y are the mouse positions relative to the document\n",
672       "    // offset() returns the position of the element relative to the document\n",
673       "    var x = e.pageX - $(targ).offset().left;\n",
674       "    var y = e.pageY - $(targ).offset().top;\n",
675       "\n",
676       "    return {\"x\": x, \"y\": y};\n",
677       "};\n",
678       "\n",
679       "/*\n",
680       " * return a copy of an object with only non-object keys\n",
681       " * we need this to avoid circular references\n",
682       " * http://stackoverflow.com/a/24161582/3208463\n",
683       " */\n",
684       "function simpleKeys (original) {\n",
685       "  return Object.keys(original).reduce(function (obj, key) {\n",
686       "    if (typeof original[key] !== 'object')\n",
687       "        obj[key] = original[key]\n",
688       "    return obj;\n",
689       "  }, {});\n",
690       "}\n",
691       "\n",
692       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
693       "    var canvas_pos = mpl.findpos(event)\n",
694       "\n",
695       "    if (name === 'button_press')\n",
696       "    {\n",
697       "        this.canvas.focus();\n",
698       "        this.canvas_div.focus();\n",
699       "    }\n",
700       "\n",
701       "    var x = canvas_pos.x * mpl.ratio;\n",
702       "    var y = canvas_pos.y * mpl.ratio;\n",
703       "\n",
704       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
705       "                             step: event.step,\n",
706       "                             guiEvent: simpleKeys(event)});\n",
707       "\n",
708       "    /* This prevents the web browser from automatically changing to\n",
709       "     * the text insertion cursor when the button is pressed.  We want\n",
710       "     * to control all of the cursor setting manually through the\n",
711       "     * 'cursor' event from matplotlib */\n",
712       "    event.preventDefault();\n",
713       "    return false;\n",
714       "}\n",
715       "\n",
716       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
717       "    // Handle any extra behaviour associated with a key event\n",
718       "}\n",
719       "\n",
720       "mpl.figure.prototype.key_event = function(event, name) {\n",
721       "\n",
722       "    // Prevent repeat events\n",
723       "    if (name == 'key_press')\n",
724       "    {\n",
725       "        if (event.which === this._key)\n",
726       "            return;\n",
727       "        else\n",
728       "            this._key = event.which;\n",
729       "    }\n",
730       "    if (name == 'key_release')\n",
731       "        this._key = null;\n",
732       "\n",
733       "    var value = '';\n",
734       "    if (event.ctrlKey && event.which != 17)\n",
735       "        value += \"ctrl+\";\n",
736       "    if (event.altKey && event.which != 18)\n",
737       "        value += \"alt+\";\n",
738       "    if (event.shiftKey && event.which != 16)\n",
739       "        value += \"shift+\";\n",
740       "\n",
741       "    value += 'k';\n",
742       "    value += event.which.toString();\n",
743       "\n",
744       "    this._key_event_extra(event, name);\n",
745       "\n",
746       "    this.send_message(name, {key: value,\n",
747       "                             guiEvent: simpleKeys(event)});\n",
748       "    return false;\n",
749       "}\n",
750       "\n",
751       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
752       "    if (name == 'download') {\n",
753       "        this.handle_save(this, null);\n",
754       "    } else {\n",
755       "        this.send_message(\"toolbar_button\", {name: name});\n",
756       "    }\n",
757       "};\n",
758       "\n",
759       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
760       "    this.message.textContent = tooltip;\n",
761       "};\n",
762       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
763       "\n",
764       "mpl.extensions = [\"eps\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\n",
765       "\n",
766       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
767       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
768       "    // object with the appropriate methods. Currently this is a non binary\n",
769       "    // socket, so there is still some room for performance tuning.\n",
770       "    var ws = {};\n",
771       "\n",
772       "    ws.close = function() {\n",
773       "        comm.close()\n",
774       "    };\n",
775       "    ws.send = function(m) {\n",
776       "        //console.log('sending', m);\n",
777       "        comm.send(m);\n",
778       "    };\n",
779       "    // Register the callback with on_msg.\n",
780       "    comm.on_msg(function(msg) {\n",
781       "        //console.log('receiving', msg['content']['data'], msg);\n",
782       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
783       "        ws.onmessage(msg['content']['data'])\n",
784       "    });\n",
785       "    return ws;\n",
786       "}\n",
787       "\n",
788       "mpl.mpl_figure_comm = function(comm, msg) {\n",
789       "    // This is the function which gets called when the mpl process\n",
790       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
791       "\n",
792       "    var id = msg.content.data.id;\n",
793       "    // Get hold of the div created by the display call when the Comm\n",
794       "    // socket was opened in Python.\n",
795       "    var element = $(\"#\" + id);\n",
796       "    var ws_proxy = comm_websocket_adapter(comm)\n",
797       "\n",
798       "    function ondownload(figure, format) {\n",
799       "        window.open(figure.imageObj.src);\n",
800       "    }\n",
801       "\n",
802       "    var fig = new mpl.figure(id, ws_proxy,\n",
803       "                           ondownload,\n",
804       "                           element.get(0));\n",
805       "\n",
806       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
807       "    // web socket which is closed, not our websocket->open comm proxy.\n",
808       "    ws_proxy.onopen();\n",
809       "\n",
810       "    fig.parent_element = element.get(0);\n",
811       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
812       "    if (!fig.cell_info) {\n",
813       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
814       "        return;\n",
815       "    }\n",
816       "\n",
817       "    var output_index = fig.cell_info[2]\n",
818       "    var cell = fig.cell_info[0];\n",
819       "\n",
820       "};\n",
821       "\n",
822       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
823       "    var width = fig.canvas.width/mpl.ratio\n",
824       "    fig.root.unbind('remove')\n",
825       "\n",
826       "    // Update the output cell to use the data from the current canvas.\n",
827       "    fig.push_to_output();\n",
828       "    var dataURL = fig.canvas.toDataURL();\n",
829       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
830       "    // the notebook keyboard shortcuts fail.\n",
831       "    IPython.keyboard_manager.enable()\n",
832       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
833       "    fig.close_ws(fig, msg);\n",
834       "}\n",
835       "\n",
836       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
837       "    fig.send_message('closing', msg);\n",
838       "    // fig.ws.close()\n",
839       "}\n",
840       "\n",
841       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
842       "    // Turn the data on the canvas into data in the output cell.\n",
843       "    var width = this.canvas.width/mpl.ratio\n",
844       "    var dataURL = this.canvas.toDataURL();\n",
845       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
846       "}\n",
847       "\n",
848       "mpl.figure.prototype.updated_canvas_event = function() {\n",
849       "    // Tell IPython that the notebook contents must change.\n",
850       "    IPython.notebook.set_dirty(true);\n",
851       "    this.send_message(\"ack\", {});\n",
852       "    var fig = this;\n",
853       "    // Wait a second, then push the new image to the DOM so\n",
854       "    // that it is saved nicely (might be nice to debounce this).\n",
855       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
856       "}\n",
857       "\n",
858       "mpl.figure.prototype._init_toolbar = function() {\n",
859       "    var fig = this;\n",
860       "\n",
861       "    var nav_element = $('<div/>');\n",
862       "    nav_element.attr('style', 'width: 100%');\n",
863       "    this.root.append(nav_element);\n",
864       "\n",
865       "    // Define a callback function for later on.\n",
866       "    function toolbar_event(event) {\n",
867       "        return fig.toolbar_button_onclick(event['data']);\n",
868       "    }\n",
869       "    function toolbar_mouse_event(event) {\n",
870       "        return fig.toolbar_button_onmouseover(event['data']);\n",
871       "    }\n",
872       "\n",
873       "    for(var toolbar_ind in mpl.toolbar_items){\n",
874       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
875       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
876       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
877       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
878       "\n",
879       "        if (!name) { continue; };\n",
880       "\n",
881       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
882       "        button.click(method_name, toolbar_event);\n",
883       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
884       "        nav_element.append(button);\n",
885       "    }\n",
886       "\n",
887       "    // Add the status bar.\n",
888       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
889       "    nav_element.append(status_bar);\n",
890       "    this.message = status_bar[0];\n",
891       "\n",
892       "    // Add the close button to the window.\n",
893       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
894       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
895       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
896       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
897       "    buttongrp.append(button);\n",
898       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
899       "    titlebar.prepend(buttongrp);\n",
900       "}\n",
901       "\n",
902       "mpl.figure.prototype._root_extra_style = function(el){\n",
903       "    var fig = this\n",
904       "    el.on(\"remove\", function(){\n",
905       "\tfig.close_ws(fig, {});\n",
906       "    });\n",
907       "}\n",
908       "\n",
909       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
910       "    // this is important to make the div 'focusable\n",
911       "    el.attr('tabindex', 0)\n",
912       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
913       "    // off when our div gets focus\n",
914       "\n",
915       "    // location in version 3\n",
916       "    if (IPython.notebook.keyboard_manager) {\n",
917       "        IPython.notebook.keyboard_manager.register_events(el);\n",
918       "    }\n",
919       "    else {\n",
920       "        // location in version 2\n",
921       "        IPython.keyboard_manager.register_events(el);\n",
922       "    }\n",
923       "\n",
924       "}\n",
925       "\n",
926       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
927       "    var manager = IPython.notebook.keyboard_manager;\n",
928       "    if (!manager)\n",
929       "        manager = IPython.keyboard_manager;\n",
930       "\n",
931       "    // Check for shift+enter\n",
932       "    if (event.shiftKey && event.which == 13) {\n",
933       "        this.canvas_div.blur();\n",
934       "        // select the cell after this one\n",
935       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
936       "        IPython.notebook.select(index + 1);\n",
937       "    }\n",
938       "}\n",
939       "\n",
940       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
941       "    fig.ondownload(fig, null);\n",
942       "}\n",
943       "\n",
944       "\n",
945       "mpl.find_output_cell = function(html_output) {\n",
946       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
947       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
948       "    // IPython event is triggered only after the cells have been serialised, which for\n",
949       "    // our purposes (turning an active figure into a static one), is too late.\n",
950       "    var cells = IPython.notebook.get_cells();\n",
951       "    var ncells = cells.length;\n",
952       "    for (var i=0; i<ncells; i++) {\n",
953       "        var cell = cells[i];\n",
954       "        if (cell.cell_type === 'code'){\n",
955       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
956       "                var data = cell.output_area.outputs[j];\n",
957       "                if (data.data) {\n",
958       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
959       "                    data = data.data;\n",
960       "                }\n",
961       "                if (data['text/html'] == html_output) {\n",
962       "                    return [cell, data, j];\n",
963       "                }\n",
964       "            }\n",
965       "        }\n",
966       "    }\n",
967       "}\n",
968       "\n",
969       "// Register the function which deals with the matplotlib target/channel.\n",
970       "// The kernel may be null if the page has been refreshed.\n",
971       "if (IPython.notebook.kernel != null) {\n",
972       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
973       "}\n"
974      ],
975      "text/plain": [
976       "<IPython.core.display.Javascript object>"
977      ]
978     },
979     "metadata": {},
980     "output_type": "display_data"
981    },
982    {
983     "data": {
984      "text/html": [
985       "<img src=\"\" width=\"639.85\">"
986      ],
987      "text/plain": [
988       "<IPython.core.display.HTML object>"
989      ]
990     },
991     "metadata": {},
992     "output_type": "display_data"
993    },
994    {
995     "data": {
996      "text/plain": [
997       "(-4.0, 4.0)"
998      ]
999     },
1000     "execution_count": 11,
1001     "metadata": {},
1002     "output_type": "execute_result"
1003    }
1004   ],
1005   "source": [
1006    "# We reinstantiate the spacecraft orbit for convenience.\n",
1007    "hera = pk.planet.spice('-999', 'SUN', 'ECLIPJ2000', 'NONE', pk.MU_SUN, 100, 100, 100) \n",
1008    "# These are the solar system plants anorbiting objects we are intrerested in\n",
1009    "earth = pk.planet.spice('EARTH', 'SUN', 'ECLIPJ2000', 'NONE', pk.MU_SUN, pk.MU_EARTH, pk.EARTH_RADIUS, pk.EARTH_RADIUS*1.05) \n",
1010    "mars = pk.planet.spice('4', 'SUN', 'ECLIPJ2000', 'NONE', pk.MU_SUN, 100, 100,100) \n",
1011    "didymos = pk.planet.spice('2065803', 'SUN', 'ECLIPJ2000', 'NONE', pk.MU_SUN, 100, 100, 100) \n",
1012    "\n",
1013    "# And these are the plotting commands\n",
1014    "ax = pk.orbit_plots.plot_planet(hera, t0 = launch, tf = arrival, N=500, units=pk.AU)\n",
1015    "pk.orbit_plots.plot_planet(earth, t0 = launch, axes=ax, color = 'b', units=pk.AU)\n",
1016    "pk.orbit_plots.plot_planet(mars, t0 = flyby, axes=ax, color='r', units=pk.AU)\n",
1017    "pk.orbit_plots.plot_planet(didymos, t0 = launch, axes=ax, color='g', units=pk.AU)\n",
1018    "ax.set_xlim(-4,4)\n",
1019    "ax.set_ylim(-4,4)"
1020   ]
1021  },
1022  {
1023   "cell_type": "markdown",
1024   "metadata": {},
1025   "source": [
1026    "## Plotting other relevant quantities"
1027   ]
1028  },
1029  {
1030   "cell_type": "code",
1031   "execution_count": 12,
1032   "metadata": {},
1033   "outputs": [],
1034   "source": [
1035    "# Say we are interested in the HERA-EARTH distance.\n",
1036    "tspan = np.linspace(launch.jd, arrival.jd, 100)\n",
1037    "Reh = []\n",
1038    "Red = []\n",
1039    "for jd in tspan:\n",
1040    "    r1, _ = hera_wrt_earth.eph(pk.epoch(jd, 'jd'))\n",
1041    "    r2, _ = hera_wrt_didymos.eph(pk.epoch(jd, 'jd'))\n",
1042    "    Reh.append(np.linalg.norm(r1) / pk.AU)\n",
1043    "    Red.append(np.linalg.norm(r2) / pk.AU)\n",
1044    "\n"
1045   ]
1046  },
1047  {
1048   "cell_type": "code",
1049   "execution_count": 13,
1050   "metadata": {},
1051   "outputs": [
1052    {
1053     "data": {
1054      "application/javascript": [
1055       "/* Put everything inside the global mpl namespace */\n",
1056       "window.mpl = {};\n",
1057       "\n",
1058       "\n",
1059       "mpl.get_websocket_type = function() {\n",
1060       "    if (typeof(WebSocket) !== 'undefined') {\n",
1061       "        return WebSocket;\n",
1062       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
1063       "        return MozWebSocket;\n",
1064       "    } else {\n",
1065       "        alert('Your browser does not have WebSocket support. ' +\n",
1066       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
1067       "              'Firefox 4 and 5 are also supported but you ' +\n",
1068       "              'have to enable WebSockets in about:config.');\n",
1069       "    };\n",
1070       "}\n",
1071       "\n",
1072       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
1073       "    this.id = figure_id;\n",
1074       "\n",
1075       "    this.ws = websocket;\n",
1076       "\n",
1077       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
1078       "\n",
1079       "    if (!this.supports_binary) {\n",
1080       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
1081       "        if (warnings) {\n",
1082       "            warnings.style.display = 'block';\n",
1083       "            warnings.textContent = (\n",
1084       "                \"This browser does not support binary websocket messages. \" +\n",
1085       "                    \"Performance may be slow.\");\n",
1086       "        }\n",
1087       "    }\n",
1088       "\n",
1089       "    this.imageObj = new Image();\n",
1090       "\n",
1091       "    this.context = undefined;\n",
1092       "    this.message = undefined;\n",
1093       "    this.canvas = undefined;\n",
1094       "    this.rubberband_canvas = undefined;\n",
1095       "    this.rubberband_context = undefined;\n",
1096       "    this.format_dropdown = undefined;\n",
1097       "\n",
1098       "    this.image_mode = 'full';\n",
1099       "\n",
1100       "    this.root = $('<div/>');\n",
1101       "    this._root_extra_style(this.root)\n",
1102       "    this.root.attr('style', 'display: inline-block');\n",
1103       "\n",
1104       "    $(parent_element).append(this.root);\n",
1105       "\n",
1106       "    this._init_header(this);\n",
1107       "    this._init_canvas(this);\n",
1108       "    this._init_toolbar(this);\n",
1109       "\n",
1110       "    var fig = this;\n",
1111       "\n",
1112       "    this.waiting = false;\n",
1113       "\n",
1114       "    this.ws.onopen =  function () {\n",
1115       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
1116       "            fig.send_message(\"send_image_mode\", {});\n",
1117       "            if (mpl.ratio != 1) {\n",
1118       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
1119       "            }\n",
1120       "            fig.send_message(\"refresh\", {});\n",
1121       "        }\n",
1122       "\n",
1123       "    this.imageObj.onload = function() {\n",
1124       "            if (fig.image_mode == 'full') {\n",
1125       "                // Full images could contain transparency (where diff images\n",
1126       "                // almost always do), so we need to clear the canvas so that\n",
1127       "                // there is no ghosting.\n",
1128       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
1129       "            }\n",
1130       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
1131       "        };\n",
1132       "\n",
1133       "    this.imageObj.onunload = function() {\n",
1134       "        fig.ws.close();\n",
1135       "    }\n",
1136       "\n",
1137       "    this.ws.onmessage = this._make_on_message_function(this);\n",
1138       "\n",
1139       "    this.ondownload = ondownload;\n",
1140       "}\n",
1141       "\n",
1142       "mpl.figure.prototype._init_header = function() {\n",
1143       "    var titlebar = $(\n",
1144       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
1145       "        'ui-helper-clearfix\"/>');\n",
1146       "    var titletext = $(\n",
1147       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
1148       "        'text-align: center; padding: 3px;\"/>');\n",
1149       "    titlebar.append(titletext)\n",
1150       "    this.root.append(titlebar);\n",
1151       "    this.header = titletext[0];\n",
1152       "}\n",
1153       "\n",
1154       "\n",
1155       "\n",
1156       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
1157       "\n",
1158       "}\n",
1159       "\n",
1160       "\n",
1161       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
1162       "\n",
1163       "}\n",
1164       "\n",
1165       "mpl.figure.prototype._init_canvas = function() {\n",
1166       "    var fig = this;\n",
1167       "\n",
1168       "    var canvas_div = $('<div/>');\n",
1169       "\n",
1170       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
1171       "\n",
1172       "    function canvas_keyboard_event(event) {\n",
1173       "        return fig.key_event(event, event['data']);\n",
1174       "    }\n",
1175       "\n",
1176       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
1177       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
1178       "    this.canvas_div = canvas_div\n",
1179       "    this._canvas_extra_style(canvas_div)\n",
1180       "    this.root.append(canvas_div);\n",
1181       "\n",
1182       "    var canvas = $('<canvas/>');\n",
1183       "    canvas.addClass('mpl-canvas');\n",
1184       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
1185       "\n",
1186       "    this.canvas = canvas[0];\n",
1187       "    this.context = canvas[0].getContext(\"2d\");\n",
1188       "\n",
1189       "    var backingStore = this.context.backingStorePixelRatio ||\n",
1190       "\tthis.context.webkitBackingStorePixelRatio ||\n",
1191       "\tthis.context.mozBackingStorePixelRatio ||\n",
1192       "\tthis.context.msBackingStorePixelRatio ||\n",
1193       "\tthis.context.oBackingStorePixelRatio ||\n",
1194       "\tthis.context.backingStorePixelRatio || 1;\n",
1195       "\n",
1196       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
1197       "\n",
1198       "    var rubberband = $('<canvas/>');\n",
1199       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
1200       "\n",
1201       "    var pass_mouse_events = true;\n",
1202       "\n",
1203       "    canvas_div.resizable({\n",
1204       "        start: function(event, ui) {\n",
1205       "            pass_mouse_events = false;\n",
1206       "        },\n",
1207       "        resize: function(event, ui) {\n",
1208       "            fig.request_resize(ui.size.width, ui.size.height);\n",
1209       "        },\n",
1210       "        stop: function(event, ui) {\n",
1211       "            pass_mouse_events = true;\n",
1212       "            fig.request_resize(ui.size.width, ui.size.height);\n",
1213       "        },\n",
1214       "    });\n",
1215       "\n",
1216       "    function mouse_event_fn(event) {\n",
1217       "        if (pass_mouse_events)\n",
1218       "            return fig.mouse_event(event, event['data']);\n",
1219       "    }\n",
1220       "\n",
1221       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
1222       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
1223       "    // Throttle sequential mouse events to 1 every 20ms.\n",
1224       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
1225       "\n",
1226       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
1227       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
1228       "\n",
1229       "    canvas_div.on(\"wheel\", function (event) {\n",
1230       "        event = event.originalEvent;\n",
1231       "        event['data'] = 'scroll'\n",
1232       "        if (event.deltaY < 0) {\n",
1233       "            event.step = 1;\n",
1234       "        } else {\n",
1235       "            event.step = -1;\n",
1236       "        }\n",
1237       "        mouse_event_fn(event);\n",
1238       "    });\n",
1239       "\n",
1240       "    canvas_div.append(canvas);\n",
1241       "    canvas_div.append(rubberband);\n",
1242       "\n",
1243       "    this.rubberband = rubberband;\n",
1244       "    this.rubberband_canvas = rubberband[0];\n",
1245       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
1246       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
1247       "\n",
1248       "    this._resize_canvas = function(width, height) {\n",
1249       "        // Keep the size of the canvas, canvas container, and rubber band\n",
1250       "        // canvas in sync.\n",
1251       "        canvas_div.css('width', width)\n",
1252       "        canvas_div.css('height', height)\n",
1253       "\n",
1254       "        canvas.attr('width', width * mpl.ratio);\n",
1255       "        canvas.attr('height', height * mpl.ratio);\n",
1256       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
1257       "\n",
1258       "        rubberband.attr('width', width);\n",
1259       "        rubberband.attr('height', height);\n",
1260       "    }\n",
1261       "\n",
1262       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
1263       "    // upon first draw.\n",
1264       "    this._resize_canvas(600, 600);\n",
1265       "\n",
1266       "    // Disable right mouse context menu.\n",
1267       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
1268       "        return false;\n",
1269       "    });\n",
1270       "\n",
1271       "    function set_focus () {\n",
1272       "        canvas.focus();\n",
1273       "        canvas_div.focus();\n",
1274       "    }\n",
1275       "\n",
1276       "    window.setTimeout(set_focus, 100);\n",
1277       "}\n",
1278       "\n",
1279       "mpl.figure.prototype._init_toolbar = function() {\n",
1280       "    var fig = this;\n",
1281       "\n",
1282       "    var nav_element = $('<div/>');\n",
1283       "    nav_element.attr('style', 'width: 100%');\n",
1284       "    this.root.append(nav_element);\n",
1285       "\n",
1286       "    // Define a callback function for later on.\n",
1287       "    function toolbar_event(event) {\n",
1288       "        return fig.toolbar_button_onclick(event['data']);\n",
1289       "    }\n",
1290       "    function toolbar_mouse_event(event) {\n",
1291       "        return fig.toolbar_button_onmouseover(event['data']);\n",
1292       "    }\n",
1293       "\n",
1294       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
1295       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
1296       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
1297       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
1298       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
1299       "\n",
1300       "        if (!name) {\n",
1301       "            // put a spacer in here.\n",
1302       "            continue;\n",
1303       "        }\n",
1304       "        var button = $('<button/>');\n",
1305       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
1306       "                        'ui-button-icon-only');\n",
1307       "        button.attr('role', 'button');\n",
1308       "        button.attr('aria-disabled', 'false');\n",
1309       "        button.click(method_name, toolbar_event);\n",
1310       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
1311       "\n",
1312       "        var icon_img = $('<span/>');\n",
1313       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
1314       "        icon_img.addClass(image);\n",
1315       "        icon_img.addClass('ui-corner-all');\n",
1316       "\n",
1317       "        var tooltip_span = $('<span/>');\n",
1318       "        tooltip_span.addClass('ui-button-text');\n",
1319       "        tooltip_span.html(tooltip);\n",
1320       "\n",
1321       "        button.append(icon_img);\n",
1322       "        button.append(tooltip_span);\n",
1323       "\n",
1324       "        nav_element.append(button);\n",
1325       "    }\n",
1326       "\n",
1327       "    var fmt_picker_span = $('<span/>');\n",
1328       "\n",
1329       "    var fmt_picker = $('<select/>');\n",
1330       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
1331       "    fmt_picker_span.append(fmt_picker);\n",
1332       "    nav_element.append(fmt_picker_span);\n",
1333       "    this.format_dropdown = fmt_picker[0];\n",
1334       "\n",
1335       "    for (var ind in mpl.extensions) {\n",
1336       "        var fmt = mpl.extensions[ind];\n",
1337       "        var option = $(\n",
1338       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
1339       "        fmt_picker.append(option);\n",
1340       "    }\n",
1341       "\n",
1342       "    // Add hover states to the ui-buttons\n",
1343       "    $( \".ui-button\" ).hover(\n",
1344       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
1345       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
1346       "    );\n",
1347       "\n",
1348       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
1349       "    nav_element.append(status_bar);\n",
1350       "    this.message = status_bar[0];\n",
1351       "}\n",
1352       "\n",
1353       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
1354       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
1355       "    // which will in turn request a refresh of the image.\n",
1356       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
1357       "}\n",
1358       "\n",
1359       "mpl.figure.prototype.send_message = function(type, properties) {\n",
1360       "    properties['type'] = type;\n",
1361       "    properties['figure_id'] = this.id;\n",
1362       "    this.ws.send(JSON.stringify(properties));\n",
1363       "}\n",
1364       "\n",
1365       "mpl.figure.prototype.send_draw_message = function() {\n",
1366       "    if (!this.waiting) {\n",
1367       "        this.waiting = true;\n",
1368       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
1369       "    }\n",
1370       "}\n",
1371       "\n",
1372       "\n",
1373       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
1374       "    var format_dropdown = fig.format_dropdown;\n",
1375       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
1376       "    fig.ondownload(fig, format);\n",
1377       "}\n",
1378       "\n",
1379       "\n",
1380       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
1381       "    var size = msg['size'];\n",
1382       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
1383       "        fig._resize_canvas(size[0], size[1]);\n",
1384       "        fig.send_message(\"refresh\", {});\n",
1385       "    };\n",
1386       "}\n",
1387       "\n",
1388       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
1389       "    var x0 = msg['x0'] / mpl.ratio;\n",
1390       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
1391       "    var x1 = msg['x1'] / mpl.ratio;\n",
1392       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
1393       "    x0 = Math.floor(x0) + 0.5;\n",
1394       "    y0 = Math.floor(y0) + 0.5;\n",
1395       "    x1 = Math.floor(x1) + 0.5;\n",
1396       "    y1 = Math.floor(y1) + 0.5;\n",
1397       "    var min_x = Math.min(x0, x1);\n",
1398       "    var min_y = Math.min(y0, y1);\n",
1399       "    var width = Math.abs(x1 - x0);\n",
1400       "    var height = Math.abs(y1 - y0);\n",
1401       "\n",
1402       "    fig.rubberband_context.clearRect(\n",
1403       "        0, 0, fig.canvas.width / mpl.ratio, fig.canvas.height / mpl.ratio);\n",
1404       "\n",
1405       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
1406       "}\n",
1407       "\n",
1408       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
1409       "    // Updates the figure title.\n",
1410       "    fig.header.textContent = msg['label'];\n",
1411       "}\n",
1412       "\n",
1413       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
1414       "    var cursor = msg['cursor'];\n",
1415       "    switch(cursor)\n",
1416       "    {\n",
1417       "    case 0:\n",
1418       "        cursor = 'pointer';\n",
1419       "        break;\n",
1420       "    case 1:\n",
1421       "        cursor = 'default';\n",
1422       "        break;\n",
1423       "    case 2:\n",
1424       "        cursor = 'crosshair';\n",
1425       "        break;\n",
1426       "    case 3:\n",
1427       "        cursor = 'move';\n",
1428       "        break;\n",
1429       "    }\n",
1430       "    fig.rubberband_canvas.style.cursor = cursor;\n",
1431       "}\n",
1432       "\n",
1433       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
1434       "    fig.message.textContent = msg['message'];\n",
1435       "}\n",
1436       "\n",
1437       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
1438       "    // Request the server to send over a new figure.\n",
1439       "    fig.send_draw_message();\n",
1440       "}\n",
1441       "\n",
1442       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
1443       "    fig.image_mode = msg['mode'];\n",
1444       "}\n",
1445       "\n",
1446       "mpl.figure.prototype.updated_canvas_event = function() {\n",
1447       "    // Called whenever the canvas gets updated.\n",
1448       "    this.send_message(\"ack\", {});\n",
1449       "}\n",
1450       "\n",
1451       "// A function to construct a web socket function for onmessage handling.\n",
1452       "// Called in the figure constructor.\n",
1453       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
1454       "    return function socket_on_message(evt) {\n",
1455       "        if (evt.data instanceof Blob) {\n",
1456       "            /* FIXME: We get \"Resource interpreted as Image but\n",
1457       "             * transferred with MIME type text/plain:\" errors on\n",
1458       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
1459       "             * to be part of the websocket stream */\n",
1460       "            evt.data.type = \"image/png\";\n",
1461       "\n",
1462       "            /* Free the memory for the previous frames */\n",
1463       "            if (fig.imageObj.src) {\n",
1464       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
1465       "                    fig.imageObj.src);\n",
1466       "            }\n",
1467       "\n",
1468       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
1469       "                evt.data);\n",
1470       "            fig.updated_canvas_event();\n",
1471       "            fig.waiting = false;\n",
1472       "            return;\n",
1473       "        }\n",
1474       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
1475       "            fig.imageObj.src = evt.data;\n",
1476       "            fig.updated_canvas_event();\n",
1477       "            fig.waiting = false;\n",
1478       "            return;\n",
1479       "        }\n",
1480       "\n",
1481       "        var msg = JSON.parse(evt.data);\n",
1482       "        var msg_type = msg['type'];\n",
1483       "\n",
1484       "        // Call the  \"handle_{type}\" callback, which takes\n",
1485       "        // the figure and JSON message as its only arguments.\n",
1486       "        try {\n",
1487       "            var callback = fig[\"handle_\" + msg_type];\n",
1488       "        } catch (e) {\n",
1489       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
1490       "            return;\n",
1491       "        }\n",
1492       "\n",
1493       "        if (callback) {\n",
1494       "            try {\n",
1495       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
1496       "                callback(fig, msg);\n",
1497       "            } catch (e) {\n",
1498       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
1499       "            }\n",
1500       "        }\n",
1501       "    };\n",
1502       "}\n",
1503       "\n",
1504       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
1505       "mpl.findpos = function(e) {\n",
1506       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
1507       "    var targ;\n",
1508       "    if (!e)\n",
1509       "        e = window.event;\n",
1510       "    if (e.target)\n",
1511       "        targ = e.target;\n",
1512       "    else if (e.srcElement)\n",
1513       "        targ = e.srcElement;\n",
1514       "    if (targ.nodeType == 3) // defeat Safari bug\n",
1515       "        targ = targ.parentNode;\n",
1516       "\n",
1517       "    // jQuery normalizes the pageX and pageY\n",
1518       "    // pageX,Y are the mouse positions relative to the document\n",
1519       "    // offset() returns the position of the element relative to the document\n",
1520       "    var x = e.pageX - $(targ).offset().left;\n",
1521       "    var y = e.pageY - $(targ).offset().top;\n",
1522       "\n",
1523       "    return {\"x\": x, \"y\": y};\n",
1524       "};\n",
1525       "\n",
1526       "/*\n",
1527       " * return a copy of an object with only non-object keys\n",
1528       " * we need this to avoid circular references\n",
1529       " * http://stackoverflow.com/a/24161582/3208463\n",
1530       " */\n",
1531       "function simpleKeys (original) {\n",
1532       "  return Object.keys(original).reduce(function (obj, key) {\n",
1533       "    if (typeof original[key] !== 'object')\n",
1534       "        obj[key] = original[key]\n",
1535       "    return obj;\n",
1536       "  }, {});\n",
1537       "}\n",
1538       "\n",
1539       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
1540       "    var canvas_pos = mpl.findpos(event)\n",
1541       "\n",
1542       "    if (name === 'button_press')\n",
1543       "    {\n",
1544       "        this.canvas.focus();\n",
1545       "        this.canvas_div.focus();\n",
1546       "    }\n",
1547       "\n",
1548       "    var x = canvas_pos.x * mpl.ratio;\n",
1549       "    var y = canvas_pos.y * mpl.ratio;\n",
1550       "\n",
1551       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
1552       "                             step: event.step,\n",
1553       "                             guiEvent: simpleKeys(event)});\n",
1554       "\n",
1555       "    /* This prevents the web browser from automatically changing to\n",
1556       "     * the text insertion cursor when the button is pressed.  We want\n",
1557       "     * to control all of the cursor setting manually through the\n",
1558       "     * 'cursor' event from matplotlib */\n",
1559       "    event.preventDefault();\n",
1560       "    return false;\n",
1561       "}\n",
1562       "\n",
1563       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
1564       "    // Handle any extra behaviour associated with a key event\n",
1565       "}\n",
1566       "\n",
1567       "mpl.figure.prototype.key_event = function(event, name) {\n",
1568       "\n",
1569       "    // Prevent repeat events\n",
1570       "    if (name == 'key_press')\n",
1571       "    {\n",
1572       "        if (event.which === this._key)\n",
1573       "            return;\n",
1574       "        else\n",
1575       "            this._key = event.which;\n",
1576       "    }\n",
1577       "    if (name == 'key_release')\n",
1578       "        this._key = null;\n",
1579       "\n",
1580       "    var value = '';\n",
1581       "    if (event.ctrlKey && event.which != 17)\n",
1582       "        value += \"ctrl+\";\n",
1583       "    if (event.altKey && event.which != 18)\n",
1584       "        value += \"alt+\";\n",
1585       "    if (event.shiftKey && event.which != 16)\n",
1586       "        value += \"shift+\";\n",
1587       "\n",
1588       "    value += 'k';\n",
1589       "    value += event.which.toString();\n",
1590       "\n",
1591       "    this._key_event_extra(event, name);\n",
1592       "\n",
1593       "    this.send_message(name, {key: value,\n",
1594       "                             guiEvent: simpleKeys(event)});\n",
1595       "    return false;\n",
1596       "}\n",
1597       "\n",
1598       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
1599       "    if (name == 'download') {\n",
1600       "        this.handle_save(this, null);\n",
1601       "    } else {\n",
1602       "        this.send_message(\"toolbar_button\", {name: name});\n",
1603       "    }\n",
1604       "};\n",
1605       "\n",
1606       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
1607       "    this.message.textContent = tooltip;\n",
1608       "};\n",
1609       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
1610       "\n",
1611       "mpl.extensions = [\"eps\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\n",
1612       "\n",
1613       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
1614       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
1615       "    // object with the appropriate methods. Currently this is a non binary\n",
1616       "    // socket, so there is still some room for performance tuning.\n",
1617       "    var ws = {};\n",
1618       "\n",
1619       "    ws.close = function() {\n",
1620       "        comm.close()\n",
1621       "    };\n",
1622       "    ws.send = function(m) {\n",
1623       "        //console.log('sending', m);\n",
1624       "        comm.send(m);\n",
1625       "    };\n",
1626       "    // Register the callback with on_msg.\n",
1627       "    comm.on_msg(function(msg) {\n",
1628       "        //console.log('receiving', msg['content']['data'], msg);\n",
1629       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
1630       "        ws.onmessage(msg['content']['data'])\n",
1631       "    });\n",
1632       "    return ws;\n",
1633       "}\n",
1634       "\n",
1635       "mpl.mpl_figure_comm = function(comm, msg) {\n",
1636       "    // This is the function which gets called when the mpl process\n",
1637       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
1638       "\n",
1639       "    var id = msg.content.data.id;\n",
1640       "    // Get hold of the div created by the display call when the Comm\n",
1641       "    // socket was opened in Python.\n",
1642       "    var element = $(\"#\" + id);\n",
1643       "    var ws_proxy = comm_websocket_adapter(comm)\n",
1644       "\n",
1645       "    function ondownload(figure, format) {\n",
1646       "        window.open(figure.imageObj.src);\n",
1647       "    }\n",
1648       "\n",
1649       "    var fig = new mpl.figure(id, ws_proxy,\n",
1650       "                           ondownload,\n",
1651       "                           element.get(0));\n",
1652       "\n",
1653       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
1654       "    // web socket which is closed, not our websocket->open comm proxy.\n",
1655       "    ws_proxy.onopen();\n",
1656       "\n",
1657       "    fig.parent_element = element.get(0);\n",
1658       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
1659       "    if (!fig.cell_info) {\n",
1660       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
1661       "        return;\n",
1662       "    }\n",
1663       "\n",
1664       "    var output_index = fig.cell_info[2]\n",
1665       "    var cell = fig.cell_info[0];\n",
1666       "\n",
1667       "};\n",
1668       "\n",
1669       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
1670       "    var width = fig.canvas.width/mpl.ratio\n",
1671       "    fig.root.unbind('remove')\n",
1672       "\n",
1673       "    // Update the output cell to use the data from the current canvas.\n",
1674       "    fig.push_to_output();\n",
1675       "    var dataURL = fig.canvas.toDataURL();\n",
1676       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
1677       "    // the notebook keyboard shortcuts fail.\n",
1678       "    IPython.keyboard_manager.enable()\n",
1679       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
1680       "    fig.close_ws(fig, msg);\n",
1681       "}\n",
1682       "\n",
1683       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
1684       "    fig.send_message('closing', msg);\n",
1685       "    // fig.ws.close()\n",
1686       "}\n",
1687       "\n",
1688       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
1689       "    // Turn the data on the canvas into data in the output cell.\n",
1690       "    var width = this.canvas.width/mpl.ratio\n",
1691       "    var dataURL = this.canvas.toDataURL();\n",
1692       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
1693       "}\n",
1694       "\n",
1695       "mpl.figure.prototype.updated_canvas_event = function() {\n",
1696       "    // Tell IPython that the notebook contents must change.\n",
1697       "    IPython.notebook.set_dirty(true);\n",
1698       "    this.send_message(\"ack\", {});\n",
1699       "    var fig = this;\n",
1700       "    // Wait a second, then push the new image to the DOM so\n",
1701       "    // that it is saved nicely (might be nice to debounce this).\n",
1702       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
1703       "}\n",
1704       "\n",
1705       "mpl.figure.prototype._init_toolbar = function() {\n",
1706       "    var fig = this;\n",
1707       "\n",
1708       "    var nav_element = $('<div/>');\n",
1709       "    nav_element.attr('style', 'width: 100%');\n",
1710       "    this.root.append(nav_element);\n",
1711       "\n",
1712       "    // Define a callback function for later on.\n",
1713       "    function toolbar_event(event) {\n",
1714       "        return fig.toolbar_button_onclick(event['data']);\n",
1715       "    }\n",
1716       "    function toolbar_mouse_event(event) {\n",
1717       "        return fig.toolbar_button_onmouseover(event['data']);\n",
1718       "    }\n",
1719       "\n",
1720       "    for(var toolbar_ind in mpl.toolbar_items){\n",
1721       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
1722       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
1723       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
1724       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
1725       "\n",
1726       "        if (!name) { continue; };\n",
1727       "\n",
1728       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
1729       "        button.click(method_name, toolbar_event);\n",
1730       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
1731       "        nav_element.append(button);\n",
1732       "    }\n",
1733       "\n",
1734       "    // Add the status bar.\n",
1735       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
1736       "    nav_element.append(status_bar);\n",
1737       "    this.message = status_bar[0];\n",
1738       "\n",
1739       "    // Add the close button to the window.\n",
1740       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
1741       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
1742       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
1743       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
1744       "    buttongrp.append(button);\n",
1745       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
1746       "    titlebar.prepend(buttongrp);\n",
1747       "}\n",
1748       "\n",
1749       "mpl.figure.prototype._root_extra_style = function(el){\n",
1750       "    var fig = this\n",
1751       "    el.on(\"remove\", function(){\n",
1752       "\tfig.close_ws(fig, {});\n",
1753       "    });\n",
1754       "}\n",
1755       "\n",
1756       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
1757       "    // this is important to make the div 'focusable\n",
1758       "    el.attr('tabindex', 0)\n",
1759       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
1760       "    // off when our div gets focus\n",
1761       "\n",
1762       "    // location in version 3\n",
1763       "    if (IPython.notebook.keyboard_manager) {\n",
1764       "        IPython.notebook.keyboard_manager.register_events(el);\n",
1765       "    }\n",
1766       "    else {\n",
1767       "        // location in version 2\n",
1768       "        IPython.keyboard_manager.register_events(el);\n",
1769       "    }\n",
1770       "\n",
1771       "}\n",
1772       "\n",
1773       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
1774       "    var manager = IPython.notebook.keyboard_manager;\n",
1775       "    if (!manager)\n",
1776       "        manager = IPython.keyboard_manager;\n",
1777       "\n",
1778       "    // Check for shift+enter\n",
1779       "    if (event.shiftKey && event.which == 13) {\n",
1780       "        this.canvas_div.blur();\n",
1781       "        // select the cell after this one\n",
1782       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
1783       "        IPython.notebook.select(index + 1);\n",
1784       "    }\n",
1785       "}\n",
1786       "\n",
1787       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
1788       "    fig.ondownload(fig, null);\n",
1789       "}\n",
1790       "\n",
1791       "\n",
1792       "mpl.find_output_cell = function(html_output) {\n",
1793       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
1794       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
1795       "    // IPython event is triggered only after the cells have been serialised, which for\n",
1796       "    // our purposes (turning an active figure into a static one), is too late.\n",
1797       "    var cells = IPython.notebook.get_cells();\n",
1798       "    var ncells = cells.length;\n",
1799       "    for (var i=0; i<ncells; i++) {\n",
1800       "        var cell = cells[i];\n",
1801       "        if (cell.cell_type === 'code'){\n",
1802       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
1803       "                var data = cell.output_area.outputs[j];\n",
1804       "                if (data.data) {\n",
1805       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
1806       "                    data = data.data;\n",
1807       "                }\n",
1808       "                if (data['text/html'] == html_output) {\n",
1809       "                    return [cell, data, j];\n",
1810       "                }\n",
1811       "            }\n",
1812       "        }\n",
1813       "    }\n",
1814       "}\n",
1815       "\n",
1816       "// Register the function which deals with the matplotlib target/channel.\n",
1817       "// The kernel may be null if the page has been refreshed.\n",
1818       "if (IPython.notebook.kernel != null) {\n",
1819       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
1820       "}\n"
1821      ],
1822      "text/plain": [
1823       "<IPython.core.display.Javascript object>"
1824      ]
1825     },
1826     "metadata": {},
1827     "output_type": "display_data"
1828    },
1829    {
1830     "data": {
1831      "text/html": [
1832       "<img src=\"\" width=\"639.85\">"
1833      ],
1834      "text/plain": [
1835       "<IPython.core.display.HTML object>"
1836      ]
1837     },
1838     "metadata": {},
1839     "output_type": "display_data"
1840    },
1841    {
1842     "data": {
1843      "text/plain": [
1844       "<matplotlib.legend.Legend at 0x7f1ba1e4b2d0>"
1845      ]
1846     },
1847     "execution_count": 13,
1848     "metadata": {},
1849     "output_type": "execute_result"
1850    }
1851   ],
1852   "source": [
1853    "days_from_launch = tspan - launch.jd\n",
1854    "plt.figure()\n",
1855    "plt.plot(days_from_launch, Reh, label='SC / Earth distance (AU)')\n",
1856    "plt.plot(days_from_launch, Red, label='SC / Didymos distance (AU)')\n",
1857    "plt.legend()"
1858   ]
1859  }
1860 ],
1861 "metadata": {
1862  "kernelspec": {
1863   "display_name": "Python 3",
1864   "language": "python",
1865   "name": "python3"
1866  },
1867  "language_info": {
1868   "codemirror_mode": {
1869    "name": "ipython",
1870    "version": 3
1871   },
1872   "file_extension": ".py",
1873   "mimetype": "text/x-python",
1874   "name": "python",
1875   "nbconvert_exporter": "python",
1876   "pygments_lexer": "ipython3",
1877   "version": "3.7.6"
1878  }
1879 },
1880 "nbformat": 4,
1881 "nbformat_minor": 4
1882}
1883