Skip to content

Instantly share code, notes, and snippets.

@ulfmueller
Last active November 6, 2018 10:17
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save ulfmueller/2c1fd6c4c29d606b313ab32bc0391dd2 to your computer and use it in GitHub Desktop.
Save ulfmueller/2c1fd6c4c29d606b313ab32bc0391dd2 to your computer and use it in GitHub Desktop.
Jupyter-Notebook-Tutorial der Session eTraGo auf dem Abschlussworkshop von open_eGo in Berlin 2018
Display the source blob
Display the rendered blob
Raw
{
"cells": [
{
"cell_type": "markdown",
"metadata": {},
"source": [
"<img src=\"http://etrago.readthedocs.io/en/latest/_images/etrago_logo.png\" alt=\"HSF\" height=\"200\" width=\"200\" align=\"center\" >\n"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"\n",
"# <center> Abschlussworkshop open_eGo 2018 </center>\n",
"\n",
"## <center> Session eTraGo - Optimaler Netz- und Speicherausbau auf Hoch- und Höchstspannungsebene <center>\n"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"\n",
"# Agenda <a class=\"anchor\" id=\"toc\"></a>\n",
"\n",
"\n",
"* [Wichtige Links](#links)\n",
"* [Installation](#install)\n",
"* [Allgemeine Vorgehensweise](#allgemein)\n",
"* Mögliche Fragestellungen\n",
" * [Optimaler Speicherausbau](#speicher)\n",
" * [Optimaler Netz- und Speicherausbau](#netz_speicher)\n",
" * [Bewertung der NEP-Maßnahmen](#nep)"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"# Wichtige Links <a class=\"anchor\" id=\"links\"></a>\n",
"\n",
"* __[eTraGo Source Code](https://github.com/openego/eTraGo)__\n",
"* __[eTraGo Documentation](http://etrago.readthedocs.io/)__\n"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"# Installation <a class=\"anchor\" id=\"install\"></a>\n",
"Zur Verwendung dieses Notebooks wird die aktuelle eTraGo Version sowie das python package jupyter benötigt. Installiere diese mit\n",
"\n",
"`pip3 install eTraGo --process-dependency-links`\n",
"\n",
"`pip3 install jupyter`"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"# Allgemeine Vorgehensweise<a class=\"anchor\" id=\"allgemein\"></a>\n",
"\n",
"1. Parameter-Einstellungen (args) in appl.py vornehmen.\n",
" (evtl. darüber hinaus default-Annahmen in untergelagerten Funktionsaufrufen anpassen)\n",
"2. appl.py bzw. etrago(args) ausführen.\n",
"3. Ergebnisse auswerten.\n",
"4. evtl als csv oder in der oedb speichern. \n",
"\n",
"\n",
"# Optimaler Speicherausbau im NEP 2035 Szenario<a class=\"anchor\" id=\"speicher\"></a>\n",
"\n",
"\n",
"## Fragestellung\n",
"\n",
"1. An welchen Netzknoten entstehen in Zukunft welche Speicherbedarfe? (zentrale vs. dezentrale Lösungen)\n",
"2. Welches Einsatzverhalten weisen diese optimierten Speicher auf? (Langzeit- vs. Kurzzeitspeicher)"
]
},
{
"cell_type": "code",
"execution_count": 1,
"metadata": {},
"outputs": [],
"source": [
"# enable jupyter interactive plotting\n",
"%matplotlib notebook\n",
"from ipywidgets import *"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Importieren der benötigten allgemeinen und eTraGo spezifischen python Packete "
]
},
{
"cell_type": "code",
"execution_count": 2,
"metadata": {},
"outputs": [
{
"name": "stderr",
"output_type": "stream",
"text": [
"/home/ulf/venv/lib/python3.5/site-packages/psycopg2/__init__.py:144: UserWarning: The psycopg2 wheel package will be renamed from release 2.8; in order to keep installing from binary please use \"pip install psycopg2-binary\" instead. For details see: <http://initd.org/psycopg/docs/install.html#binary-install-from-pypi>.\n",
" \"\"\")\n",
"/home/ulf/venv/lib/python3.5/site-packages/sqlalchemy/ext/declarative/base.py:297: SAWarning: Ignoring declarative-like tuple value of attribute segment: possibly a copy-and-paste error with a comma left at the end of the line?\n",
" \"left at the end of the line?\" % k)\n",
"/home/ulf/venv/lib/python3.5/site-packages/sqlalchemy/ext/declarative/base.py:297: SAWarning: Ignoring declarative-like tuple value of attribute cable: possibly a copy-and-paste error with a comma left at the end of the line?\n",
" \"left at the end of the line?\" % k)\n",
"INFO:keyring.backend:Loading Gnome\n",
"INFO:keyring.backend:Loading Google\n",
"INFO:keyring.backend:Loading Windows (alt)\n",
"INFO:keyring.backend:Loading file\n",
"INFO:keyring.backend:Loading keyczar\n",
"INFO:keyring.backend:Loading multi\n",
"INFO:keyring.backend:Loading pyfs\n",
"INFO:keyring.backend:Loading KWallet\n",
"INFO:keyring.backend:Loading SecretService\n",
"INFO:keyring.backend:Loading Windows\n",
"INFO:keyring.backend:Loading macOS\n"
]
}
],
"source": [
"import datetime\n",
"import os\n",
"import os.path\n",
"import time\n",
"import numpy as np\n",
"import pandas as pd\n",
"\n",
"if 'READTHEDOCS' not in os.environ:\n",
" # Sphinx does not run this code.\n",
" # Do not import internal packages directly\n",
" from etrago.cluster.disaggregation import (\n",
" MiniSolverDisaggregation,\n",
" UniformDisaggregation)\n",
" \n",
" from etrago.cluster.networkclustering import (\n",
" busmap_from_psql,\n",
" cluster_on_extra_high_voltage,\n",
" kmean_clustering)\n",
" \n",
" from etrago.tools.io import (\n",
" NetworkScenario,\n",
" results_to_oedb,\n",
" extension,\n",
" decommissioning)\n",
" \n",
" from etrago.tools.plot import (\n",
" plot_line_loading,\n",
" plot_stacked_gen,\n",
" add_coordinates,\n",
" curtailment,\n",
" gen_dist,\n",
" storage_distribution,\n",
" storage_expansion,\n",
" nodal_gen_dispatch,\n",
" network_expansion)\n",
"\n",
" from etrago.tools.utilities import (\n",
" load_shedding,\n",
" data_manipulation_sh,\n",
" convert_capital_costs,\n",
" results_to_csv,\n",
" parallelisation,\n",
" pf_post_lopf,\n",
" loading_minimization,\n",
" calc_line_losses,\n",
" group_parallel_lines,\n",
" add_missing_components,\n",
" distribute_q,\n",
" set_q_foreign_loads,\n",
" clip_foreign,\n",
" foreign_links,\n",
" crossborder_capacity,\n",
" ramp_limits,\n",
" geolocation_buses,\n",
" get_args_setting,\n",
" set_branch_capacity)\n",
" \n",
" from etrago.tools.extendable import extendable, extension_preselection\n",
" from etrago.cluster.snapshot import snapshot_clustering, daily_bounds\n",
" from egoio.tools import db\n",
" from sqlalchemy.orm import sessionmaker\n",
" import oedialect\n",
" \n",
" from etrago.appl import etrago\n",
"\n"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Parameter-Einstellungen (args) in appl.py vornehmen\n",
"\n",
"Für diesen Fall sollen die Netzkapazitäten keine Variable sein. Neben der dispatch-Optimierung soll ausschließlich Speicherausbau optimiert werden. Hierzu wird die array Variable 'extendable': ['storage'] gesetzt."
]
},
{
"cell_type": "code",
"execution_count": 3,
"metadata": {},
"outputs": [],
"source": [
"args = { # Setup and Configuration:\n",
" 'db': 'local', # database session\n",
" 'gridversion': 'v0.4.5', # None for model_draft or Version number\n",
" 'method': 'lopf', # lopf or pf\n",
" 'pf_post_lopf': False, # perform a pf after a lopf simulation\n",
" 'start_snapshot': 48,\n",
" 'end_snapshot': 52,\n",
" 'solver': 'gurobi', # glpk, cplex or gurobi\n",
" 'solver_options': {}, # {} for default or dict of solver options\n",
" 'scn_name': 'NEP 2035', # a scenario: Status Quo, NEP 2035, eGo 100\n",
" # Scenario variations:\n",
" 'scn_extension': None, # None or array of extension scenarios\n",
" 'scn_decommissioning': None, # None or decommissioning scenario\n",
" # Export options:\n",
" 'lpfile': False, # save pyomo's lp file: False or /path/tofolder\n",
" 'csv_export': './results', # save results as csv: False or /path/tofolder\n",
" 'db_export': False, # export the results back to the oedb\n",
" # Settings:\n",
" 'extendable': ['storage'], # Array of components to optimize\n",
" 'generator_noise': 789456, # apply generator noise, False or seed number\n",
" 'minimize_loading': False,\n",
" 'ramp_limits': False, # Choose if using ramp limit of generators\n",
" 'extra_functionality': None, # Choose function name or None\n",
" # Clustering:\n",
" 'network_clustering_kmeans': 500, # False or the value k for clustering\n",
" 'load_cluster': \"load_cluster_nep_500\", # False or predefined busmap for k-means\n",
" 'network_clustering_ehv': False, # clustering of HV buses to EHV buses.\n",
" 'disaggregation': None, # or None, 'mini' or 'uniform'\n",
" 'snapshot_clustering': False, # False or the number of 'periods'\n",
" # Simplifications:\n",
" 'parallelisation': False, # run snapshots parallely.\n",
" 'skip_snapshots': False,\n",
" 'line_grouping': False, # group lines parallel lines\n",
" 'branch_capacity_factor': {'HV': 0.5, 'eHV' : 0.7}, # factors to change branch capacities\n",
" 'load_shedding': False, # meet the demand at very high cost\n",
" 'foreign_lines' : {'carrier': 'AC', 'capacity': 'osmTGmod'}, # dict containing carrier and capacity settings of foreign lines\n",
" 'comments': None}"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Starten der Rechnung \n",
"\n",
"Hierzu muss die appl.py bzw. die Funktion etrago(args) ausgeführt werden. Die Ergebnisse werden in Form eines PyPSA container (pandas) zur Verfügung gestellt."
]
},
{
"cell_type": "code",
"execution_count": 4,
"metadata": {},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"No data for StoragePqSet in column p_set.\n",
"Series p_set of component StorageUnit could not be imported\n",
"start k-mean clustering\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"/home/ulf/venv/lib/python3.5/site-packages/sklearn/cluster/k_means_.py:893: RuntimeWarning: Explicit initial center position passed: performing only one init in k-means instead of n_init=10\n",
" return_n_iter=True)\n"
]
},
{
"name": "stdout",
"output_type": "stream",
"text": [
"Inertia of k-means = 315.5491611310268\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"INFO:pypsa.pf:Slack bus for sub-network 0 is 0\n",
"INFO:pypsa.opf:Performed preliminary steps\n",
"INFO:pypsa.opf:Building pyomo model using `angles` formulation\n",
"INFO:pypsa.opf:Solving model using gurobi\n",
"INFO:pypsa.opf:Optimization successful\n"
]
},
{
"name": "stdout",
"output_type": "stream",
"text": [
"Time for LOPF [min]: 0.15\n",
"Investment costs for all storage units in selected snapshots [EUR]: 22088.15\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"INFO:pypsa.io:Exported network results has lines, generators, links, storage_units, carriers, buses, loads\n"
]
}
],
"source": [
"network, disagg_network = etrago(args)"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Auswertung\n",
"\n",
"Insbesondere die Plotting-Funktionen in etrago.tools.plot können für die Analyse der Ergebnisse verwendet werden."
]
},
{
"cell_type": "code",
"execution_count": 5,
"metadata": {},
"outputs": [
{
"data": {
"application/javascript": [
"/* Put everything inside the global mpl namespace */\n",
"window.mpl = {};\n",
"\n",
"\n",
"mpl.get_websocket_type = function() {\n",
" if (typeof(WebSocket) !== 'undefined') {\n",
" return WebSocket;\n",
" } else if (typeof(MozWebSocket) !== 'undefined') {\n",
" return MozWebSocket;\n",
" } else {\n",
" alert('Your browser does not have WebSocket support.' +\n",
" 'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
" 'Firefox 4 and 5 are also supported but you ' +\n",
" 'have to enable WebSockets in about:config.');\n",
" };\n",
"}\n",
"\n",
"mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
" this.id = figure_id;\n",
"\n",
" this.ws = websocket;\n",
"\n",
" this.supports_binary = (this.ws.binaryType != undefined);\n",
"\n",
" if (!this.supports_binary) {\n",
" var warnings = document.getElementById(\"mpl-warnings\");\n",
" if (warnings) {\n",
" warnings.style.display = 'block';\n",
" warnings.textContent = (\n",
" \"This browser does not support binary websocket messages. \" +\n",
" \"Performance may be slow.\");\n",
" }\n",
" }\n",
"\n",
" this.imageObj = new Image();\n",
"\n",
" this.context = undefined;\n",
" this.message = undefined;\n",
" this.canvas = undefined;\n",
" this.rubberband_canvas = undefined;\n",
" this.rubberband_context = undefined;\n",
" this.format_dropdown = undefined;\n",
"\n",
" this.image_mode = 'full';\n",
"\n",
" this.root = $('<div/>');\n",
" this._root_extra_style(this.root)\n",
" this.root.attr('style', 'display: inline-block');\n",
"\n",
" $(parent_element).append(this.root);\n",
"\n",
" this._init_header(this);\n",
" this._init_canvas(this);\n",
" this._init_toolbar(this);\n",
"\n",
" var fig = this;\n",
"\n",
" this.waiting = false;\n",
"\n",
" this.ws.onopen = function () {\n",
" fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
" fig.send_message(\"send_image_mode\", {});\n",
" if (mpl.ratio != 1) {\n",
" fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
" }\n",
" fig.send_message(\"refresh\", {});\n",
" }\n",
"\n",
" this.imageObj.onload = function() {\n",
" if (fig.image_mode == 'full') {\n",
" // Full images could contain transparency (where diff images\n",
" // almost always do), so we need to clear the canvas so that\n",
" // there is no ghosting.\n",
" fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
" }\n",
" fig.context.drawImage(fig.imageObj, 0, 0);\n",
" };\n",
"\n",
" this.imageObj.onunload = function() {\n",
" fig.ws.close();\n",
" }\n",
"\n",
" this.ws.onmessage = this._make_on_message_function(this);\n",
"\n",
" this.ondownload = ondownload;\n",
"}\n",
"\n",
"mpl.figure.prototype._init_header = function() {\n",
" var titlebar = $(\n",
" '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
" 'ui-helper-clearfix\"/>');\n",
" var titletext = $(\n",
" '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
" 'text-align: center; padding: 3px;\"/>');\n",
" titlebar.append(titletext)\n",
" this.root.append(titlebar);\n",
" this.header = titletext[0];\n",
"}\n",
"\n",
"\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._init_canvas = function() {\n",
" var fig = this;\n",
"\n",
" var canvas_div = $('<div/>');\n",
"\n",
" canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
"\n",
" function canvas_keyboard_event(event) {\n",
" return fig.key_event(event, event['data']);\n",
" }\n",
"\n",
" canvas_div.keydown('key_press', canvas_keyboard_event);\n",
" canvas_div.keyup('key_release', canvas_keyboard_event);\n",
" this.canvas_div = canvas_div\n",
" this._canvas_extra_style(canvas_div)\n",
" this.root.append(canvas_div);\n",
"\n",
" var canvas = $('<canvas/>');\n",
" canvas.addClass('mpl-canvas');\n",
" canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
"\n",
" this.canvas = canvas[0];\n",
" this.context = canvas[0].getContext(\"2d\");\n",
"\n",
" var backingStore = this.context.backingStorePixelRatio ||\n",
"\tthis.context.webkitBackingStorePixelRatio ||\n",
"\tthis.context.mozBackingStorePixelRatio ||\n",
"\tthis.context.msBackingStorePixelRatio ||\n",
"\tthis.context.oBackingStorePixelRatio ||\n",
"\tthis.context.backingStorePixelRatio || 1;\n",
"\n",
" mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
"\n",
" var rubberband = $('<canvas/>');\n",
" rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
"\n",
" var pass_mouse_events = true;\n",
"\n",
" canvas_div.resizable({\n",
" start: function(event, ui) {\n",
" pass_mouse_events = false;\n",
" },\n",
" resize: function(event, ui) {\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" stop: function(event, ui) {\n",
" pass_mouse_events = true;\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" });\n",
"\n",
" function mouse_event_fn(event) {\n",
" if (pass_mouse_events)\n",
" return fig.mouse_event(event, event['data']);\n",
" }\n",
"\n",
" rubberband.mousedown('button_press', mouse_event_fn);\n",
" rubberband.mouseup('button_release', mouse_event_fn);\n",
" // Throttle sequential mouse events to 1 every 20ms.\n",
" rubberband.mousemove('motion_notify', mouse_event_fn);\n",
"\n",
" rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
" rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
"\n",
" canvas_div.on(\"wheel\", function (event) {\n",
" event = event.originalEvent;\n",
" event['data'] = 'scroll'\n",
" if (event.deltaY < 0) {\n",
" event.step = 1;\n",
" } else {\n",
" event.step = -1;\n",
" }\n",
" mouse_event_fn(event);\n",
" });\n",
"\n",
" canvas_div.append(canvas);\n",
" canvas_div.append(rubberband);\n",
"\n",
" this.rubberband = rubberband;\n",
" this.rubberband_canvas = rubberband[0];\n",
" this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
" this.rubberband_context.strokeStyle = \"#000000\";\n",
"\n",
" this._resize_canvas = function(width, height) {\n",
" // Keep the size of the canvas, canvas container, and rubber band\n",
" // canvas in synch.\n",
" canvas_div.css('width', width)\n",
" canvas_div.css('height', height)\n",
"\n",
" canvas.attr('width', width * mpl.ratio);\n",
" canvas.attr('height', height * mpl.ratio);\n",
" canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
"\n",
" rubberband.attr('width', width);\n",
" rubberband.attr('height', height);\n",
" }\n",
"\n",
" // Set the figure to an initial 600x600px, this will subsequently be updated\n",
" // upon first draw.\n",
" this._resize_canvas(600, 600);\n",
"\n",
" // Disable right mouse context menu.\n",
" $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
" return false;\n",
" });\n",
"\n",
" function set_focus () {\n",
" canvas.focus();\n",
" canvas_div.focus();\n",
" }\n",
"\n",
" window.setTimeout(set_focus, 100);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items) {\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) {\n",
" // put a spacer in here.\n",
" continue;\n",
" }\n",
" var button = $('<button/>');\n",
" button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
" 'ui-button-icon-only');\n",
" button.attr('role', 'button');\n",
" button.attr('aria-disabled', 'false');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
"\n",
" var icon_img = $('<span/>');\n",
" icon_img.addClass('ui-button-icon-primary ui-icon');\n",
" icon_img.addClass(image);\n",
" icon_img.addClass('ui-corner-all');\n",
"\n",
" var tooltip_span = $('<span/>');\n",
" tooltip_span.addClass('ui-button-text');\n",
" tooltip_span.html(tooltip);\n",
"\n",
" button.append(icon_img);\n",
" button.append(tooltip_span);\n",
"\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" var fmt_picker_span = $('<span/>');\n",
"\n",
" var fmt_picker = $('<select/>');\n",
" fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
" fmt_picker_span.append(fmt_picker);\n",
" nav_element.append(fmt_picker_span);\n",
" this.format_dropdown = fmt_picker[0];\n",
"\n",
" for (var ind in mpl.extensions) {\n",
" var fmt = mpl.extensions[ind];\n",
" var option = $(\n",
" '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
" fmt_picker.append(option)\n",
" }\n",
"\n",
" // Add hover states to the ui-buttons\n",
" $( \".ui-button\" ).hover(\n",
" function() { $(this).addClass(\"ui-state-hover\");},\n",
" function() { $(this).removeClass(\"ui-state-hover\");}\n",
" );\n",
"\n",
" var status_bar = $('<span class=\"mpl-message\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"}\n",
"\n",
"mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
" // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
" // which will in turn request a refresh of the image.\n",
" this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
"}\n",
"\n",
"mpl.figure.prototype.send_message = function(type, properties) {\n",
" properties['type'] = type;\n",
" properties['figure_id'] = this.id;\n",
" this.ws.send(JSON.stringify(properties));\n",
"}\n",
"\n",
"mpl.figure.prototype.send_draw_message = function() {\n",
" if (!this.waiting) {\n",
" this.waiting = true;\n",
" this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
" }\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" var format_dropdown = fig.format_dropdown;\n",
" var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
" fig.ondownload(fig, format);\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
" var size = msg['size'];\n",
" if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
" fig._resize_canvas(size[0], size[1]);\n",
" fig.send_message(\"refresh\", {});\n",
" };\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
" var x0 = msg['x0'] / mpl.ratio;\n",
" var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
" var x1 = msg['x1'] / mpl.ratio;\n",
" var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
" x0 = Math.floor(x0) + 0.5;\n",
" y0 = Math.floor(y0) + 0.5;\n",
" x1 = Math.floor(x1) + 0.5;\n",
" y1 = Math.floor(y1) + 0.5;\n",
" var min_x = Math.min(x0, x1);\n",
" var min_y = Math.min(y0, y1);\n",
" var width = Math.abs(x1 - x0);\n",
" var height = Math.abs(y1 - y0);\n",
"\n",
" fig.rubberband_context.clearRect(\n",
" 0, 0, fig.canvas.width, fig.canvas.height);\n",
"\n",
" fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
" // Updates the figure title.\n",
" fig.header.textContent = msg['label'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
" var cursor = msg['cursor'];\n",
" switch(cursor)\n",
" {\n",
" case 0:\n",
" cursor = 'pointer';\n",
" break;\n",
" case 1:\n",
" cursor = 'default';\n",
" break;\n",
" case 2:\n",
" cursor = 'crosshair';\n",
" break;\n",
" case 3:\n",
" cursor = 'move';\n",
" break;\n",
" }\n",
" fig.rubberband_canvas.style.cursor = cursor;\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_message = function(fig, msg) {\n",
" fig.message.textContent = msg['message'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
" // Request the server to send over a new figure.\n",
" fig.send_draw_message();\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
" fig.image_mode = msg['mode'];\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Called whenever the canvas gets updated.\n",
" this.send_message(\"ack\", {});\n",
"}\n",
"\n",
"// A function to construct a web socket function for onmessage handling.\n",
"// Called in the figure constructor.\n",
"mpl.figure.prototype._make_on_message_function = function(fig) {\n",
" return function socket_on_message(evt) {\n",
" if (evt.data instanceof Blob) {\n",
" /* FIXME: We get \"Resource interpreted as Image but\n",
" * transferred with MIME type text/plain:\" errors on\n",
" * Chrome. But how to set the MIME type? It doesn't seem\n",
" * to be part of the websocket stream */\n",
" evt.data.type = \"image/png\";\n",
"\n",
" /* Free the memory for the previous frames */\n",
" if (fig.imageObj.src) {\n",
" (window.URL || window.webkitURL).revokeObjectURL(\n",
" fig.imageObj.src);\n",
" }\n",
"\n",
" fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
" evt.data);\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
" else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
" fig.imageObj.src = evt.data;\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
"\n",
" var msg = JSON.parse(evt.data);\n",
" var msg_type = msg['type'];\n",
"\n",
" // Call the \"handle_{type}\" callback, which takes\n",
" // the figure and JSON message as its only arguments.\n",
" try {\n",
" var callback = fig[\"handle_\" + msg_type];\n",
" } catch (e) {\n",
" console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
" return;\n",
" }\n",
"\n",
" if (callback) {\n",
" try {\n",
" // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
" callback(fig, msg);\n",
" } catch (e) {\n",
" console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
" }\n",
" }\n",
" };\n",
"}\n",
"\n",
"// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
"mpl.findpos = function(e) {\n",
" //this section is from http://www.quirksmode.org/js/events_properties.html\n",
" var targ;\n",
" if (!e)\n",
" e = window.event;\n",
" if (e.target)\n",
" targ = e.target;\n",
" else if (e.srcElement)\n",
" targ = e.srcElement;\n",
" if (targ.nodeType == 3) // defeat Safari bug\n",
" targ = targ.parentNode;\n",
"\n",
" // jQuery normalizes the pageX and pageY\n",
" // pageX,Y are the mouse positions relative to the document\n",
" // offset() returns the position of the element relative to the document\n",
" var x = e.pageX - $(targ).offset().left;\n",
" var y = e.pageY - $(targ).offset().top;\n",
"\n",
" return {\"x\": x, \"y\": y};\n",
"};\n",
"\n",
"/*\n",
" * return a copy of an object with only non-object keys\n",
" * we need this to avoid circular references\n",
" * http://stackoverflow.com/a/24161582/3208463\n",
" */\n",
"function simpleKeys (original) {\n",
" return Object.keys(original).reduce(function (obj, key) {\n",
" if (typeof original[key] !== 'object')\n",
" obj[key] = original[key]\n",
" return obj;\n",
" }, {});\n",
"}\n",
"\n",
"mpl.figure.prototype.mouse_event = function(event, name) {\n",
" var canvas_pos = mpl.findpos(event)\n",
"\n",
" if (name === 'button_press')\n",
" {\n",
" this.canvas.focus();\n",
" this.canvas_div.focus();\n",
" }\n",
"\n",
" var x = canvas_pos.x * mpl.ratio;\n",
" var y = canvas_pos.y * mpl.ratio;\n",
"\n",
" this.send_message(name, {x: x, y: y, button: event.button,\n",
" step: event.step,\n",
" guiEvent: simpleKeys(event)});\n",
"\n",
" /* This prevents the web browser from automatically changing to\n",
" * the text insertion cursor when the button is pressed. We want\n",
" * to control all of the cursor setting manually through the\n",
" * 'cursor' event from matplotlib */\n",
" event.preventDefault();\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" // Handle any extra behaviour associated with a key event\n",
"}\n",
"\n",
"mpl.figure.prototype.key_event = function(event, name) {\n",
"\n",
" // Prevent repeat events\n",
" if (name == 'key_press')\n",
" {\n",
" if (event.which === this._key)\n",
" return;\n",
" else\n",
" this._key = event.which;\n",
" }\n",
" if (name == 'key_release')\n",
" this._key = null;\n",
"\n",
" var value = '';\n",
" if (event.ctrlKey && event.which != 17)\n",
" value += \"ctrl+\";\n",
" if (event.altKey && event.which != 18)\n",
" value += \"alt+\";\n",
" if (event.shiftKey && event.which != 16)\n",
" value += \"shift+\";\n",
"\n",
" value += 'k';\n",
" value += event.which.toString();\n",
"\n",
" this._key_event_extra(event, name);\n",
"\n",
" this.send_message(name, {key: value,\n",
" guiEvent: simpleKeys(event)});\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
" if (name == 'download') {\n",
" this.handle_save(this, null);\n",
" } else {\n",
" this.send_message(\"toolbar_button\", {name: name});\n",
" }\n",
"};\n",
"\n",
"mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
" this.message.textContent = tooltip;\n",
"};\n",
"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",
"\n",
"mpl.extensions = [\"eps\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\n",
"\n",
"mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
" // Create a \"websocket\"-like object which calls the given IPython comm\n",
" // object with the appropriate methods. Currently this is a non binary\n",
" // socket, so there is still some room for performance tuning.\n",
" var ws = {};\n",
"\n",
" ws.close = function() {\n",
" comm.close()\n",
" };\n",
" ws.send = function(m) {\n",
" //console.log('sending', m);\n",
" comm.send(m);\n",
" };\n",
" // Register the callback with on_msg.\n",
" comm.on_msg(function(msg) {\n",
" //console.log('receiving', msg['content']['data'], msg);\n",
" // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
" ws.onmessage(msg['content']['data'])\n",
" });\n",
" return ws;\n",
"}\n",
"\n",
"mpl.mpl_figure_comm = function(comm, msg) {\n",
" // This is the function which gets called when the mpl process\n",
" // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
"\n",
" var id = msg.content.data.id;\n",
" // Get hold of the div created by the display call when the Comm\n",
" // socket was opened in Python.\n",
" var element = $(\"#\" + id);\n",
" var ws_proxy = comm_websocket_adapter(comm)\n",
"\n",
" function ondownload(figure, format) {\n",
" window.open(figure.imageObj.src);\n",
" }\n",
"\n",
" var fig = new mpl.figure(id, ws_proxy,\n",
" ondownload,\n",
" element.get(0));\n",
"\n",
" // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
" // web socket which is closed, not our websocket->open comm proxy.\n",
" ws_proxy.onopen();\n",
"\n",
" fig.parent_element = element.get(0);\n",
" fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
" if (!fig.cell_info) {\n",
" console.error(\"Failed to find cell for figure\", id, fig);\n",
" return;\n",
" }\n",
"\n",
" var output_index = fig.cell_info[2]\n",
" var cell = fig.cell_info[0];\n",
"\n",
"};\n",
"\n",
"mpl.figure.prototype.handle_close = function(fig, msg) {\n",
" var width = fig.canvas.width/mpl.ratio\n",
" fig.root.unbind('remove')\n",
"\n",
" // Update the output cell to use the data from the current canvas.\n",
" fig.push_to_output();\n",
" var dataURL = fig.canvas.toDataURL();\n",
" // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
" // the notebook keyboard shortcuts fail.\n",
" IPython.keyboard_manager.enable()\n",
" $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
" fig.close_ws(fig, msg);\n",
"}\n",
"\n",
"mpl.figure.prototype.close_ws = function(fig, msg){\n",
" fig.send_message('closing', msg);\n",
" // fig.ws.close()\n",
"}\n",
"\n",
"mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
" // Turn the data on the canvas into data in the output cell.\n",
" var width = this.canvas.width/mpl.ratio\n",
" var dataURL = this.canvas.toDataURL();\n",
" this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Tell IPython that the notebook contents must change.\n",
" IPython.notebook.set_dirty(true);\n",
" this.send_message(\"ack\", {});\n",
" var fig = this;\n",
" // Wait a second, then push the new image to the DOM so\n",
" // that it is saved nicely (might be nice to debounce this).\n",
" setTimeout(function () { fig.push_to_output() }, 1000);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items){\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) { continue; };\n",
"\n",
" var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" // Add the status bar.\n",
" var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"\n",
" // Add the close button to the window.\n",
" var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
" 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",
" button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
" button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
" buttongrp.append(button);\n",
" var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
" titlebar.prepend(buttongrp);\n",
"}\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(el){\n",
" var fig = this\n",
" el.on(\"remove\", function(){\n",
"\tfig.close_ws(fig, {});\n",
" });\n",
"}\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(el){\n",
" // this is important to make the div 'focusable\n",
" el.attr('tabindex', 0)\n",
" // reach out to IPython and tell the keyboard manager to turn it's self\n",
" // off when our div gets focus\n",
"\n",
" // location in version 3\n",
" if (IPython.notebook.keyboard_manager) {\n",
" IPython.notebook.keyboard_manager.register_events(el);\n",
" }\n",
" else {\n",
" // location in version 2\n",
" IPython.keyboard_manager.register_events(el);\n",
" }\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" var manager = IPython.notebook.keyboard_manager;\n",
" if (!manager)\n",
" manager = IPython.keyboard_manager;\n",
"\n",
" // Check for shift+enter\n",
" if (event.shiftKey && event.which == 13) {\n",
" this.canvas_div.blur();\n",
" event.shiftKey = false;\n",
" // Send a \"J\" for go to next cell\n",
" event.which = 74;\n",
" event.keyCode = 74;\n",
" manager.command_mode();\n",
" manager.handle_keydown(event);\n",
" }\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" fig.ondownload(fig, null);\n",
"}\n",
"\n",
"\n",
"mpl.find_output_cell = function(html_output) {\n",
" // Return the cell and output element which can be found *uniquely* in the notebook.\n",
" // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
" // IPython event is triggered only after the cells have been serialised, which for\n",
" // our purposes (turning an active figure into a static one), is too late.\n",
" var cells = IPython.notebook.get_cells();\n",
" var ncells = cells.length;\n",
" for (var i=0; i<ncells; i++) {\n",
" var cell = cells[i];\n",
" if (cell.cell_type === 'code'){\n",
" for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
" var data = cell.output_area.outputs[j];\n",
" if (data.data) {\n",
" // IPython >= 3 moved mimebundle to data attribute of output\n",
" data = data.data;\n",
" }\n",
" if (data['text/html'] == html_output) {\n",
" return [cell, data, j];\n",
" }\n",
" }\n",
" }\n",
" }\n",
"}\n",
"\n",
"// Register the function which deals with the matplotlib target/channel.\n",
"// The kernel may be null if the page has been refreshed.\n",
"if (IPython.notebook.kernel != null) {\n",
" IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
"}\n"
],
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"text/html": [
"<img src=\"\" width=\"600\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"# Speicherausbau je Technologie und Knoten\n",
"\n",
"storage_expansion(network, scaling=1)"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"\n",
"# Optimaler Netz- und Speicherausbau<a class=\"anchor\" id=\"netz_speicher\"></a>\n",
"\n",
"\n",
"## Fragestellung\n",
"\n",
"Welche Kombination aus Netz- und Speicherausbau ist ökonmisch sinnvoll und technisch machbar?\n",
"\n",
"## args Einstellungen vornehmen\n",
"\n",
"1. Zeitraum einstellen: 'start_snapshot': 48, 'end_snapshot': 72\n",
"2. Szenario einstellen: 'scn_name': 'eGo 100'\n",
"3. Netzausbau aktivieren und Speicherausbau im Ausland als Variable setzen: 'extendable': ['network', 'storage', 'foreign_storage']\n"
]
},
{
"cell_type": "code",
"execution_count": 8,
"metadata": {},
"outputs": [],
"source": [
"args_gs = { # Setup and Configuration:\n",
" 'db': 'local', # database session\n",
" 'gridversion': 'v0.4.5', # None for model_draft or Version number\n",
" 'method': 'lopf', # lopf or pf\n",
" 'pf_post_lopf': False, # perform a pf after a lopf simulation\n",
" 'start_snapshot': 48,\n",
" 'end_snapshot': 72,\n",
" 'solver': 'gurobi', # glpk, cplex or gurobi\n",
" 'solver_options': {}, # {} for default or dict of solver options\n",
" 'scn_name': 'eGo 100', # a scenario: Status Quo, NEP 2035, eGo 100\n",
" # Scenario variations:\n",
" 'scn_extension': None, # None or array of extension scenarios\n",
" 'scn_decommissioning': None, # None or decommissioning scenario\n",
" # Export options:\n",
" 'lpfile': False, # save pyomo's lp file: False or /path/tofolder\n",
" 'csv_export': './results', # save results as csv: False or /path/tofolder\n",
" 'db_export': False, # export the results back to the oedb\n",
" # Settings:\n",
" 'extendable': ['network', 'storage', 'foreign_storage'], # Array of components to optimize\n",
" 'generator_noise': 789456, # apply generator noise, False or seed number\n",
" 'minimize_loading': False,\n",
" 'ramp_limits': False, # Choose if using ramp limit of generators\n",
" 'extra_functionality': None, # Choose function name or None\n",
" # Clustering:\n",
" 'network_clustering_kmeans': 50, # False or the value k for clustering\n",
" 'load_cluster': False, # False or predefined busmap for k-means\n",
" 'network_clustering_ehv': False, # clustering of HV buses to EHV buses.\n",
" 'disaggregation': None, # or None, 'mini' or 'uniform'\n",
" 'snapshot_clustering': False, # False or the number of 'periods'\n",
" # Simplifications:\n",
" 'parallelisation': False, # run snapshots parallely.\n",
" 'skip_snapshots': False,\n",
" 'line_grouping': False, # group lines parallel lines\n",
" 'branch_capacity_factor': {'HV': 0.5, 'eHV' : 0.7}, # factors to change branch capacities\n",
" 'load_shedding': False, # meet the demand at very high cost\n",
" 'foreign_lines' : {'carrier': 'AC', 'capacity': 'osmTGmod'}, # dict containing carrier and capacity settings of foreign lines\n",
" 'comments': None}"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Starten der Rechnung "
]
},
{
"cell_type": "code",
"execution_count": 9,
"metadata": {},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"No data for StoragePqSet in column p_set.\n",
"Series p_set of component StorageUnit could not be imported\n",
"start k-mean clustering\n",
"Inertia of k-means = 5985.797310710579\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"INFO:pypsa.pf:Slack bus for sub-network 0 is 0\n",
"INFO:pypsa.opf:Performed preliminary steps\n",
"INFO:pypsa.opf:Building pyomo model using `angles` formulation\n",
"INFO:pypsa.opf:Solving model using gurobi\n",
"INFO:pypsa.opf:Optimization successful\n",
"INFO:pypsa.io:Exported network results has lines, generators, links, storage_units, carriers, buses, loads\n"
]
},
{
"name": "stdout",
"output_type": "stream",
"text": [
"Time for LOPF [min]: 1.01\n",
"Investment costs for all storage units in selected snapshots [EUR]: 826715.57\n",
"Investment costs for all lines and transformers in selected snapshots [EUR]: 339075.76\n"
]
}
],
"source": [
"network, disagg_network = etrago(args_gs)"
]
},
{
"cell_type": "code",
"execution_count": 10,
"metadata": {},
"outputs": [
{
"data": {
"application/javascript": [
"/* Put everything inside the global mpl namespace */\n",
"window.mpl = {};\n",
"\n",
"\n",
"mpl.get_websocket_type = function() {\n",
" if (typeof(WebSocket) !== 'undefined') {\n",
" return WebSocket;\n",
" } else if (typeof(MozWebSocket) !== 'undefined') {\n",
" return MozWebSocket;\n",
" } else {\n",
" alert('Your browser does not have WebSocket support.' +\n",
" 'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
" 'Firefox 4 and 5 are also supported but you ' +\n",
" 'have to enable WebSockets in about:config.');\n",
" };\n",
"}\n",
"\n",
"mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
" this.id = figure_id;\n",
"\n",
" this.ws = websocket;\n",
"\n",
" this.supports_binary = (this.ws.binaryType != undefined);\n",
"\n",
" if (!this.supports_binary) {\n",
" var warnings = document.getElementById(\"mpl-warnings\");\n",
" if (warnings) {\n",
" warnings.style.display = 'block';\n",
" warnings.textContent = (\n",
" \"This browser does not support binary websocket messages. \" +\n",
" \"Performance may be slow.\");\n",
" }\n",
" }\n",
"\n",
" this.imageObj = new Image();\n",
"\n",
" this.context = undefined;\n",
" this.message = undefined;\n",
" this.canvas = undefined;\n",
" this.rubberband_canvas = undefined;\n",
" this.rubberband_context = undefined;\n",
" this.format_dropdown = undefined;\n",
"\n",
" this.image_mode = 'full';\n",
"\n",
" this.root = $('<div/>');\n",
" this._root_extra_style(this.root)\n",
" this.root.attr('style', 'display: inline-block');\n",
"\n",
" $(parent_element).append(this.root);\n",
"\n",
" this._init_header(this);\n",
" this._init_canvas(this);\n",
" this._init_toolbar(this);\n",
"\n",
" var fig = this;\n",
"\n",
" this.waiting = false;\n",
"\n",
" this.ws.onopen = function () {\n",
" fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
" fig.send_message(\"send_image_mode\", {});\n",
" if (mpl.ratio != 1) {\n",
" fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
" }\n",
" fig.send_message(\"refresh\", {});\n",
" }\n",
"\n",
" this.imageObj.onload = function() {\n",
" if (fig.image_mode == 'full') {\n",
" // Full images could contain transparency (where diff images\n",
" // almost always do), so we need to clear the canvas so that\n",
" // there is no ghosting.\n",
" fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
" }\n",
" fig.context.drawImage(fig.imageObj, 0, 0);\n",
" };\n",
"\n",
" this.imageObj.onunload = function() {\n",
" fig.ws.close();\n",
" }\n",
"\n",
" this.ws.onmessage = this._make_on_message_function(this);\n",
"\n",
" this.ondownload = ondownload;\n",
"}\n",
"\n",
"mpl.figure.prototype._init_header = function() {\n",
" var titlebar = $(\n",
" '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
" 'ui-helper-clearfix\"/>');\n",
" var titletext = $(\n",
" '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
" 'text-align: center; padding: 3px;\"/>');\n",
" titlebar.append(titletext)\n",
" this.root.append(titlebar);\n",
" this.header = titletext[0];\n",
"}\n",
"\n",
"\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._init_canvas = function() {\n",
" var fig = this;\n",
"\n",
" var canvas_div = $('<div/>');\n",
"\n",
" canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
"\n",
" function canvas_keyboard_event(event) {\n",
" return fig.key_event(event, event['data']);\n",
" }\n",
"\n",
" canvas_div.keydown('key_press', canvas_keyboard_event);\n",
" canvas_div.keyup('key_release', canvas_keyboard_event);\n",
" this.canvas_div = canvas_div\n",
" this._canvas_extra_style(canvas_div)\n",
" this.root.append(canvas_div);\n",
"\n",
" var canvas = $('<canvas/>');\n",
" canvas.addClass('mpl-canvas');\n",
" canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
"\n",
" this.canvas = canvas[0];\n",
" this.context = canvas[0].getContext(\"2d\");\n",
"\n",
" var backingStore = this.context.backingStorePixelRatio ||\n",
"\tthis.context.webkitBackingStorePixelRatio ||\n",
"\tthis.context.mozBackingStorePixelRatio ||\n",
"\tthis.context.msBackingStorePixelRatio ||\n",
"\tthis.context.oBackingStorePixelRatio ||\n",
"\tthis.context.backingStorePixelRatio || 1;\n",
"\n",
" mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
"\n",
" var rubberband = $('<canvas/>');\n",
" rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
"\n",
" var pass_mouse_events = true;\n",
"\n",
" canvas_div.resizable({\n",
" start: function(event, ui) {\n",
" pass_mouse_events = false;\n",
" },\n",
" resize: function(event, ui) {\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" stop: function(event, ui) {\n",
" pass_mouse_events = true;\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" });\n",
"\n",
" function mouse_event_fn(event) {\n",
" if (pass_mouse_events)\n",
" return fig.mouse_event(event, event['data']);\n",
" }\n",
"\n",
" rubberband.mousedown('button_press', mouse_event_fn);\n",
" rubberband.mouseup('button_release', mouse_event_fn);\n",
" // Throttle sequential mouse events to 1 every 20ms.\n",
" rubberband.mousemove('motion_notify', mouse_event_fn);\n",
"\n",
" rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
" rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
"\n",
" canvas_div.on(\"wheel\", function (event) {\n",
" event = event.originalEvent;\n",
" event['data'] = 'scroll'\n",
" if (event.deltaY < 0) {\n",
" event.step = 1;\n",
" } else {\n",
" event.step = -1;\n",
" }\n",
" mouse_event_fn(event);\n",
" });\n",
"\n",
" canvas_div.append(canvas);\n",
" canvas_div.append(rubberband);\n",
"\n",
" this.rubberband = rubberband;\n",
" this.rubberband_canvas = rubberband[0];\n",
" this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
" this.rubberband_context.strokeStyle = \"#000000\";\n",
"\n",
" this._resize_canvas = function(width, height) {\n",
" // Keep the size of the canvas, canvas container, and rubber band\n",
" // canvas in synch.\n",
" canvas_div.css('width', width)\n",
" canvas_div.css('height', height)\n",
"\n",
" canvas.attr('width', width * mpl.ratio);\n",
" canvas.attr('height', height * mpl.ratio);\n",
" canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
"\n",
" rubberband.attr('width', width);\n",
" rubberband.attr('height', height);\n",
" }\n",
"\n",
" // Set the figure to an initial 600x600px, this will subsequently be updated\n",
" // upon first draw.\n",
" this._resize_canvas(600, 600);\n",
"\n",
" // Disable right mouse context menu.\n",
" $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
" return false;\n",
" });\n",
"\n",
" function set_focus () {\n",
" canvas.focus();\n",
" canvas_div.focus();\n",
" }\n",
"\n",
" window.setTimeout(set_focus, 100);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items) {\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) {\n",
" // put a spacer in here.\n",
" continue;\n",
" }\n",
" var button = $('<button/>');\n",
" button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
" 'ui-button-icon-only');\n",
" button.attr('role', 'button');\n",
" button.attr('aria-disabled', 'false');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
"\n",
" var icon_img = $('<span/>');\n",
" icon_img.addClass('ui-button-icon-primary ui-icon');\n",
" icon_img.addClass(image);\n",
" icon_img.addClass('ui-corner-all');\n",
"\n",
" var tooltip_span = $('<span/>');\n",
" tooltip_span.addClass('ui-button-text');\n",
" tooltip_span.html(tooltip);\n",
"\n",
" button.append(icon_img);\n",
" button.append(tooltip_span);\n",
"\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" var fmt_picker_span = $('<span/>');\n",
"\n",
" var fmt_picker = $('<select/>');\n",
" fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
" fmt_picker_span.append(fmt_picker);\n",
" nav_element.append(fmt_picker_span);\n",
" this.format_dropdown = fmt_picker[0];\n",
"\n",
" for (var ind in mpl.extensions) {\n",
" var fmt = mpl.extensions[ind];\n",
" var option = $(\n",
" '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
" fmt_picker.append(option)\n",
" }\n",
"\n",
" // Add hover states to the ui-buttons\n",
" $( \".ui-button\" ).hover(\n",
" function() { $(this).addClass(\"ui-state-hover\");},\n",
" function() { $(this).removeClass(\"ui-state-hover\");}\n",
" );\n",
"\n",
" var status_bar = $('<span class=\"mpl-message\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"}\n",
"\n",
"mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
" // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
" // which will in turn request a refresh of the image.\n",
" this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
"}\n",
"\n",
"mpl.figure.prototype.send_message = function(type, properties) {\n",
" properties['type'] = type;\n",
" properties['figure_id'] = this.id;\n",
" this.ws.send(JSON.stringify(properties));\n",
"}\n",
"\n",
"mpl.figure.prototype.send_draw_message = function() {\n",
" if (!this.waiting) {\n",
" this.waiting = true;\n",
" this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
" }\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" var format_dropdown = fig.format_dropdown;\n",
" var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
" fig.ondownload(fig, format);\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
" var size = msg['size'];\n",
" if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
" fig._resize_canvas(size[0], size[1]);\n",
" fig.send_message(\"refresh\", {});\n",
" };\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
" var x0 = msg['x0'] / mpl.ratio;\n",
" var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
" var x1 = msg['x1'] / mpl.ratio;\n",
" var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
" x0 = Math.floor(x0) + 0.5;\n",
" y0 = Math.floor(y0) + 0.5;\n",
" x1 = Math.floor(x1) + 0.5;\n",
" y1 = Math.floor(y1) + 0.5;\n",
" var min_x = Math.min(x0, x1);\n",
" var min_y = Math.min(y0, y1);\n",
" var width = Math.abs(x1 - x0);\n",
" var height = Math.abs(y1 - y0);\n",
"\n",
" fig.rubberband_context.clearRect(\n",
" 0, 0, fig.canvas.width, fig.canvas.height);\n",
"\n",
" fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
" // Updates the figure title.\n",
" fig.header.textContent = msg['label'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
" var cursor = msg['cursor'];\n",
" switch(cursor)\n",
" {\n",
" case 0:\n",
" cursor = 'pointer';\n",
" break;\n",
" case 1:\n",
" cursor = 'default';\n",
" break;\n",
" case 2:\n",
" cursor = 'crosshair';\n",
" break;\n",
" case 3:\n",
" cursor = 'move';\n",
" break;\n",
" }\n",
" fig.rubberband_canvas.style.cursor = cursor;\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_message = function(fig, msg) {\n",
" fig.message.textContent = msg['message'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
" // Request the server to send over a new figure.\n",
" fig.send_draw_message();\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
" fig.image_mode = msg['mode'];\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Called whenever the canvas gets updated.\n",
" this.send_message(\"ack\", {});\n",
"}\n",
"\n",
"// A function to construct a web socket function for onmessage handling.\n",
"// Called in the figure constructor.\n",
"mpl.figure.prototype._make_on_message_function = function(fig) {\n",
" return function socket_on_message(evt) {\n",
" if (evt.data instanceof Blob) {\n",
" /* FIXME: We get \"Resource interpreted as Image but\n",
" * transferred with MIME type text/plain:\" errors on\n",
" * Chrome. But how to set the MIME type? It doesn't seem\n",
" * to be part of the websocket stream */\n",
" evt.data.type = \"image/png\";\n",
"\n",
" /* Free the memory for the previous frames */\n",
" if (fig.imageObj.src) {\n",
" (window.URL || window.webkitURL).revokeObjectURL(\n",
" fig.imageObj.src);\n",
" }\n",
"\n",
" fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
" evt.data);\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
" else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
" fig.imageObj.src = evt.data;\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
"\n",
" var msg = JSON.parse(evt.data);\n",
" var msg_type = msg['type'];\n",
"\n",
" // Call the \"handle_{type}\" callback, which takes\n",
" // the figure and JSON message as its only arguments.\n",
" try {\n",
" var callback = fig[\"handle_\" + msg_type];\n",
" } catch (e) {\n",
" console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
" return;\n",
" }\n",
"\n",
" if (callback) {\n",
" try {\n",
" // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
" callback(fig, msg);\n",
" } catch (e) {\n",
" console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
" }\n",
" }\n",
" };\n",
"}\n",
"\n",
"// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
"mpl.findpos = function(e) {\n",
" //this section is from http://www.quirksmode.org/js/events_properties.html\n",
" var targ;\n",
" if (!e)\n",
" e = window.event;\n",
" if (e.target)\n",
" targ = e.target;\n",
" else if (e.srcElement)\n",
" targ = e.srcElement;\n",
" if (targ.nodeType == 3) // defeat Safari bug\n",
" targ = targ.parentNode;\n",
"\n",
" // jQuery normalizes the pageX and pageY\n",
" // pageX,Y are the mouse positions relative to the document\n",
" // offset() returns the position of the element relative to the document\n",
" var x = e.pageX - $(targ).offset().left;\n",
" var y = e.pageY - $(targ).offset().top;\n",
"\n",
" return {\"x\": x, \"y\": y};\n",
"};\n",
"\n",
"/*\n",
" * return a copy of an object with only non-object keys\n",
" * we need this to avoid circular references\n",
" * http://stackoverflow.com/a/24161582/3208463\n",
" */\n",
"function simpleKeys (original) {\n",
" return Object.keys(original).reduce(function (obj, key) {\n",
" if (typeof original[key] !== 'object')\n",
" obj[key] = original[key]\n",
" return obj;\n",
" }, {});\n",
"}\n",
"\n",
"mpl.figure.prototype.mouse_event = function(event, name) {\n",
" var canvas_pos = mpl.findpos(event)\n",
"\n",
" if (name === 'button_press')\n",
" {\n",
" this.canvas.focus();\n",
" this.canvas_div.focus();\n",
" }\n",
"\n",
" var x = canvas_pos.x * mpl.ratio;\n",
" var y = canvas_pos.y * mpl.ratio;\n",
"\n",
" this.send_message(name, {x: x, y: y, button: event.button,\n",
" step: event.step,\n",
" guiEvent: simpleKeys(event)});\n",
"\n",
" /* This prevents the web browser from automatically changing to\n",
" * the text insertion cursor when the button is pressed. We want\n",
" * to control all of the cursor setting manually through the\n",
" * 'cursor' event from matplotlib */\n",
" event.preventDefault();\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" // Handle any extra behaviour associated with a key event\n",
"}\n",
"\n",
"mpl.figure.prototype.key_event = function(event, name) {\n",
"\n",
" // Prevent repeat events\n",
" if (name == 'key_press')\n",
" {\n",
" if (event.which === this._key)\n",
" return;\n",
" else\n",
" this._key = event.which;\n",
" }\n",
" if (name == 'key_release')\n",
" this._key = null;\n",
"\n",
" var value = '';\n",
" if (event.ctrlKey && event.which != 17)\n",
" value += \"ctrl+\";\n",
" if (event.altKey && event.which != 18)\n",
" value += \"alt+\";\n",
" if (event.shiftKey && event.which != 16)\n",
" value += \"shift+\";\n",
"\n",
" value += 'k';\n",
" value += event.which.toString();\n",
"\n",
" this._key_event_extra(event, name);\n",
"\n",
" this.send_message(name, {key: value,\n",
" guiEvent: simpleKeys(event)});\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
" if (name == 'download') {\n",
" this.handle_save(this, null);\n",
" } else {\n",
" this.send_message(\"toolbar_button\", {name: name});\n",
" }\n",
"};\n",
"\n",
"mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
" this.message.textContent = tooltip;\n",
"};\n",
"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",
"\n",
"mpl.extensions = [\"eps\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\n",
"\n",
"mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
" // Create a \"websocket\"-like object which calls the given IPython comm\n",
" // object with the appropriate methods. Currently this is a non binary\n",
" // socket, so there is still some room for performance tuning.\n",
" var ws = {};\n",
"\n",
" ws.close = function() {\n",
" comm.close()\n",
" };\n",
" ws.send = function(m) {\n",
" //console.log('sending', m);\n",
" comm.send(m);\n",
" };\n",
" // Register the callback with on_msg.\n",
" comm.on_msg(function(msg) {\n",
" //console.log('receiving', msg['content']['data'], msg);\n",
" // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
" ws.onmessage(msg['content']['data'])\n",
" });\n",
" return ws;\n",
"}\n",
"\n",
"mpl.mpl_figure_comm = function(comm, msg) {\n",
" // This is the function which gets called when the mpl process\n",
" // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
"\n",
" var id = msg.content.data.id;\n",
" // Get hold of the div created by the display call when the Comm\n",
" // socket was opened in Python.\n",
" var element = $(\"#\" + id);\n",
" var ws_proxy = comm_websocket_adapter(comm)\n",
"\n",
" function ondownload(figure, format) {\n",
" window.open(figure.imageObj.src);\n",
" }\n",
"\n",
" var fig = new mpl.figure(id, ws_proxy,\n",
" ondownload,\n",
" element.get(0));\n",
"\n",
" // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
" // web socket which is closed, not our websocket->open comm proxy.\n",
" ws_proxy.onopen();\n",
"\n",
" fig.parent_element = element.get(0);\n",
" fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
" if (!fig.cell_info) {\n",
" console.error(\"Failed to find cell for figure\", id, fig);\n",
" return;\n",
" }\n",
"\n",
" var output_index = fig.cell_info[2]\n",
" var cell = fig.cell_info[0];\n",
"\n",
"};\n",
"\n",
"mpl.figure.prototype.handle_close = function(fig, msg) {\n",
" var width = fig.canvas.width/mpl.ratio\n",
" fig.root.unbind('remove')\n",
"\n",
" // Update the output cell to use the data from the current canvas.\n",
" fig.push_to_output();\n",
" var dataURL = fig.canvas.toDataURL();\n",
" // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
" // the notebook keyboard shortcuts fail.\n",
" IPython.keyboard_manager.enable()\n",
" $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
" fig.close_ws(fig, msg);\n",
"}\n",
"\n",
"mpl.figure.prototype.close_ws = function(fig, msg){\n",
" fig.send_message('closing', msg);\n",
" // fig.ws.close()\n",
"}\n",
"\n",
"mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
" // Turn the data on the canvas into data in the output cell.\n",
" var width = this.canvas.width/mpl.ratio\n",
" var dataURL = this.canvas.toDataURL();\n",
" this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Tell IPython that the notebook contents must change.\n",
" IPython.notebook.set_dirty(true);\n",
" this.send_message(\"ack\", {});\n",
" var fig = this;\n",
" // Wait a second, then push the new image to the DOM so\n",
" // that it is saved nicely (might be nice to debounce this).\n",
" setTimeout(function () { fig.push_to_output() }, 1000);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items){\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) { continue; };\n",
"\n",
" var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" // Add the status bar.\n",
" var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"\n",
" // Add the close button to the window.\n",
" var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
" 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",
" button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
" button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
" buttongrp.append(button);\n",
" var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
" titlebar.prepend(buttongrp);\n",
"}\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(el){\n",
" var fig = this\n",
" el.on(\"remove\", function(){\n",
"\tfig.close_ws(fig, {});\n",
" });\n",
"}\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(el){\n",
" // this is important to make the div 'focusable\n",
" el.attr('tabindex', 0)\n",
" // reach out to IPython and tell the keyboard manager to turn it's self\n",
" // off when our div gets focus\n",
"\n",
" // location in version 3\n",
" if (IPython.notebook.keyboard_manager) {\n",
" IPython.notebook.keyboard_manager.register_events(el);\n",
" }\n",
" else {\n",
" // location in version 2\n",
" IPython.keyboard_manager.register_events(el);\n",
" }\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" var manager = IPython.notebook.keyboard_manager;\n",
" if (!manager)\n",
" manager = IPython.keyboard_manager;\n",
"\n",
" // Check for shift+enter\n",
" if (event.shiftKey && event.which == 13) {\n",
" this.canvas_div.blur();\n",
" event.shiftKey = false;\n",
" // Send a \"J\" for go to next cell\n",
" event.which = 74;\n",
" event.keyCode = 74;\n",
" manager.command_mode();\n",
" manager.handle_keydown(event);\n",
" }\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" fig.ondownload(fig, null);\n",
"}\n",
"\n",
"\n",
"mpl.find_output_cell = function(html_output) {\n",
" // Return the cell and output element which can be found *uniquely* in the notebook.\n",
" // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
" // IPython event is triggered only after the cells have been serialised, which for\n",
" // our purposes (turning an active figure into a static one), is too late.\n",
" var cells = IPython.notebook.get_cells();\n",
" var ncells = cells.length;\n",
" for (var i=0; i<ncells; i++) {\n",
" var cell = cells[i];\n",
" if (cell.cell_type === 'code'){\n",
" for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
" var data = cell.output_area.outputs[j];\n",
" if (data.data) {\n",
" // IPython >= 3 moved mimebundle to data attribute of output\n",
" data = data.data;\n",
" }\n",
" if (data['text/html'] == html_output) {\n",
" return [cell, data, j];\n",
" }\n",
" }\n",
" }\n",
" }\n",
"}\n",
"\n",
"// Register the function which deals with the matplotlib target/channel.\n",
"// The kernel may be null if the page has been refreshed.\n",
"if (IPython.notebook.kernel != null) {\n",
" IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
"}\n"
],
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"text/html": [
"<img src=\"\" width=\"640\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"/home/ulf/venv/lib/python3.5/site-packages/matplotlib/colorbar.py:890: RuntimeWarning: invalid value encountered in true_divide\n",
" z = np.take(y, i0) + (xn - np.take(b, i0)) * dy / db\n"
]
}
],
"source": [
"# Netzausbau je Leitung relativ zur ursprünglich installierten Leistung\n",
"# maximal-möglicher Leitungsausbau je Leitung beträgt vier Mal der ursprünglich installierten Leistung\n",
"\n",
"network_expansion(network, ext_width=80)"
]
},
{
"cell_type": "code",
"execution_count": 11,
"metadata": {},
"outputs": [
{
"data": {
"application/javascript": [
"/* Put everything inside the global mpl namespace */\n",
"window.mpl = {};\n",
"\n",
"\n",
"mpl.get_websocket_type = function() {\n",
" if (typeof(WebSocket) !== 'undefined') {\n",
" return WebSocket;\n",
" } else if (typeof(MozWebSocket) !== 'undefined') {\n",
" return MozWebSocket;\n",
" } else {\n",
" alert('Your browser does not have WebSocket support.' +\n",
" 'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
" 'Firefox 4 and 5 are also supported but you ' +\n",
" 'have to enable WebSockets in about:config.');\n",
" };\n",
"}\n",
"\n",
"mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
" this.id = figure_id;\n",
"\n",
" this.ws = websocket;\n",
"\n",
" this.supports_binary = (this.ws.binaryType != undefined);\n",
"\n",
" if (!this.supports_binary) {\n",
" var warnings = document.getElementById(\"mpl-warnings\");\n",
" if (warnings) {\n",
" warnings.style.display = 'block';\n",
" warnings.textContent = (\n",
" \"This browser does not support binary websocket messages. \" +\n",
" \"Performance may be slow.\");\n",
" }\n",
" }\n",
"\n",
" this.imageObj = new Image();\n",
"\n",
" this.context = undefined;\n",
" this.message = undefined;\n",
" this.canvas = undefined;\n",
" this.rubberband_canvas = undefined;\n",
" this.rubberband_context = undefined;\n",
" this.format_dropdown = undefined;\n",
"\n",
" this.image_mode = 'full';\n",
"\n",
" this.root = $('<div/>');\n",
" this._root_extra_style(this.root)\n",
" this.root.attr('style', 'display: inline-block');\n",
"\n",
" $(parent_element).append(this.root);\n",
"\n",
" this._init_header(this);\n",
" this._init_canvas(this);\n",
" this._init_toolbar(this);\n",
"\n",
" var fig = this;\n",
"\n",
" this.waiting = false;\n",
"\n",
" this.ws.onopen = function () {\n",
" fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
" fig.send_message(\"send_image_mode\", {});\n",
" if (mpl.ratio != 1) {\n",
" fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
" }\n",
" fig.send_message(\"refresh\", {});\n",
" }\n",
"\n",
" this.imageObj.onload = function() {\n",
" if (fig.image_mode == 'full') {\n",
" // Full images could contain transparency (where diff images\n",
" // almost always do), so we need to clear the canvas so that\n",
" // there is no ghosting.\n",
" fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
" }\n",
" fig.context.drawImage(fig.imageObj, 0, 0);\n",
" };\n",
"\n",
" this.imageObj.onunload = function() {\n",
" fig.ws.close();\n",
" }\n",
"\n",
" this.ws.onmessage = this._make_on_message_function(this);\n",
"\n",
" this.ondownload = ondownload;\n",
"}\n",
"\n",
"mpl.figure.prototype._init_header = function() {\n",
" var titlebar = $(\n",
" '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
" 'ui-helper-clearfix\"/>');\n",
" var titletext = $(\n",
" '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
" 'text-align: center; padding: 3px;\"/>');\n",
" titlebar.append(titletext)\n",
" this.root.append(titlebar);\n",
" this.header = titletext[0];\n",
"}\n",
"\n",
"\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._init_canvas = function() {\n",
" var fig = this;\n",
"\n",
" var canvas_div = $('<div/>');\n",
"\n",
" canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
"\n",
" function canvas_keyboard_event(event) {\n",
" return fig.key_event(event, event['data']);\n",
" }\n",
"\n",
" canvas_div.keydown('key_press', canvas_keyboard_event);\n",
" canvas_div.keyup('key_release', canvas_keyboard_event);\n",
" this.canvas_div = canvas_div\n",
" this._canvas_extra_style(canvas_div)\n",
" this.root.append(canvas_div);\n",
"\n",
" var canvas = $('<canvas/>');\n",
" canvas.addClass('mpl-canvas');\n",
" canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
"\n",
" this.canvas = canvas[0];\n",
" this.context = canvas[0].getContext(\"2d\");\n",
"\n",
" var backingStore = this.context.backingStorePixelRatio ||\n",
"\tthis.context.webkitBackingStorePixelRatio ||\n",
"\tthis.context.mozBackingStorePixelRatio ||\n",
"\tthis.context.msBackingStorePixelRatio ||\n",
"\tthis.context.oBackingStorePixelRatio ||\n",
"\tthis.context.backingStorePixelRatio || 1;\n",
"\n",
" mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
"\n",
" var rubberband = $('<canvas/>');\n",
" rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
"\n",
" var pass_mouse_events = true;\n",
"\n",
" canvas_div.resizable({\n",
" start: function(event, ui) {\n",
" pass_mouse_events = false;\n",
" },\n",
" resize: function(event, ui) {\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" stop: function(event, ui) {\n",
" pass_mouse_events = true;\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" });\n",
"\n",
" function mouse_event_fn(event) {\n",
" if (pass_mouse_events)\n",
" return fig.mouse_event(event, event['data']);\n",
" }\n",
"\n",
" rubberband.mousedown('button_press', mouse_event_fn);\n",
" rubberband.mouseup('button_release', mouse_event_fn);\n",
" // Throttle sequential mouse events to 1 every 20ms.\n",
" rubberband.mousemove('motion_notify', mouse_event_fn);\n",
"\n",
" rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
" rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
"\n",
" canvas_div.on(\"wheel\", function (event) {\n",
" event = event.originalEvent;\n",
" event['data'] = 'scroll'\n",
" if (event.deltaY < 0) {\n",
" event.step = 1;\n",
" } else {\n",
" event.step = -1;\n",
" }\n",
" mouse_event_fn(event);\n",
" });\n",
"\n",
" canvas_div.append(canvas);\n",
" canvas_div.append(rubberband);\n",
"\n",
" this.rubberband = rubberband;\n",
" this.rubberband_canvas = rubberband[0];\n",
" this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
" this.rubberband_context.strokeStyle = \"#000000\";\n",
"\n",
" this._resize_canvas = function(width, height) {\n",
" // Keep the size of the canvas, canvas container, and rubber band\n",
" // canvas in synch.\n",
" canvas_div.css('width', width)\n",
" canvas_div.css('height', height)\n",
"\n",
" canvas.attr('width', width * mpl.ratio);\n",
" canvas.attr('height', height * mpl.ratio);\n",
" canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
"\n",
" rubberband.attr('width', width);\n",
" rubberband.attr('height', height);\n",
" }\n",
"\n",
" // Set the figure to an initial 600x600px, this will subsequently be updated\n",
" // upon first draw.\n",
" this._resize_canvas(600, 600);\n",
"\n",
" // Disable right mouse context menu.\n",
" $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
" return false;\n",
" });\n",
"\n",
" function set_focus () {\n",
" canvas.focus();\n",
" canvas_div.focus();\n",
" }\n",
"\n",
" window.setTimeout(set_focus, 100);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items) {\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) {\n",
" // put a spacer in here.\n",
" continue;\n",
" }\n",
" var button = $('<button/>');\n",
" button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
" 'ui-button-icon-only');\n",
" button.attr('role', 'button');\n",
" button.attr('aria-disabled', 'false');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
"\n",
" var icon_img = $('<span/>');\n",
" icon_img.addClass('ui-button-icon-primary ui-icon');\n",
" icon_img.addClass(image);\n",
" icon_img.addClass('ui-corner-all');\n",
"\n",
" var tooltip_span = $('<span/>');\n",
" tooltip_span.addClass('ui-button-text');\n",
" tooltip_span.html(tooltip);\n",
"\n",
" button.append(icon_img);\n",
" button.append(tooltip_span);\n",
"\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" var fmt_picker_span = $('<span/>');\n",
"\n",
" var fmt_picker = $('<select/>');\n",
" fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
" fmt_picker_span.append(fmt_picker);\n",
" nav_element.append(fmt_picker_span);\n",
" this.format_dropdown = fmt_picker[0];\n",
"\n",
" for (var ind in mpl.extensions) {\n",
" var fmt = mpl.extensions[ind];\n",
" var option = $(\n",
" '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
" fmt_picker.append(option)\n",
" }\n",
"\n",
" // Add hover states to the ui-buttons\n",
" $( \".ui-button\" ).hover(\n",
" function() { $(this).addClass(\"ui-state-hover\");},\n",
" function() { $(this).removeClass(\"ui-state-hover\");}\n",
" );\n",
"\n",
" var status_bar = $('<span class=\"mpl-message\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"}\n",
"\n",
"mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
" // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
" // which will in turn request a refresh of the image.\n",
" this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
"}\n",
"\n",
"mpl.figure.prototype.send_message = function(type, properties) {\n",
" properties['type'] = type;\n",
" properties['figure_id'] = this.id;\n",
" this.ws.send(JSON.stringify(properties));\n",
"}\n",
"\n",
"mpl.figure.prototype.send_draw_message = function() {\n",
" if (!this.waiting) {\n",
" this.waiting = true;\n",
" this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
" }\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" var format_dropdown = fig.format_dropdown;\n",
" var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
" fig.ondownload(fig, format);\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
" var size = msg['size'];\n",
" if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
" fig._resize_canvas(size[0], size[1]);\n",
" fig.send_message(\"refresh\", {});\n",
" };\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
" var x0 = msg['x0'] / mpl.ratio;\n",
" var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
" var x1 = msg['x1'] / mpl.ratio;\n",
" var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
" x0 = Math.floor(x0) + 0.5;\n",
" y0 = Math.floor(y0) + 0.5;\n",
" x1 = Math.floor(x1) + 0.5;\n",
" y1 = Math.floor(y1) + 0.5;\n",
" var min_x = Math.min(x0, x1);\n",
" var min_y = Math.min(y0, y1);\n",
" var width = Math.abs(x1 - x0);\n",
" var height = Math.abs(y1 - y0);\n",
"\n",
" fig.rubberband_context.clearRect(\n",
" 0, 0, fig.canvas.width, fig.canvas.height);\n",
"\n",
" fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
" // Updates the figure title.\n",
" fig.header.textContent = msg['label'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
" var cursor = msg['cursor'];\n",
" switch(cursor)\n",
" {\n",
" case 0:\n",
" cursor = 'pointer';\n",
" break;\n",
" case 1:\n",
" cursor = 'default';\n",
" break;\n",
" case 2:\n",
" cursor = 'crosshair';\n",
" break;\n",
" case 3:\n",
" cursor = 'move';\n",
" break;\n",
" }\n",
" fig.rubberband_canvas.style.cursor = cursor;\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_message = function(fig, msg) {\n",
" fig.message.textContent = msg['message'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
" // Request the server to send over a new figure.\n",
" fig.send_draw_message();\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
" fig.image_mode = msg['mode'];\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Called whenever the canvas gets updated.\n",
" this.send_message(\"ack\", {});\n",
"}\n",
"\n",
"// A function to construct a web socket function for onmessage handling.\n",
"// Called in the figure constructor.\n",
"mpl.figure.prototype._make_on_message_function = function(fig) {\n",
" return function socket_on_message(evt) {\n",
" if (evt.data instanceof Blob) {\n",
" /* FIXME: We get \"Resource interpreted as Image but\n",
" * transferred with MIME type text/plain:\" errors on\n",
" * Chrome. But how to set the MIME type? It doesn't seem\n",
" * to be part of the websocket stream */\n",
" evt.data.type = \"image/png\";\n",
"\n",
" /* Free the memory for the previous frames */\n",
" if (fig.imageObj.src) {\n",
" (window.URL || window.webkitURL).revokeObjectURL(\n",
" fig.imageObj.src);\n",
" }\n",
"\n",
" fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
" evt.data);\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
" else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
" fig.imageObj.src = evt.data;\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
"\n",
" var msg = JSON.parse(evt.data);\n",
" var msg_type = msg['type'];\n",
"\n",
" // Call the \"handle_{type}\" callback, which takes\n",
" // the figure and JSON message as its only arguments.\n",
" try {\n",
" var callback = fig[\"handle_\" + msg_type];\n",
" } catch (e) {\n",
" console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
" return;\n",
" }\n",
"\n",
" if (callback) {\n",
" try {\n",
" // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
" callback(fig, msg);\n",
" } catch (e) {\n",
" console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
" }\n",
" }\n",
" };\n",
"}\n",
"\n",
"// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
"mpl.findpos = function(e) {\n",
" //this section is from http://www.quirksmode.org/js/events_properties.html\n",
" var targ;\n",
" if (!e)\n",
" e = window.event;\n",
" if (e.target)\n",
" targ = e.target;\n",
" else if (e.srcElement)\n",
" targ = e.srcElement;\n",
" if (targ.nodeType == 3) // defeat Safari bug\n",
" targ = targ.parentNode;\n",
"\n",
" // jQuery normalizes the pageX and pageY\n",
" // pageX,Y are the mouse positions relative to the document\n",
" // offset() returns the position of the element relative to the document\n",
" var x = e.pageX - $(targ).offset().left;\n",
" var y = e.pageY - $(targ).offset().top;\n",
"\n",
" return {\"x\": x, \"y\": y};\n",
"};\n",
"\n",
"/*\n",
" * return a copy of an object with only non-object keys\n",
" * we need this to avoid circular references\n",
" * http://stackoverflow.com/a/24161582/3208463\n",
" */\n",
"function simpleKeys (original) {\n",
" return Object.keys(original).reduce(function (obj, key) {\n",
" if (typeof original[key] !== 'object')\n",
" obj[key] = original[key]\n",
" return obj;\n",
" }, {});\n",
"}\n",
"\n",
"mpl.figure.prototype.mouse_event = function(event, name) {\n",
" var canvas_pos = mpl.findpos(event)\n",
"\n",
" if (name === 'button_press')\n",
" {\n",
" this.canvas.focus();\n",
" this.canvas_div.focus();\n",
" }\n",
"\n",
" var x = canvas_pos.x * mpl.ratio;\n",
" var y = canvas_pos.y * mpl.ratio;\n",
"\n",
" this.send_message(name, {x: x, y: y, button: event.button,\n",
" step: event.step,\n",
" guiEvent: simpleKeys(event)});\n",
"\n",
" /* This prevents the web browser from automatically changing to\n",
" * the text insertion cursor when the button is pressed. We want\n",
" * to control all of the cursor setting manually through the\n",
" * 'cursor' event from matplotlib */\n",
" event.preventDefault();\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" // Handle any extra behaviour associated with a key event\n",
"}\n",
"\n",
"mpl.figure.prototype.key_event = function(event, name) {\n",
"\n",
" // Prevent repeat events\n",
" if (name == 'key_press')\n",
" {\n",
" if (event.which === this._key)\n",
" return;\n",
" else\n",
" this._key = event.which;\n",
" }\n",
" if (name == 'key_release')\n",
" this._key = null;\n",
"\n",
" var value = '';\n",
" if (event.ctrlKey && event.which != 17)\n",
" value += \"ctrl+\";\n",
" if (event.altKey && event.which != 18)\n",
" value += \"alt+\";\n",
" if (event.shiftKey && event.which != 16)\n",
" value += \"shift+\";\n",
"\n",
" value += 'k';\n",
" value += event.which.toString();\n",
"\n",
" this._key_event_extra(event, name);\n",
"\n",
" this.send_message(name, {key: value,\n",
" guiEvent: simpleKeys(event)});\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
" if (name == 'download') {\n",
" this.handle_save(this, null);\n",
" } else {\n",
" this.send_message(\"toolbar_button\", {name: name});\n",
" }\n",
"};\n",
"\n",
"mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
" this.message.textContent = tooltip;\n",
"};\n",
"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",
"\n",
"mpl.extensions = [\"eps\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\n",
"\n",
"mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
" // Create a \"websocket\"-like object which calls the given IPython comm\n",
" // object with the appropriate methods. Currently this is a non binary\n",
" // socket, so there is still some room for performance tuning.\n",
" var ws = {};\n",
"\n",
" ws.close = function() {\n",
" comm.close()\n",
" };\n",
" ws.send = function(m) {\n",
" //console.log('sending', m);\n",
" comm.send(m);\n",
" };\n",
" // Register the callback with on_msg.\n",
" comm.on_msg(function(msg) {\n",
" //console.log('receiving', msg['content']['data'], msg);\n",
" // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
" ws.onmessage(msg['content']['data'])\n",
" });\n",
" return ws;\n",
"}\n",
"\n",
"mpl.mpl_figure_comm = function(comm, msg) {\n",
" // This is the function which gets called when the mpl process\n",
" // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
"\n",
" var id = msg.content.data.id;\n",
" // Get hold of the div created by the display call when the Comm\n",
" // socket was opened in Python.\n",
" var element = $(\"#\" + id);\n",
" var ws_proxy = comm_websocket_adapter(comm)\n",
"\n",
" function ondownload(figure, format) {\n",
" window.open(figure.imageObj.src);\n",
" }\n",
"\n",
" var fig = new mpl.figure(id, ws_proxy,\n",
" ondownload,\n",
" element.get(0));\n",
"\n",
" // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
" // web socket which is closed, not our websocket->open comm proxy.\n",
" ws_proxy.onopen();\n",
"\n",
" fig.parent_element = element.get(0);\n",
" fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
" if (!fig.cell_info) {\n",
" console.error(\"Failed to find cell for figure\", id, fig);\n",
" return;\n",
" }\n",
"\n",
" var output_index = fig.cell_info[2]\n",
" var cell = fig.cell_info[0];\n",
"\n",
"};\n",
"\n",
"mpl.figure.prototype.handle_close = function(fig, msg) {\n",
" var width = fig.canvas.width/mpl.ratio\n",
" fig.root.unbind('remove')\n",
"\n",
" // Update the output cell to use the data from the current canvas.\n",
" fig.push_to_output();\n",
" var dataURL = fig.canvas.toDataURL();\n",
" // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
" // the notebook keyboard shortcuts fail.\n",
" IPython.keyboard_manager.enable()\n",
" $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
" fig.close_ws(fig, msg);\n",
"}\n",
"\n",
"mpl.figure.prototype.close_ws = function(fig, msg){\n",
" fig.send_message('closing', msg);\n",
" // fig.ws.close()\n",
"}\n",
"\n",
"mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
" // Turn the data on the canvas into data in the output cell.\n",
" var width = this.canvas.width/mpl.ratio\n",
" var dataURL = this.canvas.toDataURL();\n",
" this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Tell IPython that the notebook contents must change.\n",
" IPython.notebook.set_dirty(true);\n",
" this.send_message(\"ack\", {});\n",
" var fig = this;\n",
" // Wait a second, then push the new image to the DOM so\n",
" // that it is saved nicely (might be nice to debounce this).\n",
" setTimeout(function () { fig.push_to_output() }, 1000);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items){\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) { continue; };\n",
"\n",
" var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" // Add the status bar.\n",
" var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"\n",
" // Add the close button to the window.\n",
" var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
" 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",
" button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
" button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
" buttongrp.append(button);\n",
" var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
" titlebar.prepend(buttongrp);\n",
"}\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(el){\n",
" var fig = this\n",
" el.on(\"remove\", function(){\n",
"\tfig.close_ws(fig, {});\n",
" });\n",
"}\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(el){\n",
" // this is important to make the div 'focusable\n",
" el.attr('tabindex', 0)\n",
" // reach out to IPython and tell the keyboard manager to turn it's self\n",
" // off when our div gets focus\n",
"\n",
" // location in version 3\n",
" if (IPython.notebook.keyboard_manager) {\n",
" IPython.notebook.keyboard_manager.register_events(el);\n",
" }\n",
" else {\n",
" // location in version 2\n",
" IPython.keyboard_manager.register_events(el);\n",
" }\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" var manager = IPython.notebook.keyboard_manager;\n",
" if (!manager)\n",
" manager = IPython.keyboard_manager;\n",
"\n",
" // Check for shift+enter\n",
" if (event.shiftKey && event.which == 13) {\n",
" this.canvas_div.blur();\n",
" event.shiftKey = false;\n",
" // Send a \"J\" for go to next cell\n",
" event.which = 74;\n",
" event.keyCode = 74;\n",
" manager.command_mode();\n",
" manager.handle_keydown(event);\n",
" }\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" fig.ondownload(fig, null);\n",
"}\n",
"\n",
"\n",
"mpl.find_output_cell = function(html_output) {\n",
" // Return the cell and output element which can be found *uniquely* in the notebook.\n",
" // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
" // IPython event is triggered only after the cells have been serialised, which for\n",
" // our purposes (turning an active figure into a static one), is too late.\n",
" var cells = IPython.notebook.get_cells();\n",
" var ncells = cells.length;\n",
" for (var i=0; i<ncells; i++) {\n",
" var cell = cells[i];\n",
" if (cell.cell_type === 'code'){\n",
" for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
" var data = cell.output_area.outputs[j];\n",
" if (data.data) {\n",
" // IPython >= 3 moved mimebundle to data attribute of output\n",
" data = data.data;\n",
" }\n",
" if (data['text/html'] == html_output) {\n",
" return [cell, data, j];\n",
" }\n",
" }\n",
" }\n",
" }\n",
"}\n",
"\n",
"// Register the function which deals with the matplotlib target/channel.\n",
"// The kernel may be null if the page has been refreshed.\n",
"if (IPython.notebook.kernel != null) {\n",
" IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
"}\n"
],
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"text/html": [
"<img src=\"\" width=\"600\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"# Speicherausbau je Technologie und Knoten\n",
"\n",
"storage_expansion(network, scaling=100)"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"# Bewertung von NEP Maßnahmen <a class=\"anchor\" id=\"nep\"></a>\n",
"\n",
"## Fragestellung\n",
"\n",
"### Wie werden die Maßnahmen des NEP 2035 B2 in eTraGo bewertet?\n",
"\n",
"\n",
"## Daraus resultierende Einstellungen\n",
"\n",
"- Auswahl von Szenario NEP 2035\n",
"- geplante Leitungen nach dem NEP 2035 B2 hinzufügen\n",
"- Rückbau aus dem Bestandsnetz nach dem NEP 2035 B2\n",
"- technisch-wirtschaftliche Optimierung der geplanten Leitungen"
]
},
{
"cell_type": "code",
"execution_count": 15,
"metadata": {},
"outputs": [],
"source": [
"args_nep = {\n",
" # Setup and Configuration:\n",
" 'db': 'oedb', # database session\n",
" 'gridversion': None, # None for model_draft or Version number\n",
" 'method': 'lopf', # lopf or pf\n",
" 'pf_post_lopf': False, # perform a pf after a lopf simulation\n",
" 'start_snapshot': 48,\n",
" 'end_snapshot': 72,\n",
" 'solver': 'gurobi', # glpk, cplex or gurobi\n",
" 'solver_options': {'BarConvTol': 1.e-5, 'FeasibilityTol': 1.e-5,\n",
" 'method':2, 'crossover':0,\n",
" 'logFile': 'solver.log'}, # {} for default options\n",
" 'scn_name': 'NEP 2035', # a scenario: Status Quo, NEP 2035, eGo 100\n",
" # Scenario variations:\n",
" 'scn_extension': ['nep2035_b2', 'BE_NO_NEP 2035'], # None or array of extension scenarios\n",
" 'scn_decommissioning': 'nep2035_b2', # None or decommissioning scenario \n",
" # Export options:\n",
" 'lpfile': False, # save pyomo's lp file: False or /path/tofolder\n",
" 'csv_export': False, # save results as csv: False or /path/tofolder\n",
" 'db_export': False, # export the results back to the oedb\n",
" # Settings:\n",
" 'extendable': ['overlay_network'], # Array of components to optimize\n",
" 'generator_noise': 789456, # apply generator noise, False or seed number\n",
" 'minimize_loading': False,\n",
" 'ramp_limits': False, # Choose if using ramp limit of generators\n",
" 'extra_functionality': None, # Choose function name or None\n",
" # Clustering:\n",
" 'network_clustering_kmeans': 500, # False or the value k for clustering\n",
" 'load_cluster': 'load_cluster_nep_ext_500', # False or predefined busmap for k-means\n",
" 'network_clustering_ehv': False, # clustering of HV buses to EHV buses.\n",
" 'disaggregation': None, # None, 'mini' or 'uniform'\n",
" 'snapshot_clustering': False, # False or the number of 'periods'\n",
" # Simplifications:\n",
" 'parallelisation': False, # run snapshots parallely.\n",
" 'skip_snapshots': False,\n",
" 'line_grouping': False, # group lines parallel lines\n",
" 'branch_capacity_factor': {'HV': 0.5, 'eHV': 0.7}, # p.u. branch derating\n",
" 'load_shedding': True, # meet the demand at value of loss load cost\n",
" 'foreign_lines': {'carrier': 'AC', 'capacity': 'osmTGmod'},\n",
" 'comments': None}"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Ausführen der Berechnung"
]
},
{
"cell_type": "code",
"execution_count": 16,
"metadata": {},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"No data for StoragePqSet in column p_set.\n",
"Series p_set of component StorageUnit could not be imported\n",
"No data for GeneratorPqSet in column p_set.\n",
"Series p_set of component Generator could not be imported\n",
"No data for GeneratorPqSet in column p_max_pu.\n",
"Series p_max_pu of component Generator could not be imported\n",
"No data for LoadPqSet in column p_set.\n",
"Series p_set of component Load could not be imported\n",
"No data for LoadPqSet in column q_set.\n",
"Series q_set of component Load could not be imported\n",
"No data for StoragePqSet in column p_set.\n",
"Series p_set of component StorageUnit could not be imported\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"ERROR:pypsa.io:Error, new components for Carrier are not unique\n"
]
},
{
"name": "stdout",
"output_type": "stream",
"text": [
"No data for StoragePqSet in column p_set.\n",
"Series p_set of component StorageUnit could not be imported\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"ERROR:pypsa.io:Error, new components for Carrier are not unique\n"
]
},
{
"name": "stdout",
"output_type": "stream",
"text": [
"start k-mean clustering\n",
"Inertia of k-means = 423.1414097481512\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"INFO:pypsa.pf:Slack bus for sub-network 0 is 0\n",
"INFO:pypsa.pf:Slack bus for sub-network 1 is 13\n",
"INFO:pypsa.pf:Slack bus for sub-network 2 is 4\n",
"INFO:pypsa.opf:Performed preliminary steps\n",
"INFO:pypsa.opf:Building pyomo model using `angles` formulation\n",
"INFO:pypsa.opf:Solving model using gurobi\n",
"INFO:pypsa.opf:Optimization successful\n"
]
},
{
"name": "stdout",
"output_type": "stream",
"text": [
"Time for LOPF [min]: 1.18\n",
"Investment costs for all lines and transformers in selected snapshots [EUR]: 2047.98\n"
]
}
],
"source": [
"# Start eTraGo calculation with args setting\n",
"# create network object which incluedes all input and output data\n",
"network, disagg_network = etrago(args_nep)"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Berechnung des absoluten Netzausbaus"
]
},
{
"cell_type": "code",
"execution_count": 17,
"metadata": {},
"outputs": [
{
"data": {
"text/plain": [
"9159.6612861311"
]
},
"execution_count": 17,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"(network.lines.s_nom_opt - network.lines.s_nom_min)[network.lines.s_nom_extendable].sum()\\\n",
"+ (network.links.p_nom_opt - network.links.p_nom_min)[network.links.p_nom_extendable].sum()"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"### Darstellung des Ausbaus der geplanten Leitungen"
]
},
{
"cell_type": "code",
"execution_count": 18,
"metadata": {},
"outputs": [
{
"data": {
"application/javascript": [
"/* Put everything inside the global mpl namespace */\n",
"window.mpl = {};\n",
"\n",
"\n",
"mpl.get_websocket_type = function() {\n",
" if (typeof(WebSocket) !== 'undefined') {\n",
" return WebSocket;\n",
" } else if (typeof(MozWebSocket) !== 'undefined') {\n",
" return MozWebSocket;\n",
" } else {\n",
" alert('Your browser does not have WebSocket support.' +\n",
" 'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
" 'Firefox 4 and 5 are also supported but you ' +\n",
" 'have to enable WebSockets in about:config.');\n",
" };\n",
"}\n",
"\n",
"mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
" this.id = figure_id;\n",
"\n",
" this.ws = websocket;\n",
"\n",
" this.supports_binary = (this.ws.binaryType != undefined);\n",
"\n",
" if (!this.supports_binary) {\n",
" var warnings = document.getElementById(\"mpl-warnings\");\n",
" if (warnings) {\n",
" warnings.style.display = 'block';\n",
" warnings.textContent = (\n",
" \"This browser does not support binary websocket messages. \" +\n",
" \"Performance may be slow.\");\n",
" }\n",
" }\n",
"\n",
" this.imageObj = new Image();\n",
"\n",
" this.context = undefined;\n",
" this.message = undefined;\n",
" this.canvas = undefined;\n",
" this.rubberband_canvas = undefined;\n",
" this.rubberband_context = undefined;\n",
" this.format_dropdown = undefined;\n",
"\n",
" this.image_mode = 'full';\n",
"\n",
" this.root = $('<div/>');\n",
" this._root_extra_style(this.root)\n",
" this.root.attr('style', 'display: inline-block');\n",
"\n",
" $(parent_element).append(this.root);\n",
"\n",
" this._init_header(this);\n",
" this._init_canvas(this);\n",
" this._init_toolbar(this);\n",
"\n",
" var fig = this;\n",
"\n",
" this.waiting = false;\n",
"\n",
" this.ws.onopen = function () {\n",
" fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
" fig.send_message(\"send_image_mode\", {});\n",
" if (mpl.ratio != 1) {\n",
" fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
" }\n",
" fig.send_message(\"refresh\", {});\n",
" }\n",
"\n",
" this.imageObj.onload = function() {\n",
" if (fig.image_mode == 'full') {\n",
" // Full images could contain transparency (where diff images\n",
" // almost always do), so we need to clear the canvas so that\n",
" // there is no ghosting.\n",
" fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
" }\n",
" fig.context.drawImage(fig.imageObj, 0, 0);\n",
" };\n",
"\n",
" this.imageObj.onunload = function() {\n",
" fig.ws.close();\n",
" }\n",
"\n",
" this.ws.onmessage = this._make_on_message_function(this);\n",
"\n",
" this.ondownload = ondownload;\n",
"}\n",
"\n",
"mpl.figure.prototype._init_header = function() {\n",
" var titlebar = $(\n",
" '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
" 'ui-helper-clearfix\"/>');\n",
" var titletext = $(\n",
" '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
" 'text-align: center; padding: 3px;\"/>');\n",
" titlebar.append(titletext)\n",
" this.root.append(titlebar);\n",
" this.header = titletext[0];\n",
"}\n",
"\n",
"\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._init_canvas = function() {\n",
" var fig = this;\n",
"\n",
" var canvas_div = $('<div/>');\n",
"\n",
" canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
"\n",
" function canvas_keyboard_event(event) {\n",
" return fig.key_event(event, event['data']);\n",
" }\n",
"\n",
" canvas_div.keydown('key_press', canvas_keyboard_event);\n",
" canvas_div.keyup('key_release', canvas_keyboard_event);\n",
" this.canvas_div = canvas_div\n",
" this._canvas_extra_style(canvas_div)\n",
" this.root.append(canvas_div);\n",
"\n",
" var canvas = $('<canvas/>');\n",
" canvas.addClass('mpl-canvas');\n",
" canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
"\n",
" this.canvas = canvas[0];\n",
" this.context = canvas[0].getContext(\"2d\");\n",
"\n",
" var backingStore = this.context.backingStorePixelRatio ||\n",
"\tthis.context.webkitBackingStorePixelRatio ||\n",
"\tthis.context.mozBackingStorePixelRatio ||\n",
"\tthis.context.msBackingStorePixelRatio ||\n",
"\tthis.context.oBackingStorePixelRatio ||\n",
"\tthis.context.backingStorePixelRatio || 1;\n",
"\n",
" mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
"\n",
" var rubberband = $('<canvas/>');\n",
" rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
"\n",
" var pass_mouse_events = true;\n",
"\n",
" canvas_div.resizable({\n",
" start: function(event, ui) {\n",
" pass_mouse_events = false;\n",
" },\n",
" resize: function(event, ui) {\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" stop: function(event, ui) {\n",
" pass_mouse_events = true;\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" });\n",
"\n",
" function mouse_event_fn(event) {\n",
" if (pass_mouse_events)\n",
" return fig.mouse_event(event, event['data']);\n",
" }\n",
"\n",
" rubberband.mousedown('button_press', mouse_event_fn);\n",
" rubberband.mouseup('button_release', mouse_event_fn);\n",
" // Throttle sequential mouse events to 1 every 20ms.\n",
" rubberband.mousemove('motion_notify', mouse_event_fn);\n",
"\n",
" rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
" rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
"\n",
" canvas_div.on(\"wheel\", function (event) {\n",
" event = event.originalEvent;\n",
" event['data'] = 'scroll'\n",
" if (event.deltaY < 0) {\n",
" event.step = 1;\n",
" } else {\n",
" event.step = -1;\n",
" }\n",
" mouse_event_fn(event);\n",
" });\n",
"\n",
" canvas_div.append(canvas);\n",
" canvas_div.append(rubberband);\n",
"\n",
" this.rubberband = rubberband;\n",
" this.rubberband_canvas = rubberband[0];\n",
" this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
" this.rubberband_context.strokeStyle = \"#000000\";\n",
"\n",
" this._resize_canvas = function(width, height) {\n",
" // Keep the size of the canvas, canvas container, and rubber band\n",
" // canvas in synch.\n",
" canvas_div.css('width', width)\n",
" canvas_div.css('height', height)\n",
"\n",
" canvas.attr('width', width * mpl.ratio);\n",
" canvas.attr('height', height * mpl.ratio);\n",
" canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
"\n",
" rubberband.attr('width', width);\n",
" rubberband.attr('height', height);\n",
" }\n",
"\n",
" // Set the figure to an initial 600x600px, this will subsequently be updated\n",
" // upon first draw.\n",
" this._resize_canvas(600, 600);\n",
"\n",
" // Disable right mouse context menu.\n",
" $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
" return false;\n",
" });\n",
"\n",
" function set_focus () {\n",
" canvas.focus();\n",
" canvas_div.focus();\n",
" }\n",
"\n",
" window.setTimeout(set_focus, 100);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items) {\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) {\n",
" // put a spacer in here.\n",
" continue;\n",
" }\n",
" var button = $('<button/>');\n",
" button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
" 'ui-button-icon-only');\n",
" button.attr('role', 'button');\n",
" button.attr('aria-disabled', 'false');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
"\n",
" var icon_img = $('<span/>');\n",
" icon_img.addClass('ui-button-icon-primary ui-icon');\n",
" icon_img.addClass(image);\n",
" icon_img.addClass('ui-corner-all');\n",
"\n",
" var tooltip_span = $('<span/>');\n",
" tooltip_span.addClass('ui-button-text');\n",
" tooltip_span.html(tooltip);\n",
"\n",
" button.append(icon_img);\n",
" button.append(tooltip_span);\n",
"\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" var fmt_picker_span = $('<span/>');\n",
"\n",
" var fmt_picker = $('<select/>');\n",
" fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
" fmt_picker_span.append(fmt_picker);\n",
" nav_element.append(fmt_picker_span);\n",
" this.format_dropdown = fmt_picker[0];\n",
"\n",
" for (var ind in mpl.extensions) {\n",
" var fmt = mpl.extensions[ind];\n",
" var option = $(\n",
" '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
" fmt_picker.append(option)\n",
" }\n",
"\n",
" // Add hover states to the ui-buttons\n",
" $( \".ui-button\" ).hover(\n",
" function() { $(this).addClass(\"ui-state-hover\");},\n",
" function() { $(this).removeClass(\"ui-state-hover\");}\n",
" );\n",
"\n",
" var status_bar = $('<span class=\"mpl-message\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"}\n",
"\n",
"mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
" // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
" // which will in turn request a refresh of the image.\n",
" this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
"}\n",
"\n",
"mpl.figure.prototype.send_message = function(type, properties) {\n",
" properties['type'] = type;\n",
" properties['figure_id'] = this.id;\n",
" this.ws.send(JSON.stringify(properties));\n",
"}\n",
"\n",
"mpl.figure.prototype.send_draw_message = function() {\n",
" if (!this.waiting) {\n",
" this.waiting = true;\n",
" this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
" }\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" var format_dropdown = fig.format_dropdown;\n",
" var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
" fig.ondownload(fig, format);\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
" var size = msg['size'];\n",
" if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
" fig._resize_canvas(size[0], size[1]);\n",
" fig.send_message(\"refresh\", {});\n",
" };\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
" var x0 = msg['x0'] / mpl.ratio;\n",
" var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
" var x1 = msg['x1'] / mpl.ratio;\n",
" var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
" x0 = Math.floor(x0) + 0.5;\n",
" y0 = Math.floor(y0) + 0.5;\n",
" x1 = Math.floor(x1) + 0.5;\n",
" y1 = Math.floor(y1) + 0.5;\n",
" var min_x = Math.min(x0, x1);\n",
" var min_y = Math.min(y0, y1);\n",
" var width = Math.abs(x1 - x0);\n",
" var height = Math.abs(y1 - y0);\n",
"\n",
" fig.rubberband_context.clearRect(\n",
" 0, 0, fig.canvas.width, fig.canvas.height);\n",
"\n",
" fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
" // Updates the figure title.\n",
" fig.header.textContent = msg['label'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
" var cursor = msg['cursor'];\n",
" switch(cursor)\n",
" {\n",
" case 0:\n",
" cursor = 'pointer';\n",
" break;\n",
" case 1:\n",
" cursor = 'default';\n",
" break;\n",
" case 2:\n",
" cursor = 'crosshair';\n",
" break;\n",
" case 3:\n",
" cursor = 'move';\n",
" break;\n",
" }\n",
" fig.rubberband_canvas.style.cursor = cursor;\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_message = function(fig, msg) {\n",
" fig.message.textContent = msg['message'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
" // Request the server to send over a new figure.\n",
" fig.send_draw_message();\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
" fig.image_mode = msg['mode'];\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Called whenever the canvas gets updated.\n",
" this.send_message(\"ack\", {});\n",
"}\n",
"\n",
"// A function to construct a web socket function for onmessage handling.\n",
"// Called in the figure constructor.\n",
"mpl.figure.prototype._make_on_message_function = function(fig) {\n",
" return function socket_on_message(evt) {\n",
" if (evt.data instanceof Blob) {\n",
" /* FIXME: We get \"Resource interpreted as Image but\n",
" * transferred with MIME type text/plain:\" errors on\n",
" * Chrome. But how to set the MIME type? It doesn't seem\n",
" * to be part of the websocket stream */\n",
" evt.data.type = \"image/png\";\n",
"\n",
" /* Free the memory for the previous frames */\n",
" if (fig.imageObj.src) {\n",
" (window.URL || window.webkitURL).revokeObjectURL(\n",
" fig.imageObj.src);\n",
" }\n",
"\n",
" fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
" evt.data);\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
" else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
" fig.imageObj.src = evt.data;\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
"\n",
" var msg = JSON.parse(evt.data);\n",
" var msg_type = msg['type'];\n",
"\n",
" // Call the \"handle_{type}\" callback, which takes\n",
" // the figure and JSON message as its only arguments.\n",
" try {\n",
" var callback = fig[\"handle_\" + msg_type];\n",
" } catch (e) {\n",
" console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
" return;\n",
" }\n",
"\n",
" if (callback) {\n",
" try {\n",
" // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
" callback(fig, msg);\n",
" } catch (e) {\n",
" console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
" }\n",
" }\n",
" };\n",
"}\n",
"\n",
"// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
"mpl.findpos = function(e) {\n",
" //this section is from http://www.quirksmode.org/js/events_properties.html\n",
" var targ;\n",
" if (!e)\n",
" e = window.event;\n",
" if (e.target)\n",
" targ = e.target;\n",
" else if (e.srcElement)\n",
" targ = e.srcElement;\n",
" if (targ.nodeType == 3) // defeat Safari bug\n",
" targ = targ.parentNode;\n",
"\n",
" // jQuery normalizes the pageX and pageY\n",
" // pageX,Y are the mouse positions relative to the document\n",
" // offset() returns the position of the element relative to the document\n",
" var x = e.pageX - $(targ).offset().left;\n",
" var y = e.pageY - $(targ).offset().top;\n",
"\n",
" return {\"x\": x, \"y\": y};\n",
"};\n",
"\n",
"/*\n",
" * return a copy of an object with only non-object keys\n",
" * we need this to avoid circular references\n",
" * http://stackoverflow.com/a/24161582/3208463\n",
" */\n",
"function simpleKeys (original) {\n",
" return Object.keys(original).reduce(function (obj, key) {\n",
" if (typeof original[key] !== 'object')\n",
" obj[key] = original[key]\n",
" return obj;\n",
" }, {});\n",
"}\n",
"\n",
"mpl.figure.prototype.mouse_event = function(event, name) {\n",
" var canvas_pos = mpl.findpos(event)\n",
"\n",
" if (name === 'button_press')\n",
" {\n",
" this.canvas.focus();\n",
" this.canvas_div.focus();\n",
" }\n",
"\n",
" var x = canvas_pos.x * mpl.ratio;\n",
" var y = canvas_pos.y * mpl.ratio;\n",
"\n",
" this.send_message(name, {x: x, y: y, button: event.button,\n",
" step: event.step,\n",
" guiEvent: simpleKeys(event)});\n",
"\n",
" /* This prevents the web browser from automatically changing to\n",
" * the text insertion cursor when the button is pressed. We want\n",
" * to control all of the cursor setting manually through the\n",
" * 'cursor' event from matplotlib */\n",
" event.preventDefault();\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" // Handle any extra behaviour associated with a key event\n",
"}\n",
"\n",
"mpl.figure.prototype.key_event = function(event, name) {\n",
"\n",
" // Prevent repeat events\n",
" if (name == 'key_press')\n",
" {\n",
" if (event.which === this._key)\n",
" return;\n",
" else\n",
" this._key = event.which;\n",
" }\n",
" if (name == 'key_release')\n",
" this._key = null;\n",
"\n",
" var value = '';\n",
" if (event.ctrlKey && event.which != 17)\n",
" value += \"ctrl+\";\n",
" if (event.altKey && event.which != 18)\n",
" value += \"alt+\";\n",
" if (event.shiftKey && event.which != 16)\n",
" value += \"shift+\";\n",
"\n",
" value += 'k';\n",
" value += event.which.toString();\n",
"\n",
" this._key_event_extra(event, name);\n",
"\n",
" this.send_message(name, {key: value,\n",
" guiEvent: simpleKeys(event)});\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
" if (name == 'download') {\n",
" this.handle_save(this, null);\n",
" } else {\n",
" this.send_message(\"toolbar_button\", {name: name});\n",
" }\n",
"};\n",
"\n",
"mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
" this.message.textContent = tooltip;\n",
"};\n",
"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",
"\n",
"mpl.extensions = [\"eps\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\n",
"\n",
"mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
" // Create a \"websocket\"-like object which calls the given IPython comm\n",
" // object with the appropriate methods. Currently this is a non binary\n",
" // socket, so there is still some room for performance tuning.\n",
" var ws = {};\n",
"\n",
" ws.close = function() {\n",
" comm.close()\n",
" };\n",
" ws.send = function(m) {\n",
" //console.log('sending', m);\n",
" comm.send(m);\n",
" };\n",
" // Register the callback with on_msg.\n",
" comm.on_msg(function(msg) {\n",
" //console.log('receiving', msg['content']['data'], msg);\n",
" // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
" ws.onmessage(msg['content']['data'])\n",
" });\n",
" return ws;\n",
"}\n",
"\n",
"mpl.mpl_figure_comm = function(comm, msg) {\n",
" // This is the function which gets called when the mpl process\n",
" // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
"\n",
" var id = msg.content.data.id;\n",
" // Get hold of the div created by the display call when the Comm\n",
" // socket was opened in Python.\n",
" var element = $(\"#\" + id);\n",
" var ws_proxy = comm_websocket_adapter(comm)\n",
"\n",
" function ondownload(figure, format) {\n",
" window.open(figure.imageObj.src);\n",
" }\n",
"\n",
" var fig = new mpl.figure(id, ws_proxy,\n",
" ondownload,\n",
" element.get(0));\n",
"\n",
" // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
" // web socket which is closed, not our websocket->open comm proxy.\n",
" ws_proxy.onopen();\n",
"\n",
" fig.parent_element = element.get(0);\n",
" fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
" if (!fig.cell_info) {\n",
" console.error(\"Failed to find cell for figure\", id, fig);\n",
" return;\n",
" }\n",
"\n",
" var output_index = fig.cell_info[2]\n",
" var cell = fig.cell_info[0];\n",
"\n",
"};\n",
"\n",
"mpl.figure.prototype.handle_close = function(fig, msg) {\n",
" var width = fig.canvas.width/mpl.ratio\n",
" fig.root.unbind('remove')\n",
"\n",
" // Update the output cell to use the data from the current canvas.\n",
" fig.push_to_output();\n",
" var dataURL = fig.canvas.toDataURL();\n",
" // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
" // the notebook keyboard shortcuts fail.\n",
" IPython.keyboard_manager.enable()\n",
" $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
" fig.close_ws(fig, msg);\n",
"}\n",
"\n",
"mpl.figure.prototype.close_ws = function(fig, msg){\n",
" fig.send_message('closing', msg);\n",
" // fig.ws.close()\n",
"}\n",
"\n",
"mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
" // Turn the data on the canvas into data in the output cell.\n",
" var width = this.canvas.width/mpl.ratio\n",
" var dataURL = this.canvas.toDataURL();\n",
" this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Tell IPython that the notebook contents must change.\n",
" IPython.notebook.set_dirty(true);\n",
" this.send_message(\"ack\", {});\n",
" var fig = this;\n",
" // Wait a second, then push the new image to the DOM so\n",
" // that it is saved nicely (might be nice to debounce this).\n",
" setTimeout(function () { fig.push_to_output() }, 1000);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items){\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) { continue; };\n",
"\n",
" var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" // Add the status bar.\n",
" var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"\n",
" // Add the close button to the window.\n",
" var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
" 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",
" button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
" button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
" buttongrp.append(button);\n",
" var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
" titlebar.prepend(buttongrp);\n",
"}\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(el){\n",
" var fig = this\n",
" el.on(\"remove\", function(){\n",
"\tfig.close_ws(fig, {});\n",
" });\n",
"}\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(el){\n",
" // this is important to make the div 'focusable\n",
" el.attr('tabindex', 0)\n",
" // reach out to IPython and tell the keyboard manager to turn it's self\n",
" // off when our div gets focus\n",
"\n",
" // location in version 3\n",
" if (IPython.notebook.keyboard_manager) {\n",
" IPython.notebook.keyboard_manager.register_events(el);\n",
" }\n",
" else {\n",
" // location in version 2\n",
" IPython.keyboard_manager.register_events(el);\n",
" }\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" var manager = IPython.notebook.keyboard_manager;\n",
" if (!manager)\n",
" manager = IPython.keyboard_manager;\n",
"\n",
" // Check for shift+enter\n",
" if (event.shiftKey && event.which == 13) {\n",
" this.canvas_div.blur();\n",
" event.shiftKey = false;\n",
" // Send a \"J\" for go to next cell\n",
" event.which = 74;\n",
" event.keyCode = 74;\n",
" manager.command_mode();\n",
" manager.handle_keydown(event);\n",
" }\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" fig.ondownload(fig, null);\n",
"}\n",
"\n",
"\n",
"mpl.find_output_cell = function(html_output) {\n",
" // Return the cell and output element which can be found *uniquely* in the notebook.\n",
" // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
" // IPython event is triggered only after the cells have been serialised, which for\n",
" // our purposes (turning an active figure into a static one), is too late.\n",
" var cells = IPython.notebook.get_cells();\n",
" var ncells = cells.length;\n",
" for (var i=0; i<ncells; i++) {\n",
" var cell = cells[i];\n",
" if (cell.cell_type === 'code'){\n",
" for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
" var data = cell.output_area.outputs[j];\n",
" if (data.data) {\n",
" // IPython >= 3 moved mimebundle to data attribute of output\n",
" data = data.data;\n",
" }\n",
" if (data['text/html'] == html_output) {\n",
" return [cell, data, j];\n",
" }\n",
" }\n",
" }\n",
" }\n",
"}\n",
"\n",
"// Register the function which deals with the matplotlib target/channel.\n",
"// The kernel may be null if the page has been refreshed.\n",
"if (IPython.notebook.kernel != null) {\n",
" IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
"}\n"
],
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"text/html": [
"<img src=\"\" width=\"640\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"network_expansion(network, ext_min = 0)"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {},
"outputs": [],
"source": []
}
],
"metadata": {
"kernelspec": {
"display_name": "Python 3",
"language": "python",
"name": "python3"
},
"language_info": {
"codemirror_mode": {
"name": "ipython",
"version": 3
},
"file_extension": ".py",
"mimetype": "text/x-python",
"name": "python",
"nbconvert_exporter": "python",
"pygments_lexer": "ipython3",
"version": "3.5.2"
}
},
"nbformat": 4,
"nbformat_minor": 2
}
1.718597517623357440e+00 4.677365176526153334e+01
1.067531829893709627e+01 5.056571306637595598e+01
1.500022650919206768e+01 6.143341194445713427e+01
1.547499799265967724e+01 4.987102379619906145e+01
7.139557746315380982e+00 5.128279931637736411e+01
1.913433317871253792e+01 5.187840778748378767e+01
8.190205840999903231e+00 4.676623667700120279e+01
1.415840506158733980e+01 4.758712955629636099e+01
9.754817051932453964e+00 5.339887073705135379e+01
5.330228000004415989e+00 5.216902743576459756e+01
1.323721858550624653e+01 5.253431008041768280e+01
9.295835860629399150e+00 4.877158828885264796e+01
9.348121455999978124e+00 5.623445518800006937e+01
1.218220690792394123e+01 4.857604106848932446e+01
8.506755351266109599e+00 4.944894478026429141e+01
1.331134366541817116e+01 5.082695647357567736e+01
6.133266749986721500e+00 4.975290147194732526e+01
8.723043021596323499e+00 5.171371052402903246e+01
1.010511134533769884e+01 4.783339216368332103e+01
7.415303905862432821e+00 5.312813637019765878e+01
1.443213522214047728e+01 5.187014052479649706e+01
1.189286380284750066e+01 5.140547525808032958e+01
1.040244637394797955e+01 5.215249905904806127e+01
1.218072595318001206e+01 5.411073806838407307e+01
1.107493943048695684e+01 4.942889144774621002e+01
6.667962916223881820e+00 5.098190810853264310e+01
8.811608665558086528e+00 5.309934751650719420e+01
1.162510685187220361e+01 4.819185795467375755e+01
7.030952552370240305e+00 4.933469533206685043e+01
7.836866403487741550e+00 4.802817267444572735e+01
1.368150165364660964e+01 5.413695698048164928e+01
8.966631726595572616e+00 5.009518485458306003e+01
9.470846678270055108e+00 5.127921657938991729e+01
8.090322305200668751e+00 5.008868380290426359e+01
8.191546527363428254e+00 5.197779951263424181e+01
1.275414759085909822e+01 4.815732806772010832e+01
1.015976010784095251e+01 5.429797275667014134e+01
9.413008649235823810e+00 5.209489328886347437e+01
8.258288184026799428e+00 4.887671658702936384e+01
7.997826707393323353e+00 5.089264690952042258e+01
6.656702995243344745e+00 5.137966554081549475e+01
1.176135882420794765e+01 5.054528397810280893e+01
1.233354258181244489e+01 5.119199466460713666e+01
9.571939343810329959e+00 4.979629571057178339e+01
7.083265839295895816e+00 5.215364142347021925e+01
1.127772110542601069e+01 4.749587269567864922e+01
1.178494614617890690e+01 5.299595484914072330e+01
1.118350414015091587e+01 4.873329139046302316e+01
7.475942440402747025e+00 5.151430424277985765e+01
1.164158502065901146e+01 5.191045120716407268e+01
9.097362982308496271e+00 4.775483681493664534e+01
7.083065740859785464e+00 5.070919343478363572e+01
1.350323951032634540e+01 4.862164880988955673e+01
1.019453136322014863e+01 5.002591654445082270e+01
7.803881630135126812e+00 4.760007239425053172e+01
1.288213875624579252e+01 5.052471152576115543e+01
1.250546329347450225e+01 5.239926257673400301e+01
1.229389759454472753e+01 4.955674420993702967e+01
9.743225092658249764e+00 5.091638746004856841e+01
7.954604097574437205e+00 5.167825279059565702e+01
9.746551707602849390e+00 5.237306082575450716e+01
1.423200154406821127e+01 5.309120901351388966e+01
1.023013229797363266e+01 4.852708879852385593e+01
9.429630928333684992e+00 5.477511148690742715e+01
1.451383765549025995e+01 5.141343755585167230e+01
6.981845939920181543e+00 5.103048791859319522e+01
8.775833923017708926e+00 4.854618929372767155e+01
8.733026374163603833e+00 5.381663035187061439e+01
8.142850423798403625e+00 5.356365957069631634e+01
1.179123046343980619e+01 4.881513973965846986e+01
1.071483285116728013e+01 5.269430100437774200e+01
8.538948061274394874e+00 5.009364861047794193e+01
8.851321482923491857e+00 5.218171775475091323e+01
7.825188609515912219e+00 5.044276125599910898e+01
8.882136387495716434e+00 5.124481420653047792e+01
7.040005085204025015e+00 5.160211350850764234e+01
1.023538232041427776e+01 4.755467341545151072e+01
7.260897993609603418e+00 5.272442080511184770e+01
9.677247923620127423e+00 4.789361708292303632e+01
1.393665999136478817e+01 5.104501228423220027e+01
1.021504089889286959e+01 5.344830624183729384e+01
1.112913394345896023e+01 5.374197292227026423e+01
1.090742569182611099e+01 4.837943074838896962e+01
1.024909111350410917e+01 5.170423716927455615e+01
6.300931165536307610e+00 5.082425806868419471e+01
7.032209204123440749e+00 4.997277724635604557e+01
1.260871444243037054e+01 4.889286224984675755e+01
9.197070791287353941e+00 4.895695779390237590e+01
1.101601980448860552e+01 5.095965540564107954e+01
1.123989195103435179e+01 5.029398287700889369e+01
7.817210061573878654e+00 4.960551795191456392e+01
8.360890879488643179e+00 5.274596541473275124e+01
1.215604507478259499e+01 4.802374288009585968e+01
7.621078899830521003e+00 5.192930465297524023e+01
8.258904978879899161e+00 4.764066641397631230e+01
7.817073356858853295e+00 4.833137794702497558e+01
1.357885163210007207e+01 5.323044103324681942e+01
1.074293688106172162e+01 5.214950704762500067e+01
9.487782648579145217e+00 5.363647639989332561e+01
1.353067137928047714e+01 5.247495789430475099e+01
1.100417431498918752e+01 5.051494730552607137e+01
1.083232437588034003e+01 4.773374551573067492e+01
1.006924015706229980e+01 4.906304316051823378e+01
8.695537061012538871e+00 5.055939265020230522e+01
6.840814501199789177e+00 5.086350831834055697e+01
1.290152829429926840e+01 5.083959759781084387e+01
7.596682697895370318e+00 5.094914088324583901e+01
1.365722252622669863e+01 5.160077083201678505e+01
1.229339050540800926e+01 5.162949254619761774e+01
1.212168270321060248e+01 4.929131082932834573e+01
1.059939074139389348e+01 5.378972112351307544e+01
8.989784397629515311e+00 5.237408873175138524e+01
9.734623835729506069e+00 4.829305100569595055e+01
1.155120242574185241e+01 4.998170055643103638e+01
8.679966826819837777e+00 4.812336437220831442e+01
6.777796995381598855e+00 4.929621568147647537e+01
6.174287357120022079e+00 5.176819589164826141e+01
6.322218956231838938e+00 4.996817002275552255e+01
1.078114748027904923e+01 5.145299109102384705e+01
1.308014275552683969e+01 5.426001153463553806e+01
7.315896608591161865e+00 5.161763027409202209e+01
1.144229407033136425e+01 5.095158661555657176e+01
9.065584569607903376e+00 4.849078485361094693e+01
9.899301437108327661e+00 5.190154865967750908e+01
1.064307631207995541e+01 4.926384357997891073e+01
9.185345733238921895e+00 5.287632960071668009e+01
1.203432359240150973e+01 5.086945291848947903e+01
1.158264264641987573e+01 4.876183821685614106e+01
1.012873236945089417e+01 5.081063734574752289e+01
9.991676236248688170e+00 4.972965778467702336e+01
6.791110272204059939e+00 5.113917032185602807e+01
8.022900364105879589e+00 5.226439322493322237e+01
8.955806188069717066e+00 4.919752022778826728e+01
1.141811787471771211e+01 5.066026661267154196e+01
8.117916140334235919e+00 5.044710735733931983e+01
1.185789959078659983e+01 4.830577811384482345e+01
1.093269664513184303e+01 4.990786292129008217e+01
6.465372867258114553e+00 5.123832826695951326e+01
7.773831031722902907e+00 5.124404047579481869e+01
1.163976887124979953e+01 5.218794598138314456e+01
1.213693750499792046e+01 5.005770523331975852e+01
6.720655628429865303e+00 5.153213631658388749e+01
1.223809360750905562e+01 5.050840948022539578e+01
7.514993749862394878e+00 5.038594048537161996e+01
9.056813750797738649e+00 5.116406661336811368e+01
1.131673132353857447e+01 4.941278066837265470e+01
1.167823908047827075e+01 4.775145516637878984e+01
8.361346603755336560e+00 5.140127641631245581e+01
1.400260447944533126e+01 5.150287537881388999e+01
9.720372971606490609e+00 4.755212327110853465e+01
8.520468290164874503e+00 5.203171495959462334e+01
1.326673216138859601e+01 5.357972205069261662e+01
9.732423094695604959e+00 4.866771248238528358e+01
8.486035960910299991e+00 5.339617173255614091e+01
8.501332146879649443e+00 4.981810643913447478e+01
9.107740765326544974e+00 4.876602295553418998e+01
8.040798248952178540e+00 4.857668617251026433e+01
1.090670180641418696e+01 5.174311613464334414e+01
7.338619477520020418e+00 4.930527830942023115e+01
1.371087191359254120e+01 5.104997779201151076e+01
1.216921333119869253e+01 4.899650739615402983e+01
1.248518722780635670e+01 5.138726761501575879e+01
9.676783887148365793e+00 4.926929918970448341e+01
1.077373862606087052e+01 5.243441758014319731e+01
1.288167523585523710e+01 4.915095495228619171e+01
1.025516676486358492e+01 5.290035704605929823e+01
7.304430364457880742e+00 5.248499590818678939e+01
9.076929744007575351e+00 5.415395222707716272e+01
1.235730324802803537e+01 5.319266080154111620e+01
9.981268434158033998e+00 5.374769930950231611e+01
8.409795032853047303e+00 4.951475833011908634e+01
8.315813978806001927e+00 4.901687179284594720e+01
9.808029574008626739e+00 5.046153968536144419e+01
1.463080696441245365e+01 5.212979460474023341e+01
9.337513450371742962e+00 5.065310012917793614e+01
1.327283405620731216e+01 4.836205248960447989e+01
8.918494555259691481e+00 4.981474222373908134e+01
7.161743533780980719e+00 5.337529795104225627e+01
8.772666691318930177e+00 4.900856374197528709e+01
1.264834730938780893e+01 5.064166848879467864e+01
8.971825513493955739e+00 4.824637210523015085e+01
1.284159769866521472e+01 5.191615183196016403e+01
6.572410258430792851e+00 4.996341866155027134e+01
1.133629680607128698e+01 5.334423019052043458e+01
7.872282169994194057e+00 5.312583327203912376e+01
7.881201189818787256e+00 4.991098795587060266e+01
7.955786617378620917e+00 4.760138143824202217e+01
8.400310657526151914e+00 4.998916410665898269e+01
1.492036526434886845e+01 5.119205334379278582e+01
8.926729019854864333e+00 5.475345727699072995e+01
1.291077962660027367e+01 4.775912537227633692e+01
1.008805603647236993e+01 5.259729398438106074e+01
1.384296949087802275e+01 5.250546712232445401e+01
1.447661984819831105e+01 5.155706841489140402e+01
1.276226631301733860e+01 4.853851751610717713e+01
8.429091945447474643e+00 5.086681886775151895e+01
1.124778647269519460e+01 4.810874872801664281e+01
9.748644575722666517e+00 5.368829141055199017e+01
1.086462617495726413e+01 4.804222665354045319e+01
1.144033646746672517e+01 4.818083011269602878e+01
6.888297687570161365e+00 5.192948460963586399e+01
9.641281739488631430e+00 5.283626694186250461e+01
1.138215302594197631e+01 5.227027098664952831e+01
8.334016337507584637e+00 5.164012182011948227e+01
9.919440468442470404e+00 5.157982975654559965e+01
1.064186620428912633e+01 4.849903883150845019e+01
1.179984038640628441e+01 4.845623751435691418e+01
8.101987710375643914e+00 5.285090681266596846e+01
8.487106016338445968e+00 4.836174295139434065e+01
1.139860510918254377e+01 5.360126292659682434e+01
1.001496969554164096e+01 4.835822362705315669e+01
7.693292543707435982e+00 5.231826706644724112e+01
8.907465095497002494e+00 5.195788476430170277e+01
9.530254390774251405e+00 4.768296473586264028e+01
9.612486675624285937e+00 5.441393500294491758e+01
1.079651744768668920e+01 5.392169263913316257e+01
6.791024831771059311e+00 5.066224262234342035e+01
9.850324013436949855e+00 5.241809985938557759e+01
1.391520716885835540e+01 5.186745065566715596e+01
8.151555195220245409e+00 4.884651504013662304e+01
9.804509626611581652e+00 4.765420962511591085e+01
1.052640306446749108e+01 5.227099786818726557e+01
9.496522581957597353e+00 5.236007912475216841e+01
7.120049062483325919e+00 5.152218101917668491e+01
9.080922319805612375e+00 4.939133357002710767e+01
8.455865514800457561e+00 5.054566593196025792e+01
9.921029931354159004e+00 5.348905691297890286e+01
1.306237430945497735e+01 5.109893272155304089e+01
9.587446047117973436e+00 4.876764893460486405e+01
1.290708677011721761e+01 5.253313167319555532e+01
1.032184948134268332e+01 5.098276653918625101e+01
7.710218252753790580e+00 4.944622155586075962e+01
6.988933100206573101e+00 5.148672286606645088e+01
1.290594201598117508e+01 5.289228993924801614e+01
1.183167901982554149e+01 4.949163382797779320e+01
1.247179568315144849e+01 4.862188953342227649e+01
7.915068383089408854e+00 5.104118265829166035e+01
1.283962582198633839e+01 4.877670455303881880e+01
8.136986537728960300e+00 4.925160808723266825e+01
1.215775360863348098e+01 4.775875742874307406e+01
7.584896298139208959e+00 4.778471198817256749e+01
7.372235311182277151e+00 5.140324525207201134e+01
1.077576460646108991e+01 4.876851252042878571e+01
6.848316450886710705e+00 4.986188505603797694e+01
9.179133049997981075e+00 4.980005279115711403e+01
1.119835756663750104e+01 5.141868440429463050e+01
1.471649081089050703e+01 5.100485585755851758e+01
1.035887471755380496e+01 5.147588956367908253e+01
9.159492097810582933e+00 5.261643834816158005e+01
7.310803713612317623e+00 4.974745861422424298e+01
6.614482950314532594e+00 4.971818445523656749e+01
1.333896356526536664e+01 5.128877646829619863e+01
1.275442526806311605e+01 5.370956124942938459e+01
7.763206330106376285e+00 5.141190139895526556e+01
8.811740224552574219e+00 5.008361523417003269e+01
1.333009436070607023e+01 5.404648691341121491e+01
8.216274447473017162e+00 5.313644544358805177e+01
8.122940753550665249e+00 5.089264914584201449e+01
1.072639917776066198e+01 4.971123699161888965e+01
1.061393816680531010e+01 4.784032082477318681e+01
8.606102479155236296e+00 5.237205119716256974e+01
9.182633660711776002e+00 4.795895147643055623e+01
9.017634804567700613e+00 4.885003446548958550e+01
6.143261727756735624e+00 5.082698805196328351e+01
6.610211597687961849e+00 5.103714216026806838e+01
1.332202645368421301e+01 5.240563922748012970e+01
1.195897282288057184e+01 5.117641123610535914e+01
1.039683200196460255e+01 5.345548520191793784e+01
7.648217311902082649e+00 5.158061220889442211e+01
1.308063846426547983e+01 5.236904223812961590e+01
8.424348952751641306e+00 4.925565908186770514e+01
1.004436144025214794e+01 5.356192533471689643e+01
7.987300081769438620e+00 5.195438094835539999e+01
1.061915898034123096e+01 5.001080164163172270e+01
6.708744368128901847e+00 5.121577136740390301e+01
9.427962296066564107e+00 4.864201616982941800e+01
1.102164770806694882e+01 4.961103327405746199e+01
9.026733708096278619e+00 5.084633521736799366e+01
7.422023225395794732e+00 5.351535036125262934e+01
9.267902889901122876e+00 4.847882410944146159e+01
9.293929522354117623e+00 5.327667440842234470e+01
1.258327260203021325e+01 4.792105501303760207e+01
9.180480554131111859e+00 5.150441192429859427e+01
1.042376380746211773e+01 5.325638594409001314e+01
9.965502012475262461e+00 5.406997479007916496e+01
1.219364453813462923e+01 5.186253453463292828e+01
8.856161327962871255e+00 4.776823598679258964e+01
1.212053592015932502e+01 4.974607037244035013e+01
1.332169649121068566e+01 5.252812589178624592e+01
7.608459755842025274e+00 5.281472502510679590e+01
8.686681161514840710e+00 4.938346245099418752e+01
8.673850165032336079e+00 5.310834357624970181e+01
7.163704275069022742e+00 5.084634358816806099e+01
9.916165109407094391e+00 5.215267416433968606e+01
7.226150180958011759e+00 5.054176708990067368e+01
8.184355628573095842e+00 4.951325731796362817e+01
6.578633110505286652e+00 5.158185607810767692e+01
1.069084671950175469e+01 5.094895101675672322e+01
9.201114420421335538e+00 4.884524592263655052e+01
8.014445213789068490e+00 4.838778987641023122e+01
1.187677242647890630e+01 4.786165878872590440e+01
1.098312000938181932e+01 4.905687371127696395e+01
1.257589743269259053e+01 4.822681536094439281e+01
8.039352298720332968e+00 5.144124621954814813e+01
1.107414760313183955e+01 4.749446139505946718e+01
1.050588911126884817e+01 4.889800077655071675e+01
1.201119393754924047e+01 5.131507033479339697e+01
7.080576310755214209e+00 5.169505392669288568e+01
7.509678353435766063e+00 5.209982689120720067e+01
1.068354629503556907e+01 4.809486847628292594e+01
1.150662007478669757e+01 5.166437206665316495e+01
1.200112304244385619e+01 5.148596865213215779e+01
7.452389928063618996e+00 5.162532817197052282e+01
9.956460552184847046e+00 5.084762440921433324e+01
6.812751470995174863e+00 5.123038661682924300e+01
1.114345241270135745e+01 5.289699549258905620e+01
1.116229044657865899e+01 5.185950952342610520e+01
8.251993131909014068e+00 5.002762962685929438e+01
1.404546619070528735e+01 5.358908410633108588e+01
1.152981075982451031e+01 5.392949059596531924e+01
7.061442018106491858e+00 5.091811024515147466e+01
1.102579303698438018e+01 5.220175738107422347e+01
1.143249255452864510e+01 4.877440715016004447e+01
1.352793831649505663e+01 5.223477723580300847e+01
1.099246247892584094e+01 5.024124090617807070e+01
9.290814783452949754e+00 5.023853806641141517e+01
8.381320809349507073e+00 5.186763048816489885e+01
1.191664840754322086e+01 5.263795634690293923e+01
8.245590978895680934e+00 4.799913577862770353e+01
8.631272331635882367e+00 4.988078994287133838e+01
9.141697186804492503e+00 5.225420981232021234e+01
7.527939686251866824e+00 5.138818464479786030e+01
6.521898436873571292e+00 5.072064650298755595e+01
8.337483984347935362e+00 5.227593819624797788e+01
9.122795720861887503e+00 4.997315288883598328e+01
9.200007961636631038e+00 4.914217556487750471e+01
1.179488128852945295e+01 5.180534610255745775e+01
1.042824063902859955e+01 5.192833359118073844e+01
9.164538955705451073e+00 5.188234254193953632e+01
9.536882190175099083e+00 5.397489564940822504e+01
6.157991788141022660e+00 5.101833979893502402e+01
9.038085802704635796e+00 5.443838734196904028e+01
1.031622649420446436e+01 4.919446634982237043e+01
1.178946054385431808e+01 5.353066898506873628e+01
1.033852670096090343e+01 4.772035054047615432e+01
8.062918461820103744e+00 5.075808846086773229e+01
9.900617279371417112e+00 5.449124535342818376e+01
1.014706224327610684e+01 4.877527034581842713e+01
1.258605675719433137e+01 4.925947092431989205e+01
7.222251350478348542e+00 5.121547572431403239e+01
7.864992999630261217e+00 5.350008846852266231e+01
1.230196881626924110e+01 4.816915757202262682e+01
6.949092746924568686e+00 5.117240880429355343e+01
1.025216120995522573e+01 5.035078022496373507e+01
1.143855036835110184e+01 4.780551634557118490e+01
1.076318312165970070e+01 5.336620381129529278e+01
9.650329666479038337e+00 5.128404968045878576e+01
1.408983077555339491e+01 5.230966822607448563e+01
1.067568316641581738e+01 5.416210127421981468e+01
8.748612866084386042e+00 5.019973846652862903e+01
8.102829397533188427e+00 5.258707823794414082e+01
1.165766525862544434e+01 5.125589246684374700e+01
1.222419894775574534e+01 5.382359172078929532e+01
8.695478680159174800e+00 5.143822370056944493e+01
7.566573277369133876e+00 5.004918215017792704e+01
9.191137119310125669e+00 5.385984972908855184e+01
1.173152899884077982e+01 5.021967539747461018e+01
1.014993696533585066e+01 5.357178868858930798e+01
1.016707423072762495e+01 4.803933662038544128e+01
6.841543662317448593e+00 5.149774479676403161e+01
1.112755264018766610e+01 4.781660319945988391e+01
9.448954858470646201e+00 5.177682849227396389e+01
1.042960687886152904e+01 5.064767908297623933e+01
8.647880801799244566e+00 4.889006286666230494e+01
6.527885644418090827e+00 5.028927683905759238e+01
8.277705832378437378e+00 5.069982724477149105e+01
7.900430193975267912e+00 4.916631349434107534e+01
1.271381257397580811e+01 5.102239634987580530e+01
8.098970915049777730e+00 4.871676630228832039e+01
1.329013410232014536e+01 4.873249601918617913e+01
7.229754424318353578e+00 5.149354819892093360e+01
1.273974693036877071e+01 5.127275748502893293e+01
1.439304028251471834e+01 5.117406468154685939e+01
7.779964906799786029e+00 5.166321763797832745e+01
1.233370503135396135e+01 5.135669057251981684e+01
8.063557625043086929e+00 4.976504388676378454e+01
8.637376228064882255e+00 5.217957320189145065e+01
1.156182732528478141e+01 4.812808837681066620e+01
8.609674750400207799e+00 5.187268074445638177e+01
6.731424913769516571e+00 5.146057113556963003e+01
9.169763241568961831e+00 5.131887486384203356e+01
1.018612553311359115e+01 5.232657585112508514e+01
1.362636715329589876e+01 5.382931249763236536e+01
7.002462635777585653e+00 4.923528814590557090e+01
7.640781458944623239e+00 4.795478361324005334e+01
7.627732642411088726e+00 5.063816631771384635e+01
9.896849055724477040e+00 4.980592768609515275e+01
7.036290654145608059e+00 5.134749678988683996e+01
6.826491913955492485e+00 4.951468809212344269e+01
7.418825258112904386e+00 5.109259456262829246e+01
9.450118631337941721e+00 4.893064544817294603e+01
6.970921405856850050e+00 5.252369300525386109e+01
7.250059983586870871e+00 5.188422404547051769e+01
1.271812519045893630e+01 5.428264934278982423e+01
9.827776503460494695e+00 4.881121100767514065e+01
1.383975638851755008e+01 5.099247924760927475e+01
1.168073209439919680e+01 4.810754785273019962e+01
1.249661789178567162e+01 5.186004846141915436e+01
7.598534657375845214e+00 4.922014222746526713e+01
1.041938426115181215e+01 4.851344405712506358e+01
8.506308893802204452e+00 4.800184193004966460e+01
1.375934426842913716e+01 5.124273068229440042e+01
7.142127175889330104e+00 4.951780486952404914e+01
8.599907744469074800e+00 4.915049964436611418e+01
6.307642949411262911e+00 5.150958189142970411e+01
1.056763122817191736e+01 5.299825503376626301e+01
7.590932754869492705e+00 5.120373332387001142e+01
1.248066405428563108e+01 5.078027233254466921e+01
9.914112346826360778e+00 5.358960166220851562e+01
8.090467568641846796e+00 4.757412900237237352e+01
8.836868443020900799e+00 4.799623559459934086e+01
1.215415672165106287e+01 5.104563638596545871e+01
8.707751630368541385e+00 5.084252041672920797e+01
1.345547892859340422e+01 5.252730087826920169e+01
9.693605094152392354e+00 4.955721412908395962e+01
8.123991289108783320e+00 5.335257596189781992e+01
1.350616379259757593e+01 5.443679662772299110e+01
7.158024195114188082e+00 4.929631498818708479e+01
1.157435890787824917e+01 4.836680272713552142e+01
8.615420379459235889e+00 5.355005259376689253e+01
8.348084073014335971e+00 4.875741574561489955e+01
8.475215269882584224e+00 5.030782426531276030e+01
1.031619061625452360e+01 5.389219717357815398e+01
8.251731573553461985e+00 5.110565667189064953e+01
1.143762199496054244e+01 4.922024870990627932e+01
9.633828483131555487e+00 5.241976593435170884e+01
9.891678862526600113e+00 4.930390986576082923e+01
1.319723553829511076e+01 5.205194950713229218e+01
1.331412163463486564e+01 5.292766963152102022e+01
9.390103018990812345e+00 4.944174207971171597e+01
1.355242223453241834e+01 5.113804765190364776e+01
9.827353629866514950e+00 5.305449250323346888e+01
1.304451178806283984e+01 5.332778853919229078e+01
1.091886764707065538e+01 4.947613567210560603e+01
6.865215879173090840e+00 5.100699972529638870e+01
1.306483230549017627e+01 5.158396197600433197e+01
6.578220982310642562e+00 5.180122077287551718e+01
8.364680457132083191e+00 4.966419243088100899e+01
9.563475788784607801e+00 4.847735059796905688e+01
8.999952656457256950e+00 4.869957894248812380e+01
1.053030345767308695e+01 5.246871630773990347e+01
1.042619005285778933e+01 4.815656027728840627e+01
1.161987551324797785e+01 5.089960220978381500e+01
9.204782033171017730e+00 4.868133004799358332e+01
1.194566386078372666e+01 5.027918522098872245e+01
1.196674570267290250e+01 4.850265386176992877e+01
8.598539713742361812e+00 5.015307055285969540e+01
7.227372542926256749e+00 5.025226339849367463e+01
1.172611953523911943e+01 4.906270010512349700e+01
9.430418342195475034e+00 4.807936770702975338e+01
6.211020349982423028e+00 4.995862578002849830e+01
1.321613129729622749e+01 5.269752768301673740e+01
1.433228433588193340e+01 5.154687317187155315e+01
9.705738960104060453e+00 4.999901068029551254e+01
1.233076279589763047e+01 5.256191108998179118e+01
9.732188137843882814e+00 4.906511553858776153e+01
1.046478052388513547e+01 5.126060150818099714e+01
8.983156384280274409e+00 5.044453900641820354e+01
6.889361493781263945e+00 5.024208333419784367e+01
1.109798856659452149e+01 5.118191218641639750e+01
7.647659649405788684e+00 4.760708557467005164e+01
9.633256654821927611e+00 5.054192606947945166e+01
9.488451698106770138e+00 4.920983388185659635e+01
9.359777545554672784e+00 4.773320946420692223e+01
7.836745630491584791e+00 4.856702554572488140e+01
9.370408125926843468e+00 4.872426758252640866e+01
7.427645485293719041e+00 5.230484994581121327e+01
1.086707250777527989e+01 4.823747491456551018e+01
9.834371159379237781e+00 4.809446453998618409e+01
1.305656241649484173e+01 4.883186876824748879e+01
1.076244279084780864e+01 5.187548144963426466e+01
1.383168946393620757e+01 5.282035561346287267e+01
1.116448401239197885e+01 4.838787120775571537e+01
9.779852002134354905e+00 5.074567930908914803e+01
1.449338470285075076e+01 5.234724899324098857e+01
8.771319369007835931e+00 5.278740115787952192e+01
1.088112973156392549e+01 4.863349833056934557e+01
8.918338803666646442e+00 5.306247441443818502e+01
9.877419920597514391e+00 5.124371597435147407e+01
1.098583693526769878e+01 5.431411074546048212e+01
1.184789497695493843e+01 5.225451097908772624e+01
1.236521569755944050e+01 4.845825977240962601e+01
8.363662114995044305e+00 5.489050328993951666e+01
7.366523104182963522e+00 5.072488800025649169e+01
6.961375476982913035e+00 5.085457767253151928e+01
6.297144589898334033e+00 5.120272544205747067e+01
8.808785343770461296e+00 4.875245515695521448e+01
1.415672486482597137e+01 5.114458554980553373e+01
8.641010946041795648e+00 4.961075913357354494e+01
1.023623950631970025e+01 4.966517996581912087e+01
1.913433317872065942e+01 5.187840778748473980e+01
1.718597517630057858e+00 4.677365176527484891e+01
9.348121455999990559e+00 5.623445518800031806e+01
8.474315742935548457e+00 5.030555869787177414e+01
1.020782818599951014e+01 6.196962892599837147e+01
1.415840506159279677e+01 4.758712955629404462e+01
5.330228000005421407e+00 5.216902743576494572e+01
1.500022650918715428e+01 6.143341194446262676e+01
8.190205840999762898e+00 4.676623667700009435e+01
1.229338579497209238e+01 5.162787987969522163e+01
1.547499799265897735e+01 4.987102379619934567e+01
7.453487715335716857e+00 5.162737924533935541e+01
1.158315201890049551e+01 4.876070136150460854e+01
4.476940500160698555e+00 5.071240775218492303e+01
9.934270427429053640e+00 5.306617821094151566e+01
9.202534006788708254e+00 4.869324801897746369e+01
6.133266749986781008e+00 4.975290147194729684e+01
1.446618093933234306e+01 5.233875967323290013e+01
1.018843925566726938e+01 5.002735832193981480e+01
6.465608526401783962e+00 5.123831810549229715e+01
1.223161481490165770e+01 5.411638549841108414e+01
8.480574287599711880e+00 5.204253115117764139e+01
1.323188689762647385e+01 5.254636185806212012e+01
1.275507505068524949e+01 4.815567895799178189e+01
9.796605111037253621e+00 4.764393946938476887e+01
8.688115279525256085e+00 5.309517984096041232e+01
9.478679389466311278e+00 5.126084046059813204e+01
7.836021481305582626e+00 4.802820735720557366e+01
1.141813881189590596e+01 5.066129691426663584e+01
1.429770427519018838e+01 5.115819199246609372e+01
7.046219862542149492e+00 5.096055753600600724e+01
8.308330429042948495e+00 4.901624456576053035e+01
1.143474949627203863e+01 4.817244588103562108e+01
9.915435847222006771e+00 5.358986146098980186e+01
1.368172788792864303e+01 5.413642014035468009e+01
1.330979158082227976e+01 5.070496397932397059e+01
7.962289806199466469e+00 5.309863845720796860e+01
6.883993840321335433e+00 4.952706842948713017e+01
1.075194078664847197e+01 5.189848347641521542e+01
7.304289248109639310e+00 5.248514072830179344e+01
1.068874943783559317e+01 4.900314065854416157e+01
1.106532502494149739e+01 4.943987716949921918e+01
7.495010779146305424e+00 5.063454791553319723e+01
9.680112915910962457e+00 5.430876216063964534e+01
9.841539628776352799e+00 5.242318566953957060e+01
8.839771197573014661e+00 4.799677122087537384e+01
8.409151954743167678e+00 4.951367876163574522e+01
1.186822407296947723e+01 5.200277365150617470e+01
6.728916937199088011e+00 5.145654653223166974e+01
1.033118600821982724e+01 4.772918486055986165e+01
1.339690485928286456e+01 4.859216668344399892e+01
8.045014582735969810e+00 5.088689756796571828e+01
1.230404749867227920e+01 4.892413043155943342e+01
1.023439901932310825e+01 4.851512244418717046e+01
7.954136067787931808e+00 5.167764573884188906e+01
1.233224592305620959e+01 5.118871358255184845e+01
1.053629571573751456e+01 5.354794287809059483e+01
1.127772110542602135e+01 4.749587269567865633e+01
7.961860400190393428e+00 4.759908580598549577e+01
8.987476461826405583e+00 5.237244507356380296e+01
9.746323480794558236e+00 5.090549987512498831e+01
6.299917806342854476e+00 5.082242284562252621e+01
1.369506386407700127e+01 5.107469102322315280e+01
1.201497814923983931e+01 4.970825285819417871e+01
1.063228934460630271e+01 5.056326068131237861e+01
8.974669451858623859e+00 5.009159918280500534e+01
9.050261739501758385e+00 5.116749186697371954e+01
1.249640652861394585e+01 5.241171906424948190e+01
8.400216069665425778e+00 4.998453151638378245e+01
6.668457391123786770e+00 5.098086895454130740e+01
1.422993066475077129e+01 5.309044214404347173e+01
1.052794176476802868e+01 5.230145550438870572e+01
1.332285464642793116e+01 5.252693934914400842e+01
9.823542622915764966e+00 4.880660772850323781e+01
9.197151297080578658e+00 4.914395589460587388e+01
1.172829883312606647e+01 4.776124450187620596e+01
8.130539138773603014e+00 5.354187086707996457e+01
1.016237274219696474e+01 5.430098756070852062e+01
1.189369618877232426e+01 5.140699570738983226e+01
1.088407532094100816e+01 4.824488833073613847e+01
7.252804298602821831e+00 5.124710437463620849e+01
7.078207560409978782e+00 5.169703650591509358e+01
1.447756942393008117e+01 5.155790624247443077e+01
1.136666440137229550e+01 5.356893903795921119e+01
9.915792485004832457e+00 5.188211561414001949e+01
9.180700069018447707e+00 4.979755081599185473e+01
7.477215982508025327e+00 5.328929735543114532e+01
7.312538578962399782e+00 4.929260410389454705e+01
1.288473660799483511e+01 5.052430297335976661e+01
1.100391187551190164e+01 5.051707316611780385e+01
9.014823955158501434e+00 5.463608424112936746e+01
7.072574725612380320e+00 5.071540442533954263e+01
7.618828866427354463e+00 5.193043878314325212e+01
7.020963841429177066e+00 4.993810500312180523e+01
1.443284665113163712e+01 5.186670243488104859e+01
1.206600214942609384e+01 5.092242425296529262e+01
7.117600299036801914e+00 5.152221265272525841e+01
1.119712881887129541e+01 5.235882923082185414e+01
9.484572915545209426e+00 5.363218719816809710e+01
7.804486533365230905e+00 5.224997485240571393e+01
8.659013847882821580e+00 4.908421458963346140e+01
8.781384328280461560e+00 4.856710899765202072e+01
7.856296738661052714e+00 4.858331503321443279e+01
1.248624312228502831e+01 5.075189887227643482e+01
6.161144248929312894e+00 5.177068839587464311e+01
1.353237112436871925e+01 5.246146725095391616e+01
1.161582721203613033e+01 4.819683126231789316e+01
9.380880625920015703e+00 4.772220514735163732e+01
1.076968427681864426e+01 5.243889222904466862e+01
1.042411805065298935e+01 5.132117233716034121e+01
8.688963566316193265e+00 5.173561424052903135e+01
8.112620128388746465e+00 5.043114053439469302e+01
9.505527446296088812e+00 5.233160284441806454e+01
9.419756683768364169e+00 5.060600813782751572e+01
6.812823313439334783e+00 5.123082602029632682e+01
9.836690364734209879e+00 4.809410288374157716e+01
1.293961110379579083e+01 5.315503823480742085e+01
8.240995103011435674e+00 5.003802954977899731e+01
1.100803154000773176e+01 5.107915629962876380e+01
7.833602648676034619e+00 5.118820171866325097e+01
1.391610668721945032e+01 5.186630009425056187e+01
9.786423302360553578e+00 4.952522557265810832e+01
6.839347682304985909e+00 5.086197679339791478e+01
1.288035921583404075e+01 4.915439307994289209e+01
1.218121341467021423e+01 4.822805021238811207e+01
1.097148212394244560e+01 5.431161780781406634e+01
1.168941801414784543e+01 5.024614186440621211e+01
1.194142598972721636e+01 5.259608899811710359e+01
6.303134471069932410e+00 4.996694920871363621e+01
1.092075489329470628e+01 4.990646059842372750e+01
9.147458133471445763e+00 5.292188490920625554e+01
8.682839778755962357e+00 5.054705295009413391e+01
1.329716373971533727e+01 5.133516204296952168e+01
7.745957737858795156e+00 4.966360709161551057e+01
1.089863826848815442e+01 4.780106444128493592e+01
8.589876400333237072e+00 5.235736334408711912e+01
1.164131377000001955e+01 5.191186540061756460e+01
1.451522576607218085e+01 5.141326121885591505e+01
1.016871934512098008e+01 5.356999899597295922e+01
1.279868196898565813e+01 4.874956396816972415e+01
9.115638864361258697e+00 4.872897870367320650e+01
1.288984193457133820e+01 5.083724571249381796e+01
1.218024662711977335e+01 4.857590739393685197e+01
1.023787260538564325e+01 4.755378661214420077e+01
9.403389674755649352e+00 5.174120017085372325e+01
1.332682274681579315e+01 5.403968400461919686e+01
7.061439934757036241e+00 4.933389218849974611e+01
1.119595285083012826e+01 4.851476331051375013e+01
1.227881959297513426e+01 4.957936820268111688e+01
1.179896924744510223e+01 4.843857874448617906e+01
7.373655496440722601e+00 5.140714069275368558e+01
8.104829321334579006e+00 4.830056750525156417e+01
9.997128759353890004e+00 5.083581494559229697e+01
6.659794654822836613e+00 4.981253102642143205e+01
7.578972370946824633e+00 5.035784281944417273e+01
1.213164608761656105e+01 4.901393615745200094e+01
8.507439947446862050e+00 4.945157248989257681e+01
8.033823202058593793e+00 5.261034301602377639e+01
8.235214830460344970e+00 4.763012650734893327e+01
8.730798993829825605e+00 5.381526955366179266e+01
8.666930625301228375e+00 4.810909645154921321e+01
1.326488123282905640e+01 5.358106120954392537e+01
1.214424939617435761e+01 4.782377363934945436e+01
8.487912846870273142e+00 5.092482265881955783e+01
8.608930158878251859e+00 4.986416538415496547e+01
1.209732962091607611e+01 5.007966758956922604e+01
9.568226065484456200e+00 4.847609782187199556e+01
6.694137429654438698e+00 5.066195310488348724e+01
8.953219146621808733e+00 5.192465106570243449e+01
9.899573225500617468e+00 4.982146606142732992e+01
9.398330276285124185e+00 4.864765076919790943e+01
9.727313275794683634e+00 4.919068873854849500e+01
9.414948241604429668e+00 5.477193340851397352e+01
7.715217905723045888e+00 4.943665962298140926e+01
7.529486713183073832e+00 5.138648025510801176e+01
1.089346090108242393e+01 4.838022092825302423e+01
9.064649850276536469e+00 4.848957317834621961e+01
1.244484307249000210e+01 5.136369320687023077e+01
1.143478166374767646e+01 5.095536302193222866e+01
1.132394916810995511e+01 4.944279559721661599e+01
8.093035313487382254e+00 4.870230961238362255e+01
8.196684797175567638e+00 5.190351659605151724e+01
1.008530911869077862e+01 4.826010652347309104e+01
1.122255298557750436e+01 5.178904726443870032e+01
1.466199101389815418e+01 5.104016548397641628e+01
1.165612271580996584e+01 5.219474643471088626e+01
7.748330311781520763e+00 4.760364034580967285e+01
1.055652960361255488e+01 5.297065950383044708e+01
1.040487215441321744e+01 5.215032469315514874e+01
6.395974454201996728e+00 5.172514891079335086e+01
7.159420776251100627e+00 5.338508538021377348e+01
7.610521967815969013e+00 5.092614356071094761e+01
1.081323600310130217e+01 5.391699170905489780e+01
6.768498275958020116e+00 4.930064819476316984e+01
8.231287820496213214e+00 5.111337308742454155e+01
1.011419061788758000e+01 5.266935105067192779e+01
1.318872896960315622e+01 5.202311879158171593e+01
7.072416149543444774e+00 5.215364967146938824e+01
9.207457580692489429e+00 4.895689538111500383e+01
8.670788624296534408e+00 4.940618954956673292e+01
1.064996258908650439e+01 4.930236924121143716e+01
1.380393472265818566e+01 5.283518472326393578e+01
1.134136874997554756e+01 5.011823739782161624e+01
1.395283190430101428e+01 5.244300629435081618e+01
8.701338604948297473e+00 5.145797644585905317e+01
9.096286920479160543e+00 4.824176355580043918e+01
1.114801659669224954e+01 5.275358121897918551e+01
1.291035730797919712e+01 4.775868383103443904e+01
8.296501379993175718e+00 5.072764495157557008e+01
8.537652215116439791e+00 5.009959729905278891e+01
9.917852649649718799e+00 5.157764758480231393e+01
1.400363034247955341e+01 5.150398399980918640e+01
1.061461651342385082e+01 4.784142678056822717e+01
9.638070333484247243e+00 5.241064533304885487e+01
1.307716627366113471e+01 5.424678748305721854e+01
8.238417925080037563e+00 5.278772065297543747e+01
1.177360578643422784e+01 5.300369824736517188e+01
7.066886358270962454e+00 5.135396132319675644e+01
1.011416594506173183e+01 4.886573421634657421e+01
1.057007600439787076e+01 5.382163843208932974e+01
9.573583417803606110e+00 4.983607983137833486e+01
7.224774768697638017e+00 5.149378923409962283e+01
7.578611717548747961e+00 5.212150765422706655e+01
8.194394395559422506e+00 4.937268323261653080e+01
8.603626469198239590e+00 5.196471643065974888e+01
9.737017199585219274e+00 4.829364686437951804e+01
1.377311049024417855e+01 5.340681128589270088e+01
6.885301980698972812e+00 5.024155622754625483e+01
9.973543524127903126e+00 4.841180878212665561e+01
7.817235695802668971e+00 4.832927337587472749e+01
9.037629124227754218e+00 5.421688316859068379e+01
1.032217317222063002e+01 5.098248535615872612e+01
8.307053008363023849e+00 5.142262556011622365e+01
1.224010044240246309e+01 5.050864137728933656e+01
6.659045327161297934e+00 5.137261205405755504e+01
1.104301508089417538e+01 4.892090071415843511e+01
1.201047496308358298e+01 5.131534689989537412e+01
8.212195483390395623e+00 5.315056076897543846e+01
8.388316313947129288e+00 4.816999856833770366e+01
1.022064962034223612e+01 4.961081150463090950e+01
1.486266362828109777e+01 5.128033032938125046e+01
8.838623797314356878e+00 5.309012596872986478e+01
1.180755423134018045e+01 5.057998662685231750e+01
1.277147752345131870e+01 5.198416633244770679e+01
8.982660669058960323e+00 5.083063754435750070e+01
9.974118641377575756e+00 5.408833562799690498e+01
9.973267048396154522e+00 5.347875558945852958e+01
7.647938906336772646e+00 5.158102997926235389e+01
6.791618730989618768e+00 5.113911612506169035e+01
7.211182458951835628e+00 5.059061958215036725e+01
8.007600060175612100e+00 5.119977693521311579e+01
8.738418113126249054e+00 5.017560338568104328e+01
1.244598955952828945e+01 4.858044459850726327e+01
1.075481856791375357e+01 5.147856609660565397e+01
1.147242342362003598e+01 4.779442454848351218e+01
1.226311820118323226e+01 5.380456696356120005e+01
6.203371434413224250e+00 5.103018583832656674e+01
9.414104945560962889e+00 5.209400174939995765e+01
9.680236536007038595e+00 4.789187288342782267e+01
1.182129040026264377e+01 4.948693453289266841e+01
1.241883758032466467e+01 4.819766311613057752e+01
1.144158651782575831e+01 5.390202948506882308e+01
7.902058823088064621e+00 5.100845169940429003e+01
8.993230788381003649e+00 4.928583062010363847e+01
7.228117356086235290e+00 5.025614755860792826e+01
1.120522661039460210e+01 5.137758106134313607e+01
6.974557983141426831e+00 5.103517348735702797e+01
6.989190918718989565e+00 4.923670284285074672e+01
8.324573681369656697e+00 5.225344236666798281e+01
9.989174215068336693e+00 5.216237198813595199e+01
1.327170886883691736e+01 4.836621257839642851e+01
6.302514731728987130e+00 5.149990173640200908e+01
8.042523113231819565e+00 5.226064677507115874e+01
7.897294990837236739e+00 4.986093312754941564e+01
8.538334111899235523e+00 4.840629223239422174e+01
1.275579452298021721e+01 5.370854497421809270e+01
7.241261212156780758e+00 4.958490861567698005e+01
1.058903761706750046e+01 4.968193783980733258e+01
9.700789462601893121e+00 5.052389116267752911e+01
9.264341918441530765e+00 5.023998529025828219e+01
1.024872679029517997e+01 5.035657772830604273e+01
9.587597833316054974e+00 4.878752352192553587e+01
1.317048590324885105e+01 5.280034378301553488e+01
8.559060473199751939e+00 5.353014611340933726e+01
6.609363968435763681e+00 5.103717583945415015e+01
1.272343335210608650e+01 5.112373001196023381e+01
1.200259128244883833e+01 5.148599436465219270e+01
8.845060663951469948e+00 5.221178887106763256e+01
7.368291582167181275e+00 5.306579882926313019e+01
1.082568120497716535e+01 4.867225033573779314e+01
6.136313927203543628e+00 5.082365589578618170e+01
9.900743399622772500e+00 5.348506370678600774e+01
1.018704525714310094e+01 5.174529980990438105e+01
9.367121544570165881e+00 5.324314561581558536e+01
1.165340078922347722e+01 4.806439794384321829e+01
8.855967665720116955e+00 4.776884381382210165e+01
9.158941622875369504e+00 5.261685014514475256e+01
1.325948996080962594e+01 4.893988083800802968e+01
7.618245746523347961e+00 5.008236139688138877e+01
1.258375787715900529e+01 4.791589088070739422e+01
7.468747968653106639e+00 5.229693695551071642e+01
1.209932235864700090e+01 4.927505830683079324e+01
1.334354058638556850e+01 5.094189214390334541e+01
1.352885661201181300e+01 5.212971058592067664e+01
9.640288260984867463e+00 5.285665786827502899e+01
6.529291270763546429e+00 5.028753526664035434e+01
8.347353406512983298e+00 4.875960466424381679e+01
1.045467134989109503e+01 4.818137552361322662e+01
8.632238749095289876e+00 4.961171697999831309e+01
9.372619219764921183e+00 4.956296157886644238e+01
9.404717773369688061e+00 4.804745015751812787e+01
7.314398835991084979e+00 5.161627300724529732e+01
1.225150997948027687e+01 5.308452760105738832e+01
6.719881152343594621e+00 5.153562357730194776e+01
7.397724859358807414e+00 5.356159990224730194e+01
1.143722079182967555e+01 4.876162405210151007e+01
1.098266863913555724e+01 5.216872734581070858e+01
8.120944085067575813e+00 4.962898902623585684e+01
6.978531683361276272e+00 5.251793454866852073e+01
1.151382376349797454e+01 5.165656748429348255e+01
1.100944939880582751e+01 4.959237966922226803e+01
9.204752340291783241e+00 4.884264332044740797e+01
1.386469080758440597e+01 5.100671486558332646e+01
7.859282768958125054e+00 5.343543051422242485e+01
9.172129630770362141e+00 5.388137236420573828e+01
8.962407951633736047e+00 5.045482393395143816e+01
1.246787917054469119e+01 5.185541304643837890e+01
1.068688480960059906e+01 5.095032977017687870e+01
1.359582578711516732e+01 4.869279280176122882e+01
8.035010276055945866e+00 5.147015865142358138e+01
8.038434906801883884e+00 4.918399199967537783e+01
6.990127250688882299e+00 5.148450710969429878e+01
7.826973574144457046e+00 5.049370815625680819e+01
1.366289999805531430e+01 5.145747870450563966e+01
1.146323445384080308e+01 4.917612583795644809e+01
1.172286296292452690e+01 5.121902958751267931e+01
1.061513997560755840e+01 5.001643429801183061e+01
7.194406114964522203e+00 5.083944778718439039e+01
7.605456302429501214e+00 4.785100600223266554e+01
9.254179580028239371e+00 5.077401057489277036e+01
7.987664826422553155e+00 5.191187341095793073e+01
8.857942149282964195e+00 4.828435886245273423e+01
1.345210686052146798e+01 5.252430965612381186e+01
1.042161349218061339e+01 5.325643558714221371e+01
1.409736347989176330e+01 5.364435199397750864e+01
1.102359712979657758e+01 5.094344740248022418e+01
1.035720318782327709e+01 5.070782111582738594e+01
9.041038674835478517e+00 4.886301489861693881e+01
1.042586498372633486e+01 5.192493374703605014e+01
7.574182633591101776e+00 5.276164932944488584e+01
9.165294193643561016e+00 5.187866356133324075e+01
8.515970037622031086e+00 4.797905715977692864e+01
1.290797674514765170e+01 5.253320683082610998e+01
9.740867767004472100e+00 5.368132347940647264e+01
1.100504054632759221e+01 5.024950988400696872e+01
9.539077442572251542e+00 5.397551316908408836e+01
1.086476230440835877e+01 4.805835326491911985e+01
1.017203588353213917e+01 4.799861678506952956e+01
6.609929458616224274e+00 5.176643977245504402e+01
1.266362926093464125e+01 5.146272190619799858e+01
8.364167187820338967e+00 4.966576405767653313e+01
1.284093023101160469e+01 4.842399156500220414e+01
7.259989547654960518e+00 5.272371420568484979e+01
7.142537315269859022e+00 5.123199835595157481e+01
1.025459902962788306e+01 5.390068431198450583e+01
1.039329566089593904e+01 4.884571563106235459e+01
9.131403928255052094e+00 4.996902529485022626e+01
1.112921516371756070e+01 4.924550480605653036e+01
9.140043311900537759e+00 5.137519394971508291e+01
8.429266976253600774e+00 4.925243092512835119e+01
7.610416517241228362e+00 5.121863821078459722e+01
9.142671835643906064e+00 5.225329323858193931e+01
1.161204224200459478e+01 4.994206271042806122e+01
6.843482092409269413e+00 5.149698771874912495e+01
1.194297188558335243e+01 5.029266593249904815e+01
7.763266744598366742e+00 5.141239630186339582e+01
1.254034759153561396e+01 5.325991329021617560e+01
1.225068057827938262e+01 4.987974909084872621e+01
1.490374138880906330e+01 5.104132906326000096e+01
1.044960885594180766e+01 4.853182707081106884e+01
8.419034789449183265e+00 5.488374008297749640e+01
6.459965199746884323e+00 5.074798905853189268e+01
8.772300841614578459e+00 5.278874190264149036e+01
1.098195246546675286e+01 5.179465845886141295e+01
7.598656814713958596e+00 4.922064875110685733e+01
1.075160123879115126e+01 4.765184332664913569e+01
1.170605433604524848e+01 4.814215957906796461e+01
8.823324682315250911e+00 4.991497147846483529e+01
1.274307986395054293e+01 5.427584850348604562e+01
1.220759093758752556e+01 4.803537556867993175e+01
9.879390029467963075e+00 5.122934868680660259e+01
9.640806260498131053e+00 4.866990666405661159e+01
9.436145336539709660e+00 4.929265994136957119e+01
8.332742914602194517e+00 5.164102188216139666e+01
8.185071220199223063e+00 4.884888594402534068e+01
6.708791009032324837e+00 5.121494162254852256e+01
8.452907119982187467e+00 5.054300176961380942e+01
1.155895104048706301e+01 4.812146546229705280e+01
1.031315291967612069e+01 5.155711152888844850e+01
1.030768824297290465e+01 5.343363436106352538e+01
7.039430907820791283e+00 5.160083889960404946e+01
6.958347964548458719e+00 5.085554870603439781e+01
6.948618722378242296e+00 5.117365030309243679e+01
8.846523936958657686e+00 5.120388814784108433e+01
1.005635230519724033e+01 5.355742567275720489e+01
9.000546012645807892e+00 4.869826021422348106e+01
6.863577361018022849e+00 5.100396740231933990e+01
6.554341062013992136e+00 4.952687332591692382e+01
1.261878595860821228e+01 4.823081823483494901e+01
9.971738175503430668e+00 5.376631814100436202e+01
1.178819001694461832e+01 5.176410492416535192e+01
8.031009792415479964e+00 4.996726997026839001e+01
7.779970436876799234e+00 5.166355505464379405e+01
7.252461391491745957e+00 5.188339350861332377e+01
9.045879254436743011e+00 4.789768776367255043e+01
1.017802073615496461e+01 4.866712535544193940e+01
6.291882955502204666e+00 5.123444652255922449e+01
1.463288261529926793e+01 5.212564276946621789e+01
1.272914926536695113e+01 5.067838163626446146e+01
1.122072071561383666e+01 4.816004331827770102e+01
1.179061048333087136e+01 5.352799432019840253e+01
9.917697002681370222e+00 5.447677391738065467e+01
8.698841375466525250e+00 4.889554034727991194e+01
1.033634364776811054e+01 4.911156097064247916e+01
6.886937928719055968e+00 5.192873069293056432e+01
1.107726671513889549e+01 4.749837793930625196e+01
1.333131930218396377e+01 5.239421662326537898e+01
1.209116385859648801e+01 5.409804763616328671e+01
8.553521022115139871e+00 5.127264520193268282e+01
8.181027453163508767e+00 5.207648356481736585e+01
1.219514742578723521e+01 4.761223336036171361e+01
9.465487691621792266e+00 5.248216633350813254e+01
1.301894969525101686e+01 4.878070979859860046e+01
7.404929546445254473e+00 5.109053313076339720e+01
7.368410555351102786e+00 4.988581642572896868e+01
8.007016404086066075e+00 4.846667494731265435e+01
1.016737026311146508e+01 5.237437523382750726e+01
8.232609238618360692e+00 4.790474440645945720e+01
1.003251050756378859e+01 4.772804357544953291e+01
1.305196691489837590e+01 5.162384031093756676e+01
8.606982985837351663e+00 5.216616691476227174e+01
1.258566913244986729e+01 4.926112125543208009e+01
1.348681814719192218e+01 5.443343418417786239e+01
8.518557710012361284e+00 5.324640220884462849e+01
8.732774850277580825e+00 5.077707780903733692e+01
8.831838236688273724e+00 4.877356441451065905e+01
9.644659121343138963e+00 5.127709519943228145e+01
8.380455760475896909e+00 5.186846501830000733e+01
1.363558687375327239e+01 5.260433721674236551e+01
8.426578773064401062e+00 4.899773817546865473e+01
1.231146814370481302e+01 5.136308712893275441e+01
1.063099485450621273e+01 5.413398146674122557e+01
9.591190411203733390e+00 5.456734926311079192e+01
8.602451693256140786e+00 5.015212327435146022e+01
7.450815208874769446e+00 5.042190072321685790e+01
6.580919722485933931e+00 5.151402694066799626e+01
8.044328238942897613e+00 5.074537154578583653e+01
1.219220075785587198e+01 5.186365908930353186e+01
1.113067925354679844e+01 4.781846340285207475e+01
1.161911206564666443e+01 5.089953448101647382e+01
8.394941853884658300e+00 5.257965262297432929e+01
9.353281725840290051e+00 4.873545826183009666e+01
7.695843538275877194e+00 5.304411826979646349e+01
9.449244397203827361e+00 4.893081137202845809e+01
1.354935312991887741e+01 5.316787271843947593e+01
1.308093902518672991e+01 5.088474337040427287e+01
1.189752494456662291e+01 4.789934967844784808e+01
1.158084452243509688e+01 4.840063866910873713e+01
1.261154683584382497e+01 4.889287458644960793e+01
9.156933277495625134e+00 4.768978859158878691e+01
1.076301214441721577e+01 5.335303241000994490e+01
1.139345523033332519e+01 5.216506946594461169e+01
7.638589646472466299e+00 5.250167291039642237e+01
1.068664701194368583e+01 4.808566210902872484e+01
1.432946354661923749e+01 5.154688845589150503e+01
9.813421050152784986e+00 5.226630064111557061e+01
1.120531997724768125e+01 4.873875291201335358e+01
7.363445478523402699e+00 5.208196456387797468e+01
1.362121334344474377e+01 5.381961063164602166e+01
9.268589621323018335e+00 4.850842551116840440e+01
8.808104053134620415e+00 5.203546359556319345e+01
8.973876042970019284e+00 4.972235601480300460e+01
9.772122250217071482e+00 5.199270055141931834e+01
9.460787043188325640e+00 5.138641882423831930e+01
9.576368314313635111e+00 4.769108314198358300e+01
1.124436044650853184e+01 5.035092247107213836e+01
1.113331544418363883e+01 5.306133225392358810e+01
1.083512015282702023e+01 4.960702419325242118e+01
8.842791231722078038e+00 5.012017739252342352e+01
9.350176070152226515e+00 5.108937984345142524e+01
1.195016331299222934e+01 4.844411775837103562e+01
1.176858014928371077e+01 4.885480078569189999e+01
1.304444141252491463e+01 5.109672643791301994e+01
8.704544467141355213e+00 4.926850076951802038e+01
1.307941998400224293e+01 5.237381818327998673e+01
9.730025739826379194e+00 5.239243283493488690e+01
8.959903160889515661e+00 5.349483250804028955e+01
1.199791331534705208e+01 5.118770138724021734e+01
1.004732068723230398e+01 4.909880781181485787e+01
7.476603153518496470e+00 5.151345738813154895e+01
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment