diff --git a/Documentation/Tex/MiMeS.tex b/Documentation/Tex/MiMeS.tex
index f0f83fd..b9b2bcb 100644
--- a/Documentation/Tex/MiMeS.tex
+++ b/Documentation/Tex/MiMeS.tex
@@ -1,1859 +1,1867 @@
 \documentclass[11pt,a4paper]{article}
 \usepackage[utf8]{inputenc}
 \usepackage{amsmath}
 \usepackage{amsfonts}
 \usepackage{amssymb}
 \usepackage[left=2cm,right=2cm,top=2cm,bottom=2cm]{geometry}
 
 
 \usepackage{slashed}
 \usepackage{hyperref}
 \usepackage{graphicx}
 \usepackage{caption}
 \usepackage{float}
 \usepackage{subcaption}
 \usepackage{multirow}
 \usepackage[dvipsnames]{xcolor}
 \usepackage{multicol}
 \usepackage{authblk}
 
 \renewcommand{\theequation}{\arabic{section}.\arabic{equation}}
 
 
 
 \include{macros}
 
 
 
 %%%%%%%-----------------------Rules-----------------------%%%%%%%
 % 1. Put any new macros in macros.tex.
 
 % 2. Section should start with:
 	%\section{This is a section}\label{sec:Intro}
 	%\setcounter{equation}{0}
 
 % 3. labels for Figs should start with fig:, for equations should start with eq:, for sections with sec:, etc.
 %%%%%%%--------------------------------------------------%%%%%%%
 
 
 
 
 \author[ ]{Karamitros Dimitrios}
 \affil[ ]{\em School of Physics and Astronomy, The University of Manchester,
 	Manchester M13 9PL, United Kingdom}
 \affil[ ]{\textit{E-mail: } \href{mailto:dimitrios.karamitros@manchester.ac.uk}{\color{blue}{dimitrios.karamitros@manchester.ac.uk}}}
 	
 	
 \title{{\tt MiMeS}: Misalignment Mechanism Solver}
 \begin{document}
 
 \maketitle
 
 \begin{abstract}
 	We introduce a \CPP header-only library that solves the Axion equation of motion, \mimes.  
 	\mimes makes no assumptions regarding the cosmology and the thermal mass of the axion, which allows the user 
 	to consider various cosmological scenarios and axion-like models.
 	\mimes also includes a convenient \PY interface, which allows the user to use it without writing any code \CPP.
 \end{abstract}
 
 
 \section{Introduction}\label{sec:intro}
 \setcounter{equation}{0}
 
 About the axion...
 
 \section{Physics background}\label{sec:abundance}
 \setcounter{equation}{0}
 %
 Although there are several works in the literature (such as~\cite{Chang:1998ys}) that can provide a insight on the solution of the axion equation of motion (EOM), in this section we define, derive, and discuss the various quantities we need, in order to understand how \mimes works in detail.
 
 \paragraph{The EOM} %The axion field is written as 
 %
 %\begin{equation}
 %	A  \equiv \fa \ \theta,
 %	\label{eq:fa_def}
 %\end{equation}
 %%
 %with $\fa$ the scale of the axion that determines the the scale at which the PQ symmetry breaks. 
 %
 The EOM in terms of the axion angle, $\theta$, is 
 %
 \begin{equation}
 	\lrb{\dfrac{d^2}{d t^2} + 3 H(t) \ \dfrac{d}{d t} } \theta(t) + \maT^2(t) \ \sin \theta(t) = 0 \; ,
 	\label{eq:eom}
 \end{equation}
 %
 with $H(t)$ the Hubble parameter (determined by the cosmology), and $\maT(t)$  the time (temperature) dependent mass of the axion, with 
 \begin{equation}
 	\maT^2(T) = \dfrac{\chi(T)}{\fa^2}\;,
 	\label{eq:axion_mass_def}
 \end{equation} 
 %
 and $\chi$ a function of the temperature, and $\fa$ the scale at which the Peccei Quinn symmetry breaks \DK{ref??}. For the QCD axion, this has been calculated using lattice simulations in~\cite{Borsanyi:2016ksw}.~\footnote{The data provided by ref.~\cite{Borsanyi:2016ksw} are used by  \mimes out-of-the-box. However, the user is free to change them.}
 
 
 \paragraph{Initial conditions}
 %
 \begin{figure}[h!]
 	\includegraphics[width=1\textwidth]{figs/axion_mass.pdf}
 	\caption{The mass of the axion as a function of the temperature for $\fa=10^{12}~\GeV$, using the data provided in ref.~\cite{Borsanyi:2016ksw}.}
 	\label{fig:axion_mass}
 \end{figure}
 %
 Assuming that the PQ symmetry breaks before inflation, the initial conditions (\ie at some $t=0$, after inflation) for the EOM is random. However, we note that $\maT \to 0$ (see \Figs{fig:axion_mass}) -- \ie $\maT \ll H$ -- at very early times. Therefore, after inflation, the EOM is simply
 %
 \begin{equation}
 	\lrb{\dfrac{d^2}{d t^2} + 3 H(t) \ \dfrac{d}{d t}  } \theta(t) = 0 \; ,
 	\label{eq:massless_eom}
 \end{equation}
 %
 which is solved by $\theta = \thetai + C \dint_{0}^t d t' \ \lrb{ \dfrac{a(t'=0)}{a(t')} }^3$, with $\thetai$ a constant and $a$ the scale factor of the Universe. That is, as the Universe expands, $\theta \approx \thetai$. Since we would like to calculate the relic abundance of axions, we can integrate \eqs{eq:eom} from a time after inflation (call it $t = \ti$) such that $ \dot \theta |_{t=\ti} = 0$ and  $\theta|_{t=\ti} \approx \thetai$.   
 
 
 
 \subsection{The WKB approximation}
 %
 In order to solve analytically \eqs{eq:eom}, we assume $\theta \ll 1$, which results in the linearised EOM
 %
 \begin{equation}
 	\lrb{\dfrac{d^2}{d t^2} + 3 H(t) \ \dfrac{d}{d t} + \maT^2(t) } \theta(t) = 0 \; .
 	\label{eq:linear_eom}
 \end{equation}
 
 Using a trial solution $\theta_{\rm trial} = \exp\lrsb{ i \dint d t \ \lrBigb{u(t) +3/2 \ i \ H(t)} }$, and defining $\Omega^2 = \maT^2 - \dfrac{9}{4} H^2 -  \dfrac{3}{2} \dot H $ we can transform the \eqs{eq:linear_eom} to 
 %
 \begin{equation}
 	u^2 = \Omega^2 + i \ \dot u \; ,
 	\label{eq:eom_of_u}
 \end{equation}
 %
 which has a formal solution $u = \pm \sqrt{\Omega^2 + i \dot u}$. Assuming that $\dot u \ll \Omega^2$ and $\dot \Omega \ll \Omega^2$, we can approximate $u$ as
 %
 \begin{equation}
 	u \approx \pm \Omega + \dfrac{i}{2} \dfrac{d \log \Omega}{d t} \;,
 	\label{eq:u_approx}
 \end{equation}
 %
 which results in the general solution of \eqs{eq:linear_eom} 
 %
 \begin{equation}
 	\theta \approx \dfrac{1}{\sqrt{\Omega}} \exp\lrb{-\dfrac{3}{2} \int d t \ H} \lrsb{ A \cos\lrb{ \int d t \ \Omega} +  B \sin\lrb{ \int d t \ \Omega}    } \;. 
 	\label{eq:general_solution_eom_approx}
 \end{equation}
 
 Applying, then, the initial conditions $ \dot \theta |_{t=\ti} = 0$ and  $\theta|_{t=\ti} \approx \thetai$, we arrive at 
 %
 \begin{equation}
 \theta(t) \approx \thetai \sqrt{ \dfrac{ \Omegai }{\Omega (t)} } \lrb{\dfrac{a}{\ai}}^{-3/2} \  \cos\lrb{ \int_{\ti}^t d t^\prime  \ \Omega(t^\prime)}   \;.
 \label{eq:solution_eom_approx} 
 \end{equation}
 
 
 In order to further simplify this approximate result, we note that $\theta$ deviates from $\thetai$ close to $t=\tosc$ -- corresponding to $T = \Tosc$, the so-called ``oscillation temperature" -- $\maT|_{t = \tosc} = 3 H|_{t = \tosc}$, which is defined as the point at which the axion begins to oscillate. 
 %
 This observation allows us to set $\ti = \tosc$.  Moreover, at $t > \tosc$, we approximate $\Omega \approx \maT$, as $H^2$ and $\dot H$ become much smaller than $\maT^2$ quickly after $t=\tosc$. Finally, the axion angle takes the form
 %
 \begin{equation}
 	\theta(t) \approx \thetaosc \lrb{\dfrac{3}{4}}^{1/4} \sqrt{ \dfrac{ \maT|_{t=\tosc} }{\maT  (t)} } \lrb{\dfrac{a}{\aosc}}^{-3/2} \  \cos\lrb{ \int_{\tosc}^t d t^\prime  \ \maT(t^\prime)}   \;,
 	\label{eq:solution_eom_approx_theta_osc} 
 \end{equation}
 %
 where $\thetaosc = \theta|_{t=\tosc}$. This equation is further simplified if we assume that $\thetaosc \approx \thetai$, \ie
 %
 \begin{equation}
 	\theta(t) \approx \thetai \lrb{\dfrac{3}{4}}^{1/4} \sqrt{ \dfrac{ \maT|_{t=\tosc} }{\maT  (t)} } \lrb{\dfrac{a}{\aosc}}^{-3/2} \  \cos\lrb{ \int_{\tosc}^t d t^\prime  \ \maT(t^\prime)}   \;.
 	\label{eq:solution_eom_approx_final} 
 \end{equation}
 %
 It is worth mentioning that the accuracy of this approximation depends, in general, on $\Tosc$; it determines the difference between $\thetai$ and $\thetaosc$, the deviation of $\dot \theta|_{t=\tosc}$ from $0$, and whether $\dot \Omega \ll \Omega^2$. 
 
 
 \paragraph{Axion energy density}
 %
 The energy density of the axion is 
 %
 \begin{eqnarray}
 	\rho_{a} = \dfrac{1}{2} \fa^2 \lrsb{ \dot{\theta}^2 + \maT^2 \theta^2 } \;.
 	\label{eq:rho_a_def} 
 \end{eqnarray}
 %
 For the relic abundance of axions, we need to calculate their energy density at very late times. That is, $\dot{\tilde{m}}_a = 0$, $\maT \gg H$ and $\dot H \ll H^2$. After some algebra, we obtain the approximate form of the energy density (as a function of the scale factor $a$) 
 %
 \begin{eqnarray}
 	\rho_{a} \approx \dfrac{\ma }{2}  \ \fa^2 \ \thetai^2  \ \maT(\aosc) \ \lrb{\dfrac{\aosc}{a}}^3 \;,
 	\label{eq:rho_a0} 
 \end{eqnarray}
 %
 which shows that the energy density of axions at late times scales as the energy density of matter. If there is a period of entropy injection to the plasma for $T<\Tosc$, the axion energy density gets diluted, since 
 %
 \begin{equation}
 	a^3 \ s = \gamma \ \aosc^3 \ s_{\rm osc} \Rightarrow  \lrb{\dfrac{\aosc}{a}}^3 = \gamma^{-1} \dfrac{s}{s_{\rm osc}} \;,
 \end{equation}
 %
 with $\gamma$ the amount of entropy injection to the plasma between $\tosc$ and $t$. Therefore, the present (at $T=T_0$) energy density of the axion, becomes
 %
 \begin{eqnarray}
 	\rho_{a,0} = \gamma^{-1}  \dfrac{s_0}{s_{\rm osc}} \  \dfrac{1 }{2}  \ \fa^2 \ \ma \ \maT_{,{\rm osc}} \ \thetai^2    \;,
 	\label{eq:rho_a_approx} 
 \end{eqnarray}
 with $\ma$ the mass of the axion at $T=T_0$. Notice that the explicit dependence on $\fa$ cancels, since $\maT \sim 1/\fa$. That is, $\fa$ only affects the energy density of the axions through its impact on $\Tosc$. 
 %
 
 \subsection{Notation}\label{sec:notation}
 %
 %The WKB approximation is very useful in order to understand the evolution of the axion field. However, it fails to explain how the oscillation begins before $\dot \Omega \ll \Omega^2$ is reached. In this section we will try to understand the evolution of the axion as generally as possible. 
 %
 The EOM~(\ref{eq:eom}) depends on time, which is not very useful variable in a non-standard comsological setting. Therefore, we introduce 
 %
 \begin{eqnarray}
 	u = \log \dfrac{a}{\ai} \;,
 	\label{eq:natation}
 \end{eqnarray}
 %
 which results in 
 %
 \begin{eqnarray}
 	&\dfrac{d F}{dt} &=  H  \dfrac{d F}{du} 
 	\nonumber \\
 	&\dfrac{d^2 F}{dt^2} &= H^2 \ \lrb{ \dfrac{d^2 F}{du^2} + \dfrac{1}{2} \dfrac{d \log H^2}{du}  \dfrac{d F}{du} }\;.
 	\label{eq:deriv_u}
 \end{eqnarray}
 
 The EOM in terms of $u$, then, becomes
 %
 \begin{equation}
 	\dfrac{d^2  \theta}{du^2} + \lrsb{\dfrac{1}{2} \dfrac{d \log H^2}{du} + 3 } \dfrac{d  \theta}{d u} + \ \lrb{\dfrac{\maT}{H}}^2 \ \sin \theta
 	=0 \;.
 	\label{eq:eom_u}
 \end{equation}
 
 Notice that in a radiation dominated Universe
 %
 $$
 \dfrac{d \log H^2}{du} = -\lrb{ \dfrac{d \log \geff}{d \log T} +4 } \delta_h^{-1}\;,
 $$
 with  $ \delta_h = 1+ \dfrac{1}{3} \dfrac{d \log \heff}{d \log T} $. 
 %
 In a general cosmological setting, the expansion rate is dominated by an energy density that scales as $\rho \sim a^{-c}$, which results in $\dfrac{d \log H^2}{du}  = -c$. However, close to rapid particle annihilations and decays, the evolution of the energy densities change, and $\dfrac{d \log H^2}{du}$ can only be computed numerically.
 
 It is also worth mentioning that the EOM~\ref{eq:eom_u} with the initial condition $\theta(u=0)=\thetai$ and $d\theta/du (u=0)=0$ can be written as a system of first order 
 %
 \begin{eqnarray}
 & \dfrac{d  \zeta}{du} + \lrsb{\dfrac{1}{2} \dfrac{d \log H^2}{du} + 3 } \zeta + \ \lrb{\dfrac{\maT}{H}}^2 \ \sin \theta
 =0 \;. \\ \nonumber
 &\dfrac{d \theta}{d u} - \zeta=0 \;.
 \label{eq:eom_sys}
 \end{eqnarray}
 %
 This form of the EOM is suitable for integration via Runge-Kutta methods, which are briefly discussed in Appendix~\ref{app:RK}.
 
 
 \subsection{Beyond the WKB approximation}\label{sec:beyond_WKB}
 %
 \DK{This needs to be changed}
 The WKB approximation is very useful, as it helps us understand the evolution of the axion field after it begins to oscillate adiabatically. However, it fails to capture the dynamics before the adiabatic conditions are met, result in inaccurate axion relic abundance result. In this section, we examine the deviation of $\thetaosc$ from the initial value of $\theta$, as well how to treat cases where the EOM cannot be linearized.
 
 \subsubsection{Behaviour close to the initial condition}\label{sec:close_to_ini}
 %
 \DK{This subsection is not really correct. The WKB approximation {\bf underestimates} the relic for $\theta \gtrsim 1$, but the first approximation estimate is closer to the true relic. Maybe I should remove the discussion on $\thetaosc$ -- since it can be misleading, and replace it with deviation between numerical and WKB estimates.}
 %
 The accuracy of the approximate evolution of the axion angle given in \eqs{eq:solution_eom_approx_final} depends on the deviation between $\thetai$ and $\thetaosc$. In order to examine their deviation, we expand \eqs{eq:eom} at a time $t =\ti + \delta t$ with $\delta t \to 0$. That is, we use the following approximations 
 %
 \begin{equation*}
 	\ddot{\theta} \approx \dfrac{\dot \theta (\ti+ \delta t)  - \dot \theta (\ti)}{ \delta  t } =  \dfrac{\dot \theta (\ti + \delta t)  }{\delta  t } \;,
 	\qquad
 	\dot \theta(\ti + \delta t) = \dfrac{\theta(\ti + \delta t) - \thetai}{\delta t} \;.
 \end{equation*} 
 %
 and, solve the EOM~\ref{eq:eom} for $\theta(\ti + \delta t)$. This results in
 %
 \begin{equation}
 	\theta(\ti + \delta t)  \approx  \thetai -\dfrac{\delta t^2}{1+3\ \delta t \ H(\ti + \delta t) } \  \maT^2(\ti + \delta t)  \sin \theta(\ti + \delta t)  
 	\approx   \thetai - \delta t^2 \ \maT^2(\ti) \ \sin \thetai  \ + \mathcal{O}(\delta t^3)\;, 
 	\label{eq:theta_dt}
 \end{equation}
 %
 which indicates that the angle decreases (increases) as the temperature drops if $\thetai>0$ ($\thetai<0$). Using the notation introduced in section~\ref{sec:notation}, \eqs{eq:theta_dt} takes the form
 %
 \begin{eqnarray}
 	\theta \approx    \thetai - \delta u^2 \ \lrb{\dfrac{\maT}{H}}_{t=\ti}^2 \ \sin \thetai \;,
 	\label{eq:theta_du}
 \end{eqnarray}
 %
 which can be used to estimate the angle at the oscillation temperature
 %
 \begin{eqnarray}
 	\thetaosc \approx    \thetai -  \lrb{\dfrac{\maT}{H}}_{t=\ti}^2 
 	\lrsb{1- \dfrac{\Ti}{\Tosc}\lrb{\dfrac{\heff_{,\rm ini}}{\heff_{,\rm osc}}\gamma_{\rm osc}}^{1/3}}^{2}   \ \sin \thetai \;,
 	\label{eq:theta_osc}
 \end{eqnarray}
 %
 where  $\gamma_{\rm osc}$ is the entropy injection between $\Ti$ and $\Tosc$. Notice that in the derivation of \eqs{eq:solution_eom_approx_final} we used $\theta_{\rm osc} = \thetai$ as our first approximation. Thus, \eqs{eq:theta_osc} provides a correction that takes into account the deviation between $\theta_{\rm osc} $ and $ \thetai$, and \eqs{eq:rho_a_approx} becomes (for $\thetai \ll 1$)
 %
 \begin{eqnarray}
 	\rho_{a,0} = \gamma^{-1}  \dfrac{s_0}{s_{\rm osc}} \  \dfrac{1 }{2}  \ \fa^2 \ \ma \ \maT_{,{\rm osc}} \ \thetai^2 \lrBiggcb{
 		1 - \ \lrb{\dfrac{\maT}{H}}_{t=\ti}^2 \  \lrb{1- \dfrac{\Ti}{\Tosc}\lrb{\dfrac{\heff_{,\rm ini}}{\heff_{,\rm osc}}\gamma_{\rm osc}}^{1/3}}^{2}   }^2    \;,
 	\label{eq:rho_a_NLO} 
 \end{eqnarray}
 %
 which implies that the WKB approximation overestimates the energy density of the axion, since $\gamma_{\rm osc}$, $H_{\rm ini}$, and $\Tosc$ can vary. 
 
 \subsubsection{Example}\label{sec:comparison_between_WKB_approaches}
 In order to demonstrate the effect of the deviation between $\thetai$ and $\thetaosc$, we show \Figs{fig:WKB_diff_i,fig:WKB_diff_osc}, for different cosmological scenarios. In both figures, the red and black (dashed) lines correspond to standard cosmological scenario and early matter dominated (EMD) Universe, respectively.~\footnote{For the matter dominated case (following the notation of refs.~\cite{Arias:2019uol,Arias:2020qty}), we have used 
 $T_{\rm end}=10^{-2} ~\GeV,\; c=3, \; T_{\rm ini}=10^{12} ~\GeV, \; \text{and}\; r=0.1$.} For the matter dominated case, entropy is injected to the plasma both before and after $\Tosc$. The entropy injection factor, $\gamma_{\rm osc}$, as a function of $\fa$ is shown in \Figs{fig:gamma_osc}.
 
 In \Figs{fig:WKB_diff_i} we show the relative difference between $\Omega h^2$ using \eqs{eq:solution_eom_approx_final} and the result obtained from \mimes. It should be apparent that the accuracy of the estimate based on \eqs{eq:solution_eom_approx_final} depends of $\fa$, since for high values of $\fa$, $\Tosc$ is low. That is, the deviation between $\thetai$ and $\thetaosc$ is greater than the corresponding difference for lower $\fa$, especially for the matter dominated case, where the entropy injection is great (see \Figs{fig:gamma_osc}).
 
 In \Figs{fig:WKB_diff_osc} we show the relative difference between $\Omega h^2$ obtained by integrating numerically \eqs{eq:eom_u}  and using \eqs{eq:solution_eom_approx_theta_osc} with \eqs{eq:theta_osc}. The relative difference between the two estimates, once again, depends $\fa$. It should be noted that the estimate for $\thetaosc$ depends on the choice of $\Ti$. For \Figs{fig:WKB_diff_osc}, we define $\Ti$ as the temperature at which $3H/\maT = 2$. We note that this estimate is more accurate, its accuracy still depends on whether the axion evolves adiabatically at $T<\Tosc$. Moreover, an increase of $\gamma_{\rm osc}$ invalidates the estimate of $\thetaosc$; higher powers of $\delta u$ may be needed.    
 %
 \begin{figure}[t]
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/WKB_diff_i.pdf}
 		\caption{}
 		\label{fig:WKB_diff_i}
 	\end{subfigure}
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/WKB_diff_osc.pdf}
 		\caption{}
 		\label{fig:WKB_diff_osc}
 	\end{subfigure}
 	\begin{center}
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/gamma_osc.pdf}
 		\caption{}
 		\label{fig:gamma_osc}
 	\end{subfigure}
 	\end{center}
 \caption{...}
 \label{fig:WKB_diff}
 \end{figure}
 %
 
 Nevertheless, neither estimates based on \eqs{eq:solution_eom_approx_theta_osc,eq:solution_eom_approx_final} seem to provide a result comparable to numerically integrating \eqs{eq:eom_u}. Therefore, numerical integration should be preferred. 
 
 \subsubsection{Adiabatic invariant and the anharmonic factor}\label{sec:an_fac}
 %
 In oscillatory systems with varying period, the energy is not conserved, and it is usually useful to define an ``adiabatic invariant", which is an approximate constant of motion.
 
 \paragraph{Definition of the adiabatic invariant}
 %
 Given a system with Hamiltonian $\mathcal{H}(\theta,p;t)$, the equations of motion are 
 %
 \begin{equation}
 	\dot p = - \dfrac{\partial \mathcal{H}}{\partial \theta} \;, \;\; 
 	\dot \theta =  \dfrac{\partial \mathcal{H}}{\partial p} \;.
 	\label{eq:hamiltonian_eoms}
 \end{equation}
 
 Moreover, we note that
 %
 \begin{equation}
 	d \Ham = \dot \theta \ d p - \dot p \ d \theta + \dfrac{\partial \Ham}{\partial t} \ d t \;.  
 	\label{eq:total_dH}
 \end{equation}
 
 
 If this system exhibits closed orbits (\eg if it oscillates), we define 
 %
 \begin{equation}
 	J \equiv C \ \oint p \ d \theta \;,
 	\label{eq:adiabatic_inv_def}
 \end{equation}
 %
 where the integral is over a closed path (\eg a period, $T$), and $C$ indicates that $J$ can always be rescaled with a constant. This quantity is the adiabatic invariant of the system, if the Hamiltonian varies slowly during a cycle. That is,
 %
 \[
 \dfrac{d J}{d t} = C \ \oint \lrBigb{\dot p \ d \theta + p \ d \dot \theta} = C \ \dint_{t}^{t+T}  \dfrac{\partial \Ham}{\partial t^\prime} \ d t^\prime \approx T \ \dfrac{\partial \Ham(t^{\prime})}{\partial t^{\prime}}\Big|_{t^{\prime}=t} \approx 0 
 \;. 
 \]
 %
 
 
 
 
 \paragraph{Application to the axion}
 %
 The Hamiltonian that results in the EOM of \eqs{eq:eom} is
 %
 \begin{equation}
 	\Ham = \dfrac{1}{2} \dfrac{p^2}{\fa^2 \ a^3} + V(\theta) \ a^3\;,
 	\label{eq:axion_H}
 \end{equation}
 %
 with 
 %
 \begin{eqnarray}
 	& p = \fa^2 \ a^3 \ \dot \theta \\
 	\label{eq:momentum}
 	& V(\theta) = \maT^2 \fa^2 (1-\cos \theta) \;.
 	\label{eq:potential}
 \end{eqnarray}
 
 Notice that the Hamiltonian varies slowly if $\dot \maT/\maT \ll \maT$ and $H \ll \maT$, which are the adiabatic conditions.  When these conditions are met, the adiabatic invariant for this system becomes
 %
 \begin{equation}
 	J = \dfrac{\oint p \ d \theta}{\pi \fa^2} = \dfrac{1}{\pi \fa^2} \oint \sqrt{ 2\lrb{ \Ham(\theta) - V(\theta) \ a^3} \ \fa^2 a^3 \ }  \ d \theta  =
 	 \dfrac{2}{\pi \fa^2} \int_{-\thetamax}^{\thetamax} \sqrt{ 2\lrb{ \Ham(\thetamax) - V(\theta) \ a^3} \ \fa^2 a^3 \ } d \theta \;,
 	 \label{eq:J_axion_definition}
 \end{equation}
 %
 where we note that $\thetamax$ denotes the maximum of $\theta$ -- the peak of the oscillation, which corresponds to $p=0$. That is, $\Ham(\thetamax) = V(\thetamax) \ a^3$. Therefore, the adiabatic invariant, takes the form 
 
 \begin{eqnarray}
 	J=&  \dfrac{2 \sqrt{2} }{\pi \fa}  \int_{- \thetamax} ^{\thetamax}  \sqrt{ V(\thetamax) - V(\theta) } a^{3} d \theta = 
 	\dfrac{2 \sqrt{2} }{\pi} \ \maT \, a^3 \ \dint_{- \thetamax}^{\thetamax} \sqrt{\cos \theta - \cos \thetamax} \ d \theta  
 	\;,
 	\label{eq:J_axion_derivation}
 \end{eqnarray}
 %
 where, for the last equality. we have used the adiabatic conditions, \ie negligible change of $\maT$ and $a$ during one period. Usually, the adiabatic invariant is written as~\cite{Lyth:1991ub,Bae:2008ue} 
 %
 \begin{equation}
 	J = a^3 \ \maT \ \thetamax^2  \, f(\thetamax)  \;,
 	\label{eq:J_axion_final_form}
 \end{equation}
 %
 where 
 \begin{equation}
 	f(\thetamax) =\dfrac{ 2 \sqrt{2}}{\pi \thetamax^2 } \dint_{- \thetamax}^{\thetamax} d \theta \sqrt{ \cos \theta - \cos \thetamax } \;,
 	\label{eq:anharmonic_f}
 \end{equation}
 %
 is called the anharmonic factor, with $ 0.5 \lesssim f(\thetamax) \leq 1$ (see \Figs{fig:anharmonic_factor}).
 
 
 \begin{figure}[t]
 	\includegraphics[width=1\textwidth]{figs/anharmonic_factor.pdf}
 	\caption{The anharmonic factor for $0 \leq \thetamax < \pi $.}
 	\label{fig:anharmonic_factor}
 \end{figure}
 
 
 
 \paragraph{The role of the adiabatic invariant in the axion relic energy density}
 %
 The adiabatic invariant allows us to calculate the maximum value of the angle $\theta$ at late times from its corresponding value at some point after the adiabatic conditions where met. It allows us to calculate the energy density of the axions field for $\theta \gtrsim 1$ since we have taken into account the exact potential.
 
 In order to do this, we can numerically integrate \eqs{eq:eom}, and identify the maxima of $\theta$. Once the adiabatic conditions are fulfilled, we can stop the integration at a peak, $\thetamax_{,*}$ -- which corresponds to $T=T_{*}$ and $a=a_{*}$. Then, the value of the maximum angle today ($\thetamax_{,0} \ll 1$) is related to $\thetamax_{,*}$ via
 %
 \begin{eqnarray}
 	\thetamax_{,0}^2 &=  \lrb{\dfrac{a_*}{a_0}}^3 \ \dfrac{\maT_{,*}}{\ma} \ f(\thetamax_{,*}) \ \thetamax_{,*}^2  =
 	\gamma^{-1} \ \dfrac{s_0}{s_*} \ \dfrac{\maT_{,*}}{\ma} \ f(\thetamax_{,*}) \ \thetamax_{,*}^2 
 	\; .
 	\label{eq:theta_relation}
 \end{eqnarray}
 %
 Using this, and since we evaluate the energy density at the maximum of $\theta$ (\ie $\dot \theta = 0$), we can determine the energy density of axions today from \eqs{eq:rho_a_def}. That is,
 %
 \begin{equation}
 	\rho_{a,0} = \gamma^{-1} \ \dfrac{s_0}{s_*} \ \ma \ \maT_{,*} \ \dfrac{1}{2} \ \fa^2 \ \thetamax_{,*}^2 \;  \ f(\thetamax_{,*}) \;,
 	\label{eq:rho_axion_exact}
 \end{equation}
 %
 where $\gamma$ is the entropy injection coefficient between $T_{*}$ and $T_0$, defined from 
 %
 \begin{equation}
 	a^3(T_0) \ s(T_0) = \gamma \ a^3(T_*) \ s(T_*) \;
 	\label{eq:entropy_injection_gamma}
 \end{equation}
 
 Notice that \eqs{eq:rho_axion_exact} is similar to the corresponding WKB result~(\ref{eq:rho_a_approx}) at $\tosc \to t_*$, multiplied by the anharmonic factor $f(\thetamax_{,*})$. That is, if the axion starts to evolve adiabatically close to $t=\tosc$, and $\thetaosc \ll 1$, the WKB approximation is valid.
 
 
 
 \section{First steps with \mimes}\label{sec:first_steps}
 \setcounter{equation}{0}
 %
 \begin{lstlisting}[caption={Commands in order to download \mimes, the differential equation solvers, and the interpolation libraries.},captionpos=b, label=list:git_download]
 	git clone git@github.com:dkaramit/MiMeS.git
 	cd MiMeS
 	git submodule init
 	git submodule update --remote
 \end{lstlisting}
 %
 The library can downloaded from \href{https:/github.com/dkaramit/MiMeS}{github.com/dkaramit/MiMeS}. Since the library depends on {\tt NaBBODES}~\cite{NaBBODES} {\tt SpleSplines}~\cite{SimpleSplines}, one has to download the, too. However, in order to get \mimes ready, one can run the commands shown in listing~\ref{list:git_download}. Although this method should be preferred in order to get the latest version, \mimes can also be downloaded from \href{https:/mimes.hepforge.org/downloads/}{mimes.hepforge.org/downloads}.
 
 
 Once everything is downloaded successfully, we can go inside the \mimes directory, and run \run{bash configure.sh}~\footnote{It is recommended to use {\tt bash}.} and \run{make}.  The {\tt bash} script {\tt configure.sh}, just writes some paths to some files, formats the data files provided in an acceptable format (in section~\ref{sec:input} the format is explained), and makes some directories.
 %
 The {\tt makefile} is responsible for compiling some examples and checks, as well as the shared libraries that needed for the \PY interface.  If everything runs successfully, there should be two new directories {\tt exec} and {\tt lib}. Inside {\tt exec}, there are several executables that ready to run, in order to ensure that the code runs (\eg no segmentation fault occurs). For example, {\tt exec/AxionSolve\_check.run}, should print the values of the parameters $\thetai$ and $\fa$, the oscillation temperature and the corresponding value of $\theta$, the evolution of the axion (\eg temperature, $\theta$, $\rho_{a}$, etc.), and the values of various quantities on the peaks of the oscillation.  In the directory {\tt lib}, there are several shared libraries for the \PY interface.
 
 Although there are various options available at compile-time, we first discuss how \mimes can be used, in order for the role of these options to be clear. 
 
 
 \subsection{First examples}\label{sec:First_examples} 
 %
 There are several examples in \CPP ({\tt MiMeS/UserSpace/Cpp}) and \PY ({\tt MiMeS/UserSpace/Python}), as well as \JUPY  notebooks ({\tt MiMeS/UserSpace/JupyterNotebooks}), that show in detail how \mimes can be used. 
 
 \subsubsection{Using \mimes in \CPP}\label{sec:cpp_first_example}
 %
 The class that is responsible for the solution of the EOM is \cppin{mimes::Axion<LD,Solver,Method>}, located in {\tt MiMeS/src/Axion}. However, in order to use it, we have to define the mass of the axion as a function of the temperature and $\fa$. The axion mass is defined as an instance of the \cppin{mimes::AxionMass<LD>}, which is defined in the header file {\tt \mimes/src/AxionMass/AxionMass.hpp}. We should note that {\tt LD} is the numerical type we use (\eg { \tt long double}). The other template arguments are related to the differential equation solver, and their role will be explained in later sections. 
 
 Therefore, at the top of the main {\tt .cpp} file, we need to include 
 %
 \begin{cpp}
 	#include "MiMeS/src/AxionMass/AxionMass.hpp"
 	#include "MiMeS/src/Axion/AxionSolve.hpp"
 \end{cpp}
 %
 Notice that if the this {\tt .cpp} file is not in the root directory of \mimes, we need to compile it using the flag {\tt -Ipath-to-root}, "path-to-root" the relative path to the root directory of \mimes; \eg if the {\tt .cpp} is in the {\tt MiMeS/UserSpace/Cpp/Axion} directory, this flag should be {\tt -I../../../}.
 
 The mass of the axion can be defined in two ways; either via a data file or by a user defined function. 
 
 \paragraph{Axion mass form data file} 
 In many cases, axion mass cannot be written in a closed form. In these cases, the user may provide a data file. Then, the axion mass can be defined via this file as
 %
 \begin{cpp}
 	mimes::AxionMass<LD> axionMass(chi_PATH, minT, maxT);
 \end{cpp}
 %
 Here {\tt chi\_PATH} is a (relative or absolute) path to a file with two columns; $T$ (in $\GeV$) and $\chi$ (in $\GeV^4$), with increasing $T$. In this case, the axion mass is interpolated between the temperatures {\tt minT} and {\tt maxT}. These two parameters are just suggestions, and the actual interpolation limits, {\tt  TMin} and {\tt TMax}, are chosen as the closest temperatures in the data file. That is, in general, {\tt TMin} $\geq$ {\tt minT} and {\tt TMax} $\leq$ {\tt maxT}. Beyond these limits, the mass is assumed to constant by default. However, one can change this by using 
 %
 \begin{cpp}
 	axionMass.set_ma2_MAX(ma2_MAX);
 	axionMass.set_ma2_MIN(ma2_MIN);
 \end{cpp}
 %
 with {\tt ma2\_MAX} and {\tt ma2\_MIN} the axion mass squared as functions (or any other callable objects) with signatures {\tt LD ma2\_MAX(LD T, LD fa)} and {\tt LD ma2\_MIN(LD T, LD fa)}. These functions are called for $T\geq${\tt TMax} and $T\leq${\tt TMin}, respectively. Usually, in order to ensure continuity of the axion mass, one needs to know {\tt TMin}, {\tt TMax}, $\chi(${\tt TMin}$)$, and $\chi(${\tt TMax}$)$; which can be found by calling {\tt axionMass.getTMin()}, {\tt axionMass.getTMax()}, {\tt axionMass.getChiMin()}, and {\tt axionMass.getChiMax()}, respectively.
 
 \paragraph{Axion mass function} 
 In some cases, the dependence of the axion mass on the temperature is known. If this is the case, the user can define the axion mass via 
 %
 \begin{cpp}
 	mimes::AxionMass<LD> axionMass(ma2);
 \end{cpp}
 %
 with {\tt ma2} the axion mass squared as a callable object with signature {\tt LD ma2(LD T, LD fa)}.
 
 \paragraph{The EOM solver}
 Once the axion mass is defined, we can declare a variable that will be used to solve the EOM, as
 %
 \begin{cpp}
 	mimes::Axion<LD,Solver,Method> ax(theta_i, fa, umax, TSTOP, ratio_ini, 
 	N_convergence_max, convergence_lim, inputFile, &axionMass, initial_step_size,
 	minimum_step_size, maximum_step_size, absolute_tolerance, relative_tolerance, 
 	beta, fac_max, fac_min, maximum_No_steps);
 \end{cpp}
 %
 Here, {\tt LD} should be the desirable numeric type to be used; it is recommended to use {\tt long double}, but other choices are also available as we discuss later. Moreover {\tt Solver} and {\tt Method} depend on the type of Runge-Kutta the user chooses (see section~\ref{sec:options} as well as the Appendices~\refs{app:RK,app:usr_input} for more details). 
 
 The various parameters are as follows:
 %
 \begin{enumerate}
 	\item {\tt theta\_i}: initial angle.
 	\item {\tt fa}: the PQ scale.
 	\item {\tt umax }: if $u>${\tt umax} the integration stops (remember that $u=\log(a/a_i)$). Typically, this should be a large number ($\sim 1000$), in order to avoid stopping the integration before the axion begins to evolve  adiabatically.    
 	\item {\tt TSTOP}: if the temperature drops below this, integration stops. In most cases this should be around 
 	$10^{-4}~\GeV$, in order to be sure that any entropy injection has stopped before integration stops (since BBN bounds~\cite{Kolb:206230,Peebles:1993} should not be violated).
 	\item {\tt ratio\_ini}: integration starts when $3H/\maT \approx${\tt ratio\_ini} (the exact point depends on the file ``{\tt inputFile}", which we will see later). 
 	\item  {\tt N\_convergence\_max} and {\tt convergence\_lim}: integration stops when the relative difference 
 	between two consecutive peaks is less than {\tt convergence\_lim} for {\tt N\_convergence\_max} 
 	consecutive peaks.
 	\item  {\tt inputFile}: relative (or absolute) path to a file that describes the cosmology. the columns should be: $u$ $T ~[\GeV]$ $\log H$, sorted so that $u$ increases.~\footnote{One can run \run{bash MiMeS/src/FormatFile.sh inputFile} in order to sort it and remove any unwanted duplicates. See Appendix~\ref{app:util} for details of {\tt MiMeS/src/FormatFile.sh}.}
 	%
 	It is important to remember that \mimes assumes that the entropy injection has stopped before the lowest temperature of given in {\tt inputFile}. Since \mimes is unable to guess the cosmology beyond what is given in this file, the user has to make sure that there are data between the initial temperature (which corresponds to {\tt ratio\_ini}, and {\tt TSTOP}).
 	
 	\item {\tt axionMass}: an instance of the \cppin{mimes::AxionMass<LD>} class, passed by pointer. 
 	
 	\item {\tt initial\_stepsize} (optional): initial step the solver takes. 
 	
 	\item {\tt maximum\_stepsize} (optional): This limits the step-size to an upper limit. 
 	\item {\tt minimum\_stepsize} (optional): This limits the step-size to a lower limit. 
 	
 	\item {\tt absolute\_tolerance} (optional): absolute tolerance of the RK solver
 	
 	\item {\tt relative\_tolerance} (optional): relative tolerance of the RK solver.
 	%
 	Generally, both absolute and relative tolerances should be $10^{-8}$. 
 	In some cases, however, one may need more accurate result (\eg if {\tt f\_a} is extremely high, 
 	the oscillations happen violently, and the system destabilizes). Whatever the case, if the  
 	tolerances are below $10^{-8}$, long doubles have be used. \mimes by default uses {long double} variables, 
 	in order to change it see the options available in section~\ref{sec:input}.
 	
 	\item {\tt beta} (optional): controls how agreesive the adaptation is. Generally, it should be around but less than 1.
 	
 	\item {\tt fac\_max},  {\tt fac\_min} (optional): the stepsize does not increase more than fac\_max, and less than fac\_min. 
 	This ensures a better stability. Ideally, {\tt fac\_max}$=\infty$ and {\tt fac\_min}$=0$, but in reality one must 
 	tweak them in order to avoid instabilities.
 	
 	\item {\tt maximum\_No\_steps} (optional): maximum steps the solver can take Quits if this number is reached even if integration
 	is not finished. 
 \end{enumerate}
 %
 in order to understand the usage of the optional parameters, some basic techniques of Runge-Kutta methods are discussed in Appendix~\ref{app:RK}. 
 
 The EOM~(\ref{eq:eom_u}), then can be solved using 
 %
 \begin{cpp}
 	ax.solveAxion();
 \end{cpp}
 %
 Once the EOM is solved, we can access $\Tosc$, $\thetaosc$, and $\Omega h^2$  via {\tt ax.T\_osc}, {\tt ax.theta\_osc}, and {\tt ax.relic}. The entire evolution (the points the integrator took) of the axion angle is stored in {\tt ax.points}, which is a two-dimensional {\tt std::vector}, with the columns corresponding to  $a$, $T~[\GeV]$, 
 $\theta$, $d\theta/du$, $\rho_a$. Moreover, the peaks of he oscillation are stored in another two-dimensional {\tt std::vector}, with the columns corresponding to $a$, $T~[\GeV]$, $\thetamax$, $d\theta/du=0$, $\rho_a$, $J$. We should note that the peaks are identified using linear interpolation between integration points, in order to ensure that $d\theta/du = 0$. That is, the values stored in {\tt ax.peaks} do not exist in {\tt ax.points}. Finally, the local errors (see Appendix~\ref{app:RK}) of the integration points of $\theta$ and $\zeta$ are stored in {\tt ax.dtheta} and {\tt ax.dzeta}.
 
 
 \paragraph{Changing axion mass definition}
 %
 The axion mass definiton can change at any time by changing the data file (or {\tt ma2\_MIN} and {\tt ma2\_MAX}) or using using 
 %
 \begin{cpp}
 	ax.set_ma2(ma2);
 \end{cpp}
 %
 with {\tt ma2} a callable object with signature {\tt LD ma2(LD T, LD fa)}.
 
 However, since integration starts at a temperature determined by {\tt ratio\_ini}, if the mass changes (including the definitions beyond the interpolation limits), we have to remake the interpolation of the underlying cosmology described by the parameter {\tt inputFile}. Thus, if we  change the definition of the mass, we need to call 
 %
 \begin{cpp}
 	ax.restart();
 \end{cpp}
 %
 This function remakes the interpolations, clears all vector, and sets all variable to $0$. 
 
 
 \paragraph{Changing initial condition}
 %
 The final member function is \cppin{mimes::Axion::setTheta\_i}, which allows the user to set a different $\thetai$ without generating another instance.~\footnote{Since the interpolations of the data of {\tt inputFile} are made inside the constructor of the \cppin{mimes::Axion<LD,Solver,Method>} class, \cppin{mimes::Axion<LD,Solver,Method>::setTheta\_i} is a faster choice if ones needs to solve the EOM for a different initial condition.} This function is used as    
 %
 \begin{cpp}
 	ax.setTeta_i(new_theta_ini);
 \end{cpp}
 %
 where {\tt new\_theta\_ini} is the new value of $\thetai$. Running this function resets all variables to $0$ (except {\tt T\_osc} and {\tt a\_osc}, since they should not change), and clears all {\tt std::vector} variables, which allows the user to simply run \run{ax.solveAxion();} as if {\tt ax} was a freshly defined instance.  
 
 
 
 \subsubsection{Using \mimes in \PY}\label{sec:begin_py}
 %
 The modules for the \PY interface are located in {\tt MiMeS/src/interfacePy}. Although the usage of the {\tt AxionMass} and {\tt Axion} classes is similar to the \CPP case, it is worth showing explicitly how the \PY interface works.
 
  
  
  
 The two classes the modules {\tt interfacePy.AxionMass} and {\tt interfacePy.Axion}, and can be loaded in a \PY script as 
 %
 \begin{py}
 	from sys import path as sysPath
 	from os import path as osPath
 	sysPath.append(osPath.join(osPath.dirname(__file__), 'path_to_src'))
 	from interfacePy.AxionMass import AxionMass
 	from interfacePy.Axion import Axion
 \end{py}
 %
 It is important that {\tt 'path\_to\_src'} provides the relative path to the {\tt MiMeS/src} directory. For example, if the script is located in {\tt MiMeS/UserSpace/Python}, {\tt 'path\_to\_src'} should be {\tt '../../src'}.
 
 \paragraph{Axion mass definition via a data file}
 As before, we first need to define the axion mass. Similarly to the previous case, in order to define the axion mass via a file
 In many cases, axion mass cannot be written in a closed form. In these cases, the user may provide a data file. Then, the axion mass can be defined via this file as
 %
 \begin{py}
 	AxionMass axionMass(chi_PATH, minT, maxT)
 \end{py}
 %
 Here, the constructor requires the same parameters as in \CPP. Moreover, the axion mass beyond the interpolation limits can be changed via
 %
 \begin{py}
 	axionMass.set_ma2_MAX(ma2_MAX)
 	axionMass.set_ma2_MIN(ma2_MIN)
 \end{py}
 %
 Although the naming is the same as in the \CPP case, there on important difference. Namely, {\tt ma2\_MAX} and {\tt ma2\_MIN} have to be functions (that take $T$ and $\fa$ as arguments and return $\maT^2$), and cannot be any other callable object. The reason is that \mimes uses the {\tt ctypes} module, which only works with objects compatible with {\tt C}. 
 %
 Moreover, the values of {\tt TMin}, {\tt TMax}, $\chi(${\tt TMin}$)$, and $\chi(${\tt TMax}$)$ can be obtained by {\tt axionMass.getTMin()}, {\tt axionMass.getTMax()}, {\tt axionMass.getChiMin()}, and {\tt axionMass.getChiMax()}, respectively.
 
 \paragraph{Axion mass definition via a function}
 %
 Again this can be done as
 %
 \begin{py}
 	AxionMass axionMass(ma2)
 \end{py}
 %
 with {\tt ma2} the axion mass squared, which should be a function (not any callable object) of $T$ and $\fa$.
 
 
 \paragraph{The EOM solver}
 %
 We can define an {\tt Axion} instance as follows 
 %
 \begin{py}
 	ax=Axion(theta_i, fa, umax, TSTOP, ratio_ini, N_convergence_max, convergence_lim, 
 	inputFile, axionMass, initial_step_size, minimum_step_size, maximum_step_size, 
 	absolute_tolerance, relative_tolerance, beta, fac_max, fac_min, maximum_No_steps)
 \end{py}
 %
 Here the input parameters are the same as in the \CPP case (the usage of the class can be found by running {\tt ?Axion} after loading the module). The only slight difference is that the {\tt axionMass} is not passed as a pointer; which is done internally using {\tt ctypes}. 
 
 
 Using the defined variable ({\tt ax} in this example), we can simply run  
 %
 \begin{py}
 	ax.solveAxion()
 \end{py}
 %
 in order to solve the EOM of the axion. In contrast to the \CPP implementation, this only gives us access to $\Tosc$, $\thetaosc$, and $\Omega h^2$; the corresponding variables are {\tt ax.T\_osc}, {\tt ax.theta\_osc}, and {\tt ax.relic}. In order to get the evolution of the axion field, we need to run 
 %
 \begin{py}
 	ax.getPoints()
 \end{py}
 %
 This will make {\tt numpy} arrays that contain the scale factor ({\tt ax.a}), temperature ({\tt ax.T}), $\theta$ ({\tt ax.theta}), its derivative with respect to $u$ ({\tt ax.zeta}), and the energy density of the axion ({\tt ax.rho\_axion}).
 
 Moreover, in order to get the various quantities on the peaks of the oscillation, we can run
 %
 \begin{py}
 	ax.getPeaks()
 \end{py}
 %
 This makes {\tt numpy} arrays that contain the scale factor ({\tt ax.a\_peak}), temperature ({\tt ax.T\_peak}), $\theta$ ({\tt ax.theta\_peak}), its derivative with respect to $u$ ({\tt ax.zeta\_peak}, which should be equal to $0$), the energy density of the axion ({\tt ax.rho\_axion\_peak}), and the values of the adiabatic invariant on the peaks ({\tt ax.adiabatic\_invariant}).
 
 Moreover, we can run the following
 %
 \begin{py}
 	ax.getErrors()
 \end{py}
 %
 in order to store the local errors (see Appendix~\ref{app:RK}) of $\theta$ and $\zeta$ in {\tt ax.dtheta} and {\tt ax.dzeta}, respectively. 
 
 \paragraph{Changing axion mass definition}
 %
 We can change the axion mass by changing the file, {\tt ma2\_MIN} and {\tt ma2\_MAX}, or using 
 %
 \begin{cpp}
 	ax.set_ma2(ma2)
 \end{cpp}
 %
 with {\tt ma2} a function that takes $T$ and $\fa$, and returns the $\maT^2$. As in the \CPP case, if the definition of the mass of the axion changes (including the definitions beyond the interpolation limits), we have to call
 %
 \begin{cpp}
 	ax.restart()
 \end{cpp}
 %
 in order to remake the interpolation of cosmological quantities and reset the various variables.x
 
 \paragraph{Changing the initial condition}
 %
 The initial condition $\thetai$ can be changed using 
 %
 \begin{py}
 	ax.setTeta_i(new_theta_ini)
 \end{py}
 %
 which is faster than running the constructor again, since all the interpolations are reused. However, running this function, erases all the arrays, and resets all variables to $0$ (except {\tt T\_osc} and {\tt a\_osc}, as they should not change). 
 
 \paragraph{Importand note} The {\tt Axion} class constructs a pointer to an instance of the underlying \cppin{mimes::Axion} class, which has to be manually deleted. Therefore, once {\tt ax} is used it must be deleted, \ie we need to run 
 %
 \begin{py}
 	del ax
 \end{py}
 %
 We should note that this must run even if we assign another instance to the same variable {\tt ax}, otherwise we risk running out of memory.
 
 
 \section{Assumptions and user input}\label{sec:assumptions}
 \setcounter{equation}{0}
 %
 \mimes only makes a few, fairly general, assumptions. First of all, it is assumed that the axion energy density is always subdominant compared to radiation or any other component of the Universe, and that decays and annihilations of particles have a negligible effect on the axion energy density. Moreover, the initial condition is always assumed to be $\theta_{t=\ti} = \thetai$ and $\dot \theta|_{t=\ti}=0$. 
 %
 Furthermore, it is also assumed that $3H/\maT$ increases monotonically at high temperatures. 
 %
 Also, it is assumed that the entropy is resumes its conserved status at a temperature higher than the minimum temperature of  {\tt inputFile} (which is required by the constructor of the \cppin{mimes::Axion<LD,Solver,Method>} class).  
 %
 Finally, \mimes does not try to predict anything regarding the cosmology. Therefore, the temperatures in {\tt inputFile} {\em must} cover the entire region of  integration; \ie the maximum temperature has to be larger than the one required to reach {\tt ratio\_ini}, while the minimum one should be lower than {\tt TSTOP}.
 
 \subsection{Options at Compile-time}\label{sec:options}
 %
 The user has a number of options regarding different aspects for the code. The various choices for the shared libraries used by the \PY interface are given in {\tt \mimes/Definitions.mk}. while the corresponding options for the \CPP examples are in the {\tt Definitions.mk} files in the directories inside {\tt \mimes/UserSpace/Cpp}.  The options correspond to different variables, which are
 %
 \begin{enumerate}
 	\item {\tt rootDir}: the relative path of root directory of \mimes.  
 	%
 	\item {\tt LONG}: this should be either {\tt long} or omitted. If omitted, the type of the numeric values is {\tt double} (double precision). On the other hand, if {\tt LONG=long},  the type is  {\tt long double}. Generally, using {\tt double} should enough. For the sake of numerical stability, then, it is advised to always use {\tt LONG=long}, as it a safer option. The reason is that the axion angle redshifts, and can become very small, which introduces ``rounding errors". Moreover, if the parameters {\tt absolute\_tolerance} or {\tt absolute\_tolerance} are chosen to be below $\sim 10^{-8}$, then double precision numbers may not be enough, and {\tt LONG=long} is preferable.  This choice comes at the cost of speed; double precision operations are usually preformed much faster. It is important to note that {\tt LONG} defines a macro with the same name, which then defines the macro {\tt LD} (in all the {\tt .cpp} files) as {\tt \#define LD LONG double}.
 	%
 	\item {\tt LONGpy}: the same as {\tt LONG}, but for the \PY interface.
 	%
 	\item {\tt SOLVER}: \mimes uses the ordinary differential equation (ODE) integrators of ref.~\cite{NaBBODES}. Currently, there are two available choices; {\tt Rosenbrock} and {\tt RKF}. The former is a general embedded Rosenbrock implementation and it is used if {\tt Solver=1}, while the latter is a general explicit embedded Runge-Kutta implementation and can be chose by using {\tt SOLVER=2} (a brief description of how these algorithms are implemented can be found in Appendix~\ref{app:RK}). The default option for that \mimes uses is {\tt SOLVER=1}, because the axion EOM tends to oscillate rapidly. However, in some cases, a high order explicit method may also work. This variable, is used to define a macro with the same name and value in {\tt \mimes/src/Axion/AxionSolve.hpp}.
 	%
 	\item {\tt METHOD}: Depending on the type of solver is chosen, there are some available methods.~\footnote{It is worth mentioning that {\tt NaBBODES} is built in order to be a template for all possible Rosenbrock and explicit Runge-Kutta embedded methods, and one can provide their own Butcher tableau if they want to use another method, as shown in Appendix~\ref{app:RK}. This variable, is used to define a macro with the same name and value in {\tt \mimes/src/Axion/AxionSolve.hpp}.} 
 	\begin{itemize}
 		\item 	For {\tt SOLVER=1}, the available methods are 
 		{\tt METHOD=RODASPR2} and {\tt METHOD=ROS34PW2}. The {\tt RODASPR2} choice is a fourth order Rosenbrock-Wanner method (more information can be found in ref.~\cite{RANG2015128}). The {ROS34PW2} choice corresponds to a third order Rosenbrock-Wanner method~\cite{RangAngermann2005}. 
 		%
 		\item 	For {\tt SOLVER=2}, the only reliable method available in {\tt NaBBODES} is the Dormand-Prince~\cite{DORMAND198019} chosen if {\tt METHOD=DormandPrince}, which is an explicit Runge-Kutta method of seventh order.
 	\end{itemize}
 	%
 	\item {\tt CC}: the \CPP compiler that one chooses to use. The default option is {\tt CC=g++}, which is the {\tt GNU} \CPP compiler, and is available for all systems. Another option is to use the {\tt clang} compiler, which is chosen by {\tt CC=clang -lstdc++}. \mimes is mostly tested using {\tt g++}, but {\tt clang} also seems to work (and the resulting executables are sometimes faster), but the user has to make sure that their version of the compiler of choice supports the \CPP17 standard, otherwise \mimes probably will not work.
 	%
 	\item {\tt OPT}: Optimization level of the compiler. By default, this is {\tt OPT=O3}, which produces executables that are marginally faster than {\tt OPT=O1} and {\tt OPT=O2}, but significantly faster than {\tt OPT=O0}. There is another choice, {\tt OPT=Ofast}, but it can cause various numerical instabilities, and is generally considered dangerous -- although we have not observed any problems when running \mimes. 
 \end{enumerate}
 %
 These options can only be used when the compilation is done using the various {\tt makefile} files. For compilation of a user provided {\tt .cpp} file, these have to be define by the user inside this file. If one compiles the examples without {\tt makefile}, the variables {\tt LONG},  {\tt LONGpy}, {\tt SOLVER}, and {\tt METHOD} correspond to macros with the same name.~\footnote{Macros can be defined using the {\tt -D} flag
 	in the compiler. For example {\tt METHOD=DormandPrince} is the same as passing {\tt -DMETHOD=DormandPrince} to the compiler.} It should be pointed-out that,  in the examples, {\tt LONG} is used to define a macro {\tt LD} which becomes {\tt double} or {\tt long double} accordingly. the actual choice of numeric type is passed as a template parameter in all classes.
 \subsection{User input}\label{sec:input}
 %
 \subsubsection{Compile-time input}\label{sec:compile_time_input} 
 %
 \paragraph{Files} \mimes requires files that provide data for the relativistic degrees of freedom (RDOF) of the plasma, and the anharmonic factor. Although \mimes is shipped with the standard model RDOF found in~\cite{Saikawa:2020swg}, and a few points for $f(\thetamax)$ introduced in \eqs{eq:anharmonic_f}, the user is free change them via the corresponding variables in {\tt Paths.mk} (in the root directory of \mimes). Moreover, there is a set of data for the QCD axion mass as calculated in ref.\cite{Borsanyi:2016ksw}.
 %
 The variables pointing to these data files are {\tt cosmoDat}, {\tt axMDat}, and {\tt anFDat}, for the RDOF, axion mass, and the anharmonic factor; respectively.
 
 The format of the files has to be the following:
 %
 \begin{itemize}
 	\item The RDOF data must be given in three columns; $T$ (in $\GeV$), $\heff$, and $\geff$.
 	%
 	\item The axion mass data must be given in two columns; $T$ (in $\GeV$), $\chi$ (in $\GeV^4$). Here, $\chi$ is defined as in \eqs{eq:axion_mass_def}. 
 	The user can provide a function instead of data for the axion mass, by leaving the  {\tt axMDat} variable empty. 
 	%
 	\item The data for the anharmonic factor must be give in two columns   $\thetamax$ $f(\thetamax)$; with increasing $\thetamax$.
 \end{itemize}
 %
 The paths to these files should be given at compile time. That is, once {\tt Paths.mk} changes, \run{bash configure.sh} and then \run{\tt make} must be ran. The user can change the content of the data files (without changing their paths), in order to use them without compiling \mimes again. However, the user has to make sure that all the files are sorted so that the values of first column increase (with no duplicates or empty lines). In order to ensure this, it is advised to run \run{bash FormatFile.sh path-to-file} (in Appendix~\ref{app:util} there are some details on {\tt MiMeS/src/FormatFile.sh}), in order to format the file (that should exist in \run{path-to-file}) complies with the requirements of \mimes.
 
 These paths are stored as strings in {\tt \mimes/src/misc\_dir/path.hpp} at compile-time (they are defined as {\tt constexpr}), and can be accessed once this header file is included. The corresponding variables are {\tt cosmo\_PATH}, {\tt chi\_PATH}, and {\tt anharmonic\_PATH}, for the path to data file of the plasma quantities, $\chi(T)$, and $f(\thetamax)$; respectively. Although, the axion mass data file may be omitted -- since the axion mass is defined by the user, the variable {\tt chi\_PATH} is still useful if the axion mass is defined via a data file, as it is automatically converted to an absolute path.~\footnote{Absolute paths have the advantage to be accessible from everywhere else in the system. Thus, executables that seek the corresponding files can be called and copied easily.}
 
 
 
 
 \subsubsection{Run-time input}\label{sec:run_time_input}
 %
 The run-time user input is described in sec.~\ref{sec:first_steps}. The user has to provide at least the parameters that describe the axion evolution, $\thetai$ and $\fa$. 
 
 Moreover,  the maximum allowed value of $u$ and the minimum value of $T$, allow the user to decide when the integration has to stop even if the axion has not reached its adiabatic evolution. Ideally, {\tt umax}$=\infty$ and {\tt TSTOP}$=0$, but \mimes is designed to be as general as possible and there may be cases where one needs to stop the integration abruptly.~\footnote{These two variables are not optional, because the user must be aware of them, in order to choose them according to their needs.}
 
 Furthermore, {\tt ratio\_ini} allows the user to choose a desired point at which the interpolation of the data in {\tt inputFile} begins. This can save valuable computation time as well as memory, as only the necessary data are stored and searched. Generally, for {\tt ratio\_ini}$>10^{3}$, the relic abundance becomes
 independent from  {\tt ratio\_ini}, but one has to choose it carefully, in order to find a balance between accuracy and computation time.
 
 Finally, the convergence conditions -- \ie~{\tt N\_convergence\_max} and {\tt convergence\_lim} -- allow the user to decide when the adiabatic evolution of the axion begins. Generally, the relic abundance does not have a strong dependence on these parameters as log as they are  {\tt N\_convergence\_max}$>3$ and {\tt convergence\_lim}$<10^{-1}$ (\ie the adiabatic invariant does not vary more that $10\%$ for three consecutive peaks of the oscillation). 
 %
 However, we should note that greedy choices (\eg~{\tt N\_convergence\_max}$=100$ and {\tt convergence\_lim}$=10^{-5}$) are dangerous, as $\theta$ tends to oscillate rapidly and destabilize the differential equation solver. Therefore, these parameters should be chosen carefully, in order to ensure that integration stops when axion has reached its adiabatic evolution, without destabilising the EOM.
 
 \subsection{Complete Examples}\label{sec:complete_examples}
 %
 Although one can modify the examples provided in {\tt \mimes/UserSpace}, in this section we show a complete example in both \CPP and \PY. 
 %
 
 The underlying cosmology is assumed to be an EMD scenario. In terms of the parametrisation of ref.~\cite{Arias:2020qty}, it is defined through the same parameters as in the example shown in section~\ref{sec:comparison_between_WKB_approaches}.
 %
 and the evolution of the energy densities of the plasma and a fluid $\Phi$ is shown in \Figs{fig:energy_densities}.
 %
 \begin{figure}[t]
 	\includegraphics[width=1\textwidth]{figs/energy_densities.pdf}
 	\caption{The energy densities of the plasma and a hypothetical decaying fluid. The parameters of the cosmological scenario  $T_{\rm END}=10^{-2}~\GeV$, $c=3$, $T_{\rm ini}=10^{12}~\GeV$, and $r=0.1$; the parameters  follow the definitions in ref.~\cite{Arias:2020qty}. }
 	\label{fig:energy_densities}
 \end{figure}
 %
 The various regime changes are indicated as $T_{{\rm E}_{1,2}}$ -- corresponding to the first and second time where $\rho_{\Phi} = \rho_{\rm R}$,  and $T_{{\rm D}_{1,2}}$ -- which correspond to the start and end of entropy injection (for more precise definition see~\cite{Arias:2020qty}). Regarding the axion, this cosmological scenario alters it evolution, since both the Hubble parameter and the entropy of the plasma change significantly. This results in a shift in the oscillation temperature, and the dilution of the axion energy density.
 
 Moreover, we will use the QCD axion mass of ref.~\cite{Borsanyi:2016ksw}. For the axion mass beyond the minimum and maximum temperatures, $T_{\rm min, \, max}$, in the corresponding data file, we will take
 %
 \begin{equation}
 	\maT = \Bigg\{
 	\begin{matrix}
 		\dfrac{\chi(T_{\rm min})}{\fa^2} & \text{for } T<T_{\rm min} 
 		\\ \\
 		\dfrac{\chi(T_{\rm max})}{\fa^2}   \lrb{\dfrac{T}{T_{\rm max}}}^{-8.16} & \text{for } T>T_{\rm max} 
 	\end{matrix} \;.
 	\label{eq:axM-limits}
 \end{equation}
 
 
 \subsubsection{complete example in \CPP}\label{sec:cpp_example}
 %
 In order to write a \CPP program that uses \mimes in order to solve the EOM~\ref{eq:eom_sys}, we must include the header files {\tt src/AxionMass/AxionMss.hpp} and {\tt src/Axion/AxionSolve.hpp}. In the example at hand, the {\tt main} function should contain the definition of the axion mass
 %
 \begin{cpp}
 	// use chi_PATH to interpolate the axion mass.
 	mimes::AxionMass<LD> axionMass(chi_PATH,0,mimes::Cosmo<LD>::mP);
 
 	/* This is the axion mass squared beyond the interpolation limits 
 	for the current data. If you don't specify them, the axion mass 
 	is taken to be constant beyond these limits */
 
 	/*set ma2 for T>TMax*/
 	long double TMax=axionMass.getTMax();    
 	long double chiMax=axionMass.getChiMax();    
 	axionMass.set_ma2_MAX(
 		[&chiMax,&TMax](long double T, long double fa){
 			return chiMax/fa/fa*std::pow(T/TMax,-8.16);
 		}
 	);  
 	
 	/*set ma2 for T<TMin*/
 	long double TMin=axionMass.getTMin();  
 	long double chiMin=axionMass.getChiMin();    
 	axionMass.set_ma2_MIN( 
 		[&chiMin,&TMin](long double T, long double fa){
 			return chiMin/fa/fa;
 		}
 	);
 \end{cpp}
 %
 It should be noted that, since we have used {\tt chi\_PATH}, we need to include the {\tt path.hpp} header file from {\tt \mimes/src/misc\_dir}. Moreover, we note that we have used as parameter \cppin{mimes::Cosmo<LD>::mP}, which is the Planck  mass, as a static member of the {\tt Cosmo} class (described in Appendix~\ref{app:classes}), in order to interpolate the mass up to $T=m_P$, \ie the entire possible mass range.
 
 Following, the axion EOM can be solve as
 %
 \begin{cpp}
  	std::string inputFile = std::string(rootDir)+
     				std::string("/UserSpace/InputExamples/MatterInput.dat");
 	
 	mimes::Axion<long double, 1, RODASPR2<long double> > 
 	ax(0.1, 1e16, 500, 1e-4, 1e3, 15, 1e-2, inputFile, &axionMAss, 
 	1e-1, 1e-8, 1e-1, 1e-11, 1e-11, 0.9, 1.2, 0.8, int(1e7));
 	
 	ax.solveAxion();
 \end{cpp}
 %
 This program should solve the axion EOM. However, one needs to print anything that they need. For example, the relic abundance is printed by adding {\tt std::cout<<ax.relic<<"\\n";} after {\tt ax.solveAxion();}. It should be noted that, in order to make sure that the code is compiled successfully, one should add at the top of the file any other header they need. In particular, in this case, we need to add {\tt \#include<iostream>}, in order to be able to use {\tt std::cout}.
 
 
 \paragraph{Parameter choice}
 %
 It should be noted that the string that is assigned to the variable {\tt inputFile}, is the path of file that exists in {\tt \mimes/UserSpace/InputExamples/MatterInput.dat}. Also, {\tt rootDir} is a constant static ({\tt char[]}) variable that  is defined in {\tt \mimes/src/path.hpp} that , and it is automatically generated when {\tt bash configure.sh} is executed.  Moreover, the other parameters used in the constructor are:
 %
 \begin{itemize}
 	\begin{minipage}{0.3\linewidth}
 	\item {\tt theta\_i}$=0.1$
 	\item {\tt fa}$=10^{16}~\GeV$
 	\item {\tt umax }$=500$
 	\item {\tt TSTOP}$=10^{-4}~\GeV$
 	\item {\tt ratio\_ini}$=10^3$
 	\item  {\tt N\_convergence\_max}$=15$
 	\end{minipage}
 	\begin{minipage}{0.35\linewidth}
 	\item  {\tt convergence\_lim}$=10^{-2}$
 	\item {\tt initial\_stepsize}$=10^{-1}$ 
 	\item {\tt maximum\_stepsize}$=10^{-1}$ 
 	\item {\tt minimum\_stepsize}$=10^{-8}$
 	\item {\tt absolute\_tolerance}$=10^{-11}$
 	\item {\tt relative\_tolerance}$=10^{-11}$
 	\end{minipage}
 	\begin{minipage}{0.3\linewidth}
 	\item {\tt beta}$=0.9$
 	\item {\tt fac\_max}$=1.2$
 	\item {\tt fac\_min}$=0.8$
 	\item {\tt maximum\_No\_steps}$=10^7$
 	\item[]{\vfill}	\item[]{\vfill}	\item[]{\vfill}
 	\end{minipage}
 \end{itemize}
 
 \paragraph{Template arguments}
 %
 Furthermore, The first template parameter is {\tt long double}, which means that all the numeric types (apart from integers like {\tt maximum\_No\_steps}) have ``long double" precision. which is useful when considering low tolerances as in this example. The second and third template parameters are responsible for the Runge-Kutta method we use. That is, n this case, the second template parameter -- $1$ -- means that we use a Rosenbrock method, with the method being {\tt RODASPR2}. Notice that the method also needs a template parameter that declares all member variables of the {\tt RODASPR2} class.
 
 \paragraph{Compilation}
 %
 Assuming that we have already executed {\tt bash configure.sh}, compilation of the above example is trivial. Assuming that we name the file that contains the code {\tt axionExample.cpp}, and it is located in {\tt \mimes/UserSpace}, the only thing we need to do is run
 %
 \begin{pseudo}
 	g++ -O3 -std=c++17 -lm -I../ -o axion axionExample.cpp
 \end{pseudo}
 %
 or 
 %
 \begin{pseudo}
 	clang -lstdc++ -O3 -std=c++17 -lm -I../ -o axion axionExample.cpp
 \end{pseudo}
 
 Both of these commands should create an executable that solves the axion EOM. That is, assuming we have a terminal open in   {\tt \mimes/UserSpace}, we can run {\tt ./axion}, and get the results we chose. It should be noted that all paths that \mimes uses by default are written as absolute paths when we run {bash ./configure.sh}. Therefore, if the {\tt inputFile} variable is also an absolute path, the executable can be copied and used in any other directory of the same system. However, it should be preferred that executables are kept under the \mimes directory, in order to be able to compile them with different data file paths if needed.
 
 
 
 \paragraph{The entire code}
 %
 This example consists of only a few lines of code, which, including the change in the axion mass beyond the interpolation limit, is
 %
 \begin{cpp}
 	#include<iostream>
 	//include everything you need from \mimes
 	#include"src/Axion/AxionSolve.hpp"
 	#include"src/AxionMass/AxionMass.hpp"
 	#include"src/Cosmo/Cosmo.hpp"
 	#include"src/misc_dir/path.hpp"
 	
 	int main(){		
 		// use chi_PATH to interpolate the axion mass.
 		mimes::AxionMass<LD> axionMass(chi_PATH,0,mimes::Cosmo<LD>::mP);
 		
 		/*set ?$\maT^2$? for ?$T\geq T_{\rm max}$?*/
 		long double TMax=axionMass.getTMax();    
 		long double chiMax=axionMass.getChiMax();    
 		axionMass.set_ma2_MAX(
 			[&chiMax,&TMax](long double T, long double fa){
 				return chiMax/fa/fa*std::pow(T/TMax,-8.16);
 			}
 		);  
 		
 		/*set ?$\maT^2$? for ?$T\leq T_{\rm min}$?*/
 		long double TMin=axionMass.getTMin();  
 		long double chiMin=axionMass.getChiMin();    
 		axionMass.set_ma2_MIN( 
 			[&chiMin,&TMin](long double T, long double fa){
 				return chiMin/fa/fa;
 			}
 		);		
 
 		std::string inputFile = std::string(rootDir)+
 			std::string("/UserSpace/InputExamples/MatterInput.dat");
 		
 		mimes::Axion<long double, 1, RODASPR2<long double> > 
 		ax(0.1, 1e16, 500, 1e-4, 1e3, 15, 1e-2, inputFile, &axionMass, 
 		1e-1, 1e-8, 1e-1, 1e-11, 1e-11, 0.9, 1.2, 0.8, int(1e7));
 		
 		ax.solveAxion();
 		
 		std::cout<<ax.relic<<"\n";
 		
 		return 0;
 	}
 \end{cpp}
 %
 
 %
 We should point out that another example is given in {\tt \mimes/UserSpace/Cpp/Axion}, where all parameters are taken as command-line inputs, the various compilation-time options can be given {\tt \mimes/UserSpace/Cpp/Axion/Definitions.mk}, and then compiled using {\tt make}. Therefore, the user can just modify this code in order to meet their needs. That is, using this example, one only needs to add their preferred definition of {\tt ma2\_MAX} and {\tt ma2\_MIN} -- or change the {\tt axionMass} variable (in {\tt \mimes/src/static.hpp}) using a function as mentioned in section~\ref{sec:compile_time_input} without writing the entire program themselves.
 
 \paragraph{Alternative axion mass definition}
 %
 For this particular example, we could have used the approximate definition of the axion mass
 %
 \begin{equation}
 	\maT^2 \approx \ma^2 \times \Bigg\{ 
 	\begin{matrix}
 	\lrb{\dfrac{T_{\rm QCD}}{T}}^{8.16} & \quad \text{for } T\geq T_{\rm QCD} 
 	\\ \\
 	1 & \quad \text{for } T\leq T_{\rm QCD}
 \end{matrix} \;,
 	\label{eq:axM_approx}
 \end{equation}
 %
 where $T_{\rm QCD}  \approx 150~\MeV$ and $\ma=\dfrac{3.2 \times 10^{-5} ~\GeV^4}{\fa^2}$. This can be done by substituting the {\tt axionMass} definition (lines $9$-$28$) with 
 %
 \begin{cpp}
 	auto ma2 = [](long double T,long double fa){
 	    long double TQCD=150*1e-3;
 	    long double ma20=3.1575e-05/fa/fa;
 	    if(T<=TQCD){return ma20;}
 	    else{return ma20*std::pow((TQCD/T),8.16);}
 	};
 	mimes::AxionMass<LD> axionMass(ma2);
 \end{cpp}
 
 
 \subsubsection{complete example in \PY}
 %
 In order to be able use the {\tt AxionMass} and {\tt Axion} classes in \PY, we need to import the corresponding module from {\tt \mimes/src/interfacePy}. That is, assuming that the script from which we intend to import {\tt \mimes/src/interfacePy/Axion/Axion.py} is in {\tt \mimes/UserSpace}, we can import the {\tt Axion} class by writing the following
 %
 \begin{py}
 	#add the relative path for MiMeS/src
 	from sys import path as sysPath
 	from os import path as osPath
 	sysPath.append(osPath.join(osPath.dirname(__file__), '../src'))
 	
 
 	from interfacePy.AxionMass import AxionMass #import the AxionMass class
 	from interfacePy.Axion import Axion #import the Axion class
 	from interfacePy.Cosmo import mP #import the Planck mass
 \end{py} 
 %
 Once everything we need is imported, we can simply follow the steps outlined in section~\ref{sec:begin_py}. For the example at hand, we can create an instance of the {\tt AxionMAss} class as
 %
 \begin{py}
 	# AxionMass instance
 	axionMass = AxionMass(r'../src/data/chi.dat',0,mP)
 	
 	# This is the axion mass squared beyond the interpolation limits for the current data. 
 	# If you don't specify them, the axion mass is taken to be constant beyond these limits
 	TMax=axionMass.getTMax() 
 	chiMax=axionMass.getChiMax()
 	TMin=axionMass.getTMin() 
 	chiMin=axionMass.getChiMin()
 	
 	axionMass.set_ma2_MAX( lambda T,fa: chiMax/fa/fa*pow(T/TMax,-8.16) )
 	axionMass.set_ma2_MIN( lambda T,fa: chiMin/fa/fa )
 \end{py}
 %
 
 Then we can simply create an instance of the {\tt Axion} class as 
 %
 \begin{py}
 	#in python it is more convenient to use relative paths
 	inputFile = inputFile="./InputExamples/MatterInput.dat"  
 	
 	ax = Axion(0.1, 1e16, 500, 1e-4, 1e3, 15, 1e-2, inputFile, axionMass, 
 	1e-1, 1e-8, 1e-1, 1e-11, 1e-11, 0.9, 1.2, 0.8, int(1e7))
 \end{py}
 %
 Again, the parameters for the constructor are the same as in the \CPP example. The axion EOM, then, is solved using
 %
 \begin{py}
 	ax.solveAxion()
 \end{py}
 %
 In contrast to the \CPP usage of this function, this only stores the $\thetai$, $\fa$, $\thetaosc$, $\Tosc$, and $\Omega h^2$ in the variables {\tt ax.theta\_i}, {\tt ax.fa}, {\tt ax.theta\_osc}, {\tt ax.T\_osc}, and {\tt ax.relic}; respectively. Therefore, we can print the relic abundance by calling \run{print(ax.relic)}. In order to get the integration points (\ie the evolution of the angle and the other quantities), the quantities at the  peaks, and the local integration errors, we need to call
 %
 \begin{py}
 	#this gives you all the points of integration
 	ax.getPoints()
 	
 	#this gives you the peaks of the oscillation
 	ax.getPeaks()
 	
 	#this gives you local integration errors
 	ax.getErrors()
 \end{py}
 %
 The documentation of any \PY function can be read directly inside the script using {\tt ?} as a prefix. For example, in order to see what the functionality and usage of the {\tt getPeaks} function, we can call \run{?ax.getPeaks}, and its documentation will be printed. 
 
 As already mentioned, it is important to always delete any instance of the {\tt Axion} class that is not going to be used. In this case this is done by calling
 %
 \begin{py}
 	del ax
 \end{py}
 
 
 \paragraph{Compilation of the shared library}
 %
 As described in section~\ref{sec:compile_time_input}, we may need to change the default data file paths, or the various compilation options. This is done through the variables in {\tt \mimes/Definitions.mk} and {\tt \mimes/Paths.mk} described in section~\ref{sec:options}. 
 
 Once we have chosen everything according to our needs, the library can be created by opening a terminal inside the root directory of \mimes and running
 %
 \begin{py}
 	bash configure.sh
 	make lib/Axion_py.so
 \end{py}
 %
 
 \paragraph{The entire code}
 %
 As in the \CPP example, this example consists of only a few lines of code. The script we described here is
 %
 \begin{py}
 	#add the relative path for MiMeS/src
 	from sys import path as sysPath
 	from os import path as osPath
 	sysPath.append(osPath.join(osPath.dirname(__file__), '../src'))
 	
 	from interfacePy.AxionMass import AxionMass #import the AxionMass class
 	from interfacePy.Axion import Axion #import the Axion class
 	from interfacePy.Cosmo import mP #import the Planck mass
 
 	# AxionMass instance
 	axionMass = AxionMass(r'../src/data/chi.dat',0,mP)
 	
 	# define ?$\maT^2$? for ?$T\leq T_{\rm min}$?
 	TMin=axionMass.getTMin() 
 	chiMin=axionMass.getChiMin()
 	axionMass.set_ma2_MIN( lambda T,fa: chiMin/fa/fa )
 
 	# define ?$\maT^2$? for ?$T\geq T_{\rm max}$?
 	TMax=axionMass.getTMax() 
 	chiMax=axionMass.getChiMax()
 	axionMass.set_ma2_MAX( lambda T,fa: chiMax/fa/fa*pow(TMax/T),8.16) )
 	
 	#in python it is more convenient to use relative paths
 	inputFile = inputFile="./InputExamples/MatterInput.dat"  
 	
 	ax = Axion(0.1, 1e16, 500, 1e-4, 1e3, 15, 1e-2, inputFile, axionMass, 
 	1e-1, 1e-8, 1e-1, 1e-11, 1e-11, 0.9, 1.2, 0.8, int(1e7))
 
 	ax.solveAxion()
 	
 	print(ax.relic)
 	
 	#once we are done we should run the destructor
 	del ax
 \end{py} 
 
 
 One can find a complete example, including the option to create several plots, in the script {\tt \mimes/UserSpace/Python/Axion.py}. Also, the same example can be used interactively, in jupyter notebook environment, that can be found in {\tt \mimes/UserSpace/JupyterNotebooks/Axion.ipynb}. One can read the comments, and change all different parameters, in order to examine how the results are affected. 
 
 \paragraph{Alternative axion mass definition}
 %
 In order to use the approximate axion mass as defined in \eqs{eq:axM_approx}, we could replace the definition of the {\tt axionMass} variable (lines $10$-$21$) with
 %
 \begin{py}
 	def ma2(T,fa):
 		TQCD=150*1e-3
 		ma20=3.1575e-05/fa/fa
 		if T<=TQCD:
 			return ma20;
 		return ma20*(TQCD/T)**8.16
 	axionMass = AxionMass(ma2)
 \end{py}
 
 \subsubsection{Results}
 %
 In both \CPP and \PY one should get the same results (if the same numeric type, and RK parameters and method are used), which should be similar to \Figs{fig:theta_evolution-EMD}. 
 %
 It should be noted that this particular example is chosen such that the axion relic abundance is close to its observed value~\cite{Planck:2018vyg} ($\Omega h^2 \approx 0.12$), and is indicative of the features one can expect if entropy is injected to the plasma for $T \ll \Tosc$.
 %
 \begin{figure}[h]
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/theta_evolution-EMD.pdf}
 		\caption{}
 		\label{fig:theta_evolution-EMD}
 	\end{subfigure}
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/theta_evolution-RD.pdf}
 		\caption{}
 		\label{fig:theta_evolution-RD}
 	\end{subfigure}
 	\caption{...}
 	\label{fig:results}
 \end{figure}
 %
 In \Figs{fig:theta_evolution-EMD} we show the evolution  of $\theta$ for temperatures $T \in [0.025,0.15]~\GeV$, where the vertical line indicates $\Tosc$, while the horizontal one the corresponding value of $\theta$, $\thetaosc$. Also, the blue curve connects the peaks of the oscillation. For comparison, in \Figs{fig:theta_evolution-RD} we also show the evolution of the angle in a radiation dominated Universe with constant entropy (\ie standard cosmological scenario). From these figures, we can see that the effect of an early matter domination reduces the amplitude of oscillation -- due to the injection of entropy -- as well the oscillation temperature -- due to the increase of the Hubble parameter. Furthermore, as expected from \eqs{eq:theta_osc}, the angle at $T=\Tosc$ in the EMD scenario is much smaller that the corresponding value in the standard cosmological case. This shows again that approximating $\thetaosc \approx \thetai$ results in an inaccurate result, especially in cases where entropy is injected to the plasma (as expected by \eqs{eq:theta_osc}).
 
 %
 Moreover, in \Figs{fig:RK_response}, we show the relative local error of integration as well as a histogram of the number of integration steps for $0.025~\GeV < T < 0.15 ~\GeV$.
 %
 \begin{figure}[h]
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/local_errors-EMD.pdf}
 		\caption{}
 		\label{fig:local_errors-EMD}
 	\end{subfigure}
 	\begin{subfigure}[]{0.5\textwidth}
 		\includegraphics[width=1\textwidth]{figs/histogram-EMD.pdf}
 		\caption{}
 		\label{fig:histogram-EMD}
 	\end{subfigure}
 	\caption{...}
 	\label{fig:RK_response}
 \end{figure}
 %
 The local relative errors, defined in Appendix~\ref{app:RK}, are shown in \Figs{fig:local_errors-EMD}. The black and red lines correspond to $\delta \theta/\theta$ and $\delta \zeta/\zeta$, respectively. This figure indicates that the local errors are relatively well behaved for $T>\Tosc$, and they only start to oscillate violently once the oscillations start. However, the adaptation of the integration step seems to work, as the errors are kept below $\sim 10^{-7}$ (the relative error of $zeta$ is large because $\zeta \approx 0$ before the $\Tosc$).  In order to examine how the step-size is adapted to the difficulty of the problem, we also show a histogram which show how many integration steps are taken for fixed temperature intervals.~\footnote{More precisely, the histogram is made by dividing the the temperature range $0.025~\GeV\lesssim T\lesssim 0.15~\GeV$ to $30$ bins of equal size.} 
 %
 In this figure, we see that the number of integration steps increases rapidly for temperatures below the oscillation temperature. 
 %
 This is expected, since integration becomes more difficult as the frequency of the oscillation increases -- $\maT/H$ increases rapidly for $T<\Tosc$. That is, the local integration error tends to increase. Thus, in order to reduce the local error, the embedded RK method we employ, reduces the step-size. This means that for $T \lesssim \Tosc$ the number of integration steps increase drastically.
 %
 This is the general picture of how adaptation happens. However, one has to experiment with all the available parameters, in order to solve the axion EOM as accurately and fast as possible.  
 
 
 
 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
 % APPENDICES
 \pagebreak
 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
 \setcounter{section}{0}
 \section*{Appendix}
 \appendix
 
 \renewcommand{\theequation}{\Alph{section}.\arabic{equation}}
 \setcounter{equation}{0}  % reset counter
 %%%%%%%%%%%%%%%%%%%%%%%
 
 \section{Basics of embedded Runge-Kutta Mehtods}\label{app:RK}
 \setcounter{equation}{0}
 %
 Runge-Kutta (RK) methods are employed in order to solve an ordinary differential equation (ODE), or a system of ODEs of first order.~\footnote{Boundary value problems, and higher order differential equations are expressed as first order ODEs, and then solved. Similarly to  \eqs{eq:eom_sys}.}   Although there are some very insightful sources in the literature (\eg~\cite{Hairer,hairer2010solving,10.5555/1403886}) we give a brief overview of them in order to help the user to make appropriate decisions when using \mimes.
 
 The general form of a system of first order of ODEs is
 %
 \begin{equation}
 	\dfrac{d\vec{y}}{dt}=\vec{f}(\vec{y},t) \;,
 	\label{eq:ODE_definition}
 \end{equation}
 %
 with given initial condition $\vec{y}(0)$. Also, the components of $\vec{y}$ denote the unknown functions. Note that we can always shift $t$ to start at $0$, which simplifies the notation. In order to solve the the system of \eqs{eq:ODE_definition}, an RK method uses an iteration of the form 
 %
 \begin{equation}
 	\vec{y}_{n+1}=\vec{y}_{n}+ h\sum_{i=1}^{s} b_i \ \vec{k}_i \;,
 	\label{eq:RK_iter}
 \end{equation}
 %
 with $n$ denoting the iteration number, $h$ the ``step-size" that is used to progress $t$; $t_{n+1}=t_{n}+h$. Moreover, $s$, $b_i$ and $\vec{k}_i$ define the corresponding RK method. For example,
 the classic Euler method is an RK method with $s=1$, $b=1$, and $\vec{k}_1 = \vec{f}(\vec{y_n , t_n})$. Methods with $\vec{k}$ that depends on previous step (\ie $\vec{k} =\vec{k}\lrsb{y_{n},t_{n}}$), are called explicit, while the ones that try to also predict next step (\ie $\vec{k} =\vec{k}\lrsb{y_{n},t_{n}; y_{n+1},t_{n+1}}$) are called implicit.~\footnote{Generally, by substituting  implicit methods $y_{n+1}$ as in \eqs{eq:RK_iter}, we end up with a system of equations that need to be solved in order to compute $\vec{k}$.} 
 
 \subsection{Embedded RK methods}
 %
 A large category of RK methods are  the so-called embedded RK methods. These methods make two estimates for the same step simultaneously -- without evaluating  $\vec{k}$ many times within the same iteration. Therefore, together with the iteration of \eqs{eq:RK_iter}, a second estimate is given by
 %
 \begin{equation}
 	\vec{y}_{n+1}^{(*)}=\vec{y}_{n}+ h\sum_{i=1}^{s} b_i^{(*)} \vec{k}_i \;,
 	\label{eq:RK_Embedded_iter}
 \end{equation}
 %
 with $b_i^{(*)}$ is an extra parameter that characterise the ``embedded" method of different order (typically, one order higher that the estimate~\ref{eq:RK_iter}). The local (for the step $t_{n+1}$) error divided by the scale of the solution, then, estimated as
 %
 \begin{equation}
 	\Delta \equiv  \sqrt{\frac{1}{N}\sum_{d=1}^{N}\lrb{\frac{y_{n+1 ,d}-y^{*}_{n+1,d}}{ \Lambda_d }}^{2}} \;,
 	\label{eq:error_estimate}
 \end{equation}
 %
 where $n$ the iteration number, $N$ the number of ODEs, $d$ the component of $\vec{y}$, and $\Lambda_d$ defined as 
 %
 \begin{equation}
 	\Lambda_d = {\tt Atol} + \max\lrb{|y_{n+1, d}|,|y^{*}_{n+1 ,d}|} \; {\tt Rtol} \;,
 	\label{eq:RK_scale}
 \end{equation}
 %
 with {\tt Atol} and {\tt Rtol} the absolute and relative tolerances that characterise the desirable accuracy we want to achieve; user defined values, typically {\tt Atol}$=${\tt Rtol}$\ll 1$.  With these definitions, the desirable error is reached when  $\Delta \lesssim 1$. 
 
 \paragraph{Step-control} The definition~\ref{eq:error_estimate}, allows us to adjust the step-size $h$ in such way that $\Delta \approx 1$. That is, we take trial steps, and $h$ is adapted until $\Delta \approx 1$.  A simple adaptive strategy adjusts step-size, using
 %
 \begin{eqnarray}
 	h \to \beta \; h \;  {\rm max} \lrsb{f_{\rm min}, {\rm min} \lrb{\Delta^{-\frac{1}{p+1}}, f_{\rm max}}} \;,
 	\label{eq:step-control}
 \end{eqnarray}
 %
 with $p$ the order of the RK method ($p+1$ is the order of the embedded one), $\beta$  a bias factor of the adaptive strategy (typically $\beta$ is close but below $1$), used to adjust the tendency of $h$ to be somewhat smaller than what the step-control predicts. Also, $f_{min}$ and $f_{max}$ are the minimum and maximum allowed factors, respectively, that can multiply $\beta \ h$, used in order to avoid large fluctuations that can destabilise the process. All these parameters are chosen by the user, in order to make the step-control process as aggressive or safe as needed. 
 
 \paragraph{Correspondence between \mimes parameters and RK ones}
 %
 The various parameters that \mimes are described in section~\ref{sec:input}. The correspondence between them and the RK parameters is given in table~\ref{tab:RK_mimes_params}.
 %
 \begin{table}[t!]
 	\centering
 	\begin{tabular}{|c|c|}
 		\mimes & Runge-Kutta \\
 		\hline
 		{\tt absolute\_tolerance } & {\tt Atol}  \\
 		\hline
 		{\tt relative\_tolerance } & {\tt Rtol}  \\
 		\hline
 		{\tt b} & $\beta$  \\
 		\hline
 		{\tt fac\_min} & $f_{\rm min}$  \\
 		\hline
 		{\tt fac\_max} & $f_{\rm max}$  \\
 		\hline
 	\end{tabular}
 	\caption{Correspondence between the user \mimes user input and the RK parameters described in the text.}
 	\label{tab:RK_mimes_params}
 \end{table}
 
 \subsection{Explicit embedded RK methods}
 %
 Explicit methods use only the information of the previous step in order to compute $\vec{k}_i$ from  
 %
 \begin{equation}
 	\vec{k}_{i}=\vec f\lrBiggb{ \vec{y}_{n}+h \lrBigb{ \sum_{j=1}^{i-1}a_{ij}\vec{k}_{j} }, t_{n}+ c_{i} \; h}\;, \quad \forall i=1,2,\dots, s\;,
 	\label{eq:explicit_RK_k}
 \end{equation}
 %
 with $a_{ij}$, $c_i$, together with $b_i$ and $b_i^{(*)}$, consist the so-called Butcher tableau of teh corresponding method. For explicit methods this is usually presented as 
 %
 \begin{equation}
 	\begin{array}{c|cccccc}
 		0      & 0      &   0   &      0& \dots & 0& 0\\
 		c_2    & a_{21} &   0   &   0   & \dots & 0& 0\\
 		c_3    & a_{31} & a_{32}&      0& \dots & 0& 0\\
 		\vdots & \vdots & \vdots& \vdots&\ddots &\ddots& \vdots\\
 		c_s    & a_{s1} & a_{s2}& a_{s3}& \dots & a_{s s-1}& 0 \\
 		\hline
 		p      & b_1    & b_2  & b_3 & \dots & b_{s-1} & b_s \\
 		p+1      & b_1^{\star}    & b_2^{\star}  & b_3^{\star} & \dots & b_{s-1}^{\star} & b_s^{\star}
 	\end{array}
 	\label{eq:Butcher}
 \end{equation}
 %
 It should be noted that $c_i = \displaystyle\sum_{j=1}^{s} a_{ij} $.
 
 \subsection{Rosenbrock methods}
 %
 Explicit RK methods, encounter instabilities when a system is ``stiff"~\footnote{A definition of stiffness can be found in~\cite{hairer2010solving,Hairer}.}; when is oscillates rapidly, or has different elements at different scales. This problems are somewhat resolved by trying to predict the next step inside $\vec{k}$; \ie in implicit methods. However, then one has to solve a non-linear set of equations in order to compute $\vec{y}_{n+1}$, which is generally a slow process, as the Newton method (or some variation) needs to be applied. However, there exist another way, a compromise between explicit and implicit methods.  Linearly implicit RK methods, usually called Rosenbrock methods -- with popular improvements as Rosenbrock-Wanner methods, introduce  parameters in the diagonal of the Butcher tableau~\ref{eq:Butcher} and linearise the system of  non-linear equations (for details, see~\cite{hairer2010solving}). In these methods, $\vec{k}$ is determined by  
 %
 \begin{equation}
 	\left(\hat I - \gamma h \hat{J}\right)\cdot \vec{k}_{i}=
 	h \vec{f}\Big(\vec{y}_n+\sum_{j=1}^{i-1}a_{ij}\vec{k}_{j},t_n + c_{i}h \Big)+
 	h^2 \left(\gamma + \sum_{j=1}^{i-1}\gamma_{ij}\right)\dfrac{\partial \vec{f} }{\partial t}+
 	h \hat J \cdot \sum_{j=1}^{i-1}\gamma_{ij} \vec{k}_{j}\; ,
 	\label{eq:Ros_k}
 \end{equation}
 %
 which is written in such a way that everything is evaluated at $t = t_{n}$. In \eqs{eq:Ros_k},  $\hat J = \dfrac{\partial \vec f}{ \partial \vec y}$ the Jacobian of the system of ODEs, $\hat I$ the unit matrix with dimension equal to the number of ODEs. Moreover, $\gamma$ and $\gamma_{ij}$ are parameters that characterise the method (along with $a_{ij}$, $c_i$,  $b_i$, and $b_i^{(*)}$).
 
 \paragraph{Implementing a new Butcher tableau in {\tt NaBBODES}} As already mentioned, \mimes uses {\tt NaBBODES}, which supports the implementation of new Butcher tableaux. This is done by adding a new class (or struct) inside the header file {\tt METHOD.hpp} that can be found in {\tt \mimes/src/NaBBODES/RKF} for the explicit RK and  {\tt \mimes/src/NaBBODES/Rosenbrock} for the Rosenbrock embedded methods. All the new parameters must be {\tt public},  {\tt constexpr static} variables, of a type that is the template parameter of the method. For example, the Heun-Euler method can be implemented, adding the following code in  {\tt \mimes/src/NaBBODES/RKF/METHOD.hpp}
 %
 \begin{cpp}
 	/*-----Implementation of the Heun-Euler method-----*/
 	/*-----It shouldn't be used for MiMeS, since it is likely to fail-----*/
 	
 	//LD is the numeric type (\ie double, or long double).
 	template<class LD>
 	//use struct because its variables are public by default.
 	struct HeunEuler{  
 		static constexpr unsigned int s=2; // HeunEuler is a 2-stage RK
 		static constexpr unsigned int p=1; // first order, with embedded ?$p+1=2$?
 
 		//these are aliases for the arrays
 		using arr=std::array<LD,s>;
 		using arr2=std::array<std::array<LD,s>,s>;
 		
 		static constexpr arr b={0.5,0.5}; // this is ?$b_i$?
 		static constexpr arr bstar={0.5,0.5}; // this is ?$b_i^{(*)}$?
 		static constexpr arr c={0,1}; // remember that ?$c_i = \displaystyle\sum_{j=1}^{s} a_{ij}$?
 		
 		// this is ?$a_{ij}$?
 		static constexpr arr2 a={
 			arr{0,0},
 			arr{1,0}
 		};			
 	};
 \end{cpp}
 
 In order to implement the ROS3w~\cite{RangAngermann2005} method, one can add the following code in the header file {\tt \mimes/src/NaBBODES/Rosenbrock/METHOD.hpp}
 %
 \begin{cpp}
 	/*-----Implementation of the ROS3w method-----*/
 	/*-----It shouldn't be used for MiMeS, since it is likely to fail-----*/
 	
 	
 	//LD is the numeric type (?\ie? double, or long double).
 	template<class LD>
 	struct ROS3w{
 		static constexpr unsigned int s=3; // 3-stage
 		static constexpr unsigned int p=2; // second order (embedded order is ?$p+1$?)
 		
 		//aliases for the arrays
 		using arr=std::array<LD,s>;
 		using arr2=std::array<std::array<LD,s>,s>;
 
 		static constexpr arr b={0.25,0.25,0.5 };  // this is ?$b_i$?
 		// this is ?$b_i^{(*)}$?
 		static constexpr arr bstar={ 0.746704703274,0.1144064078371,0.1388888888888};  
 		
 		static constexpr arr c={0,2/3.,4/3.}; // remember that ?$c_i = \displaystyle\sum_{j=1}^{s} a_{ij}$?
 		static constexpr LD gamma=0.4358665215084; // this is ?$\gamma$?
 		
 		// this is ?$a_{ij}$?
 		static constexpr arr2 a={
 			arr{0,0,0},
 			arr{2/3.,0,0},
 			arr{2/3.,2/3.,0}	
 		};
 	
 		// this is ?$\gamma_{ij}$?	
 		static constexpr arr2 g={
 			arr{0,0,0},
 			arr{0.3635068368900,0,0},
 			arr{-0.8996866791992,-0.1537997822626,0}
 		};
 	};
 \end{cpp}
 
 \paragraph{How to compile \mimes in order to use the newly implemented method} One a new Butcher tableau is implemented, the \cppin{mimes::Axion} class can use it. This class, needs the name of method assigned to the macro {\tt METHOD}. That is, before you include the header file {\tt src/Axion/AxionSolve.hpp}, this macro needs to be defined. A convenient way to do this, is to define the macro using the {\tt -D} flag of the compiler; \eg in order to use the ROS3w method, this definition ca be done by adding  {\tt -DMETHOD=ROS3w} when compiling the code. Alternatively, if one uses the {\tt makefile} files, a method can be chosen by adding it in the corresponding {\tt Definitions.mk} file as {\tt METHOD=ROS3w}.~\footnote{One make sure to use the correct {\tt SOLVER} macro -- {\tt SOLVER=1} for Rosenbrock method  and {\tt SOLVER=2} for explicit RK method-- otherwise the compilation will fail.} 
 
 
 \section{\CPP classes}\label{app:classes}
 \setcounter{equation}{0}
 %
 \mimes is designed as an object-oriented header-only library. That is, all the basic components of the library are defined in classes in header files. All the classes relevant to the use of \mimes are under the namespace {\tt mimies}.
 
 \subsection{{\tt Cosmo} class}
 %
 The {\tt Cosmo<LD>} class is responsible for interpolation of the various quantities of the plasma. Its header file is {\tt \mimes.src/Cosmo/Cosmo.hpp}, and needs to be included in order to use this class. The template parameter {\tt LD} is the numeric type that will be used, \eg {\tt double}. The constructor of this class is
 %
 \begin{cpp}
 	template<class LD>
 	mimes::Cosmo<LD>(std::string cosmo_PATH, LD minT=0, LD maxT=mimes::Cosmo<LD>::mP)
 \end{cpp}
 %
 The argument {\tt cosmo\_PATH} is the path of the data file that contains $T$ (in $\GeV$), $\heff$, $\geff$, with increasing $T$. The parameters {\tt minT} and {\tt maxT} are minimum and maximum interpolation temperatures. These temperatures are just limits, and the action interpolation is done between the closest temperatures in the data file. Moreover, beyond the interpolation temperatures, both $\heff$ and $\geff$ are assumed to be constants.
 
 Interpolation of the RDOF, allows us to define various quantities related to the plasma; \eg the entropy density is defined as $s = \dfrac{2\pi^2}{45} \heff T^3$. These quantities are given as the member functions:
 %
 \begin{itemize}
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::heff(LD T)}: $\heff$ as a function of $T$.
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::geff(LD T)}: $\geff$ as a function of $T$.
 	% 
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::dheffdT(LD T)}: $d\heff/dT$ as a function of $T$.
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::dgeffdT(LD T)}: $d\geff/dT$ as a function of $T$.
 	%
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::dh(LD T)}: $\delta_h = 1 + \frac{1}{3} \frac{d\log \heff}{d\log T}$ as a function of $T$.
 	%
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::s(LD T)}: The entropy density of the plasma as a function of $T$.
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::rhoR(LD T)}: The energy density of the plasma as a function of $T$.
 	\item \cppin{ template<class LD> LD mimes::Cosmo<LD>::Hubble(LD T)}: The Hubble parameter assuming radiation dominated expansion as a function of $T$.
 \end{itemize}
 
 Moreover, there are several cosmological quantities are given as members variables:
 %
 \begin{itemize}
 	\item \cppin{template<class LD> constexpr static LD mimes::Cosmo<LD>::T0}: CMB temperature today~\cite{Zyla:2020zbs} in $\GeV$.
 	\item \cppin{ template<class LD> constexpr static LD mimes::Cosmo<LD>::h\_hub}: Dimensionless Hubble constant~\cite{Zyla:2020zbs}.
-	\item \cppin{ template<class LD> constexpr static LD mimes::Cosmo<LD>::rho\_crit}: Critical density~\cite{Zyla:2020zbs}.
+	\item \cppin{ template<class LD> constexpr static LD mimes::Cosmo<LD>::rho\_crit}: Critical density~\cite{Zyla:2020zbs} in $\GeV^3$.
 	\item \cppin{ template<class LD> constexpr static LD mimes::Cosmo<LD>::relicDM\_obs}: Central value of the measured DM relic abundance~\cite{Planck:2018vyg}.
 	\item \cppin{ template<class LD> constexpr static LD mimes::Cosmo<LD>::mP}: Planck mass~\cite{Zyla:2020zbs} in $\GeV$.
 \end{itemize}
 
 
 \subsection{{\tt AnharmonicFactor} class}
 %
 The class \cppin{mimes::AnharmonicFactor<LD>} is responsible interpolating the anharmonic factor as defined in \eqs{eq:anharmonic_f}. The corresponding header file is {\tt \mimes/src/AnharmonicFactor/AnharmonicFactor.hpp}. 
 
 The constructor of this class is
 %
 \begin{cpp}
 	template<class LD>
 	mimes::AnharmonicFactor<LD>(std::string anharmonic_PATH)
 \end{cpp}
 %
 Again, the template argument {\tt LD} is a numeric type, and the {\tt anharmonic\_PATH} string is the path of the data file with data for $\thetamax$ (which should be in increasing order) and $f(\thetamax)$.
 
 The member function that \mimes uses is the overloaded call operator
 %
 \begin{cpp}
 	template<class LD> LD mimes::AnharmonicFactor<LD>::operator()(LD theta_peak)
 \end{cpp}
 %
 This function returns the value of the anharmonic factor at $\thetamax=${\tt theta\_peak}. Although, there is no need to call this function beyond the interpolation limits (as long as the data file contains $0\leq \thetamax \leq \pi$), it is important to note that the anharmonic factor is taken to be constant beyond these limits. 
 
 
 \subsection{{\tt AxionMass} class}
 %
 The \cppin{mimes::AxionMass<LD>} class is responsible for the definition of the axion mass. The header file of this class is {\tt \mimes/src/AxionMass/AxionMass.hpp}. Its usage and member functions are described in the examples given in sections~\refs{sec:First_examples,sec:complete_examples}. However, it would be helpful to outline them here.
 
 The class has two constructors. The first one is
 %
 \begin{cpp}
 	template<class LD>
 	mimes::AxionMass<LD>(std::string chi_PATH, LD minT=0, LD maxT=mimes::Cosmo::mP)
 \end{cpp}
 %
 The first argument, {\tt chi\_PATH}, is the path to a data file that contains two columns; $T$ (in $\GeV$) and $chi$ (in $\GeV^4$), with increasing $T$. The arguments {\tt minT} and {\tt maxT} are the interpolation limits. These limits are used in order to stop the interpolation in the closest temperatures that exist in the data file. That is the actual interpolation limits are $T_{\min}\geq${\tt minT} and $T_{\max}\leq${\tt maxT}. Beyond these limits, by default, the axion mass is assumed to be constant. However, this can be changed by using the member functions
 %
 \begin{cpp}
 	void set_ma2_MIN(std::function<LD(LD,LD)> ma2_MIN)
 	void set_ma2_MAX(std::function<LD(LD,LD)> ma2_MAX)
 \end{cpp}
 %
 Here, {\tt ma2\_MIN} and {\tt ma2\_MAX} are functors that define the axion mass squared beyond the interpolation limits. In order to ensure that the axion mass is continues, usually we need $T_{\min}$, $T_{\max}$, $\chi(T_{\rm min})$, and $\chi(T_{\rm max})$. These values can be obtained using the member functions
 %
 \begin{itemize}
 	\item \cppin{template<class LD> LD mimes::AxionMass<LD>::getTMin()}: This function returns the minimum interpolation temperature, $T_{\rm min}$. 
 	\item \cppin{template<class LD> LD mimes::AxionMass<LD>::getTMax()}: This function returns the maximum interpolation temperature, $T_{\rm max}$.
 	\item \cppin{template<class LD> LD mimes::AxionMass<LD>::getChiMin()}: This function returns $\chi(T_{\rm min})$.
 	\item \cppin{template<class LD> LD mimes::AxionMass<LD>::getChiMax()}: This function returns $\chi(T_{\rm max})$.
 \end{itemize}
 
 An alternative way to define the axion mass is via the constructor
 %
 \begin{cpp}
 	template<class LD>
 	mimes::AxionMass<LD>(std::function<LD(LD,LD)> ma2)
 \end{cpp}
 %
 Here, the only argument is the axion mass squared, $\maT$, defined as a callable object.
 
 
 Once an instance of the class is defined, we can get $\maT^2$ using the member function
 %
 \begin{cpp}
 	template<class LD>	LD mimes::AxionMass<LD>::ma2(LD T, LD fa)
 \end{cpp}
 %
 We should note that {\tt ma2} is a public {\tt std::function<LD(LD,LD)>} member variable. Therefore, it can be assigned using the assignment operator. However,  in order to change its definition, we can use the following member function:
 %
 \begin{cpp}
 	template<class LD> void mimes::AxionMass<LD>::set_ma2(std::function<LD(LD,LD)> ma2)
 \end{cpp}
 
 
 \subsection{{\tt AxionEOM} class}
 %
 The \cppin{mimes::AxionEOM<LD>} class is not useful for the user. However, it is responsible for the interpolation of the underlying cosmology, and the definition of the axion EOM~\ref{eq:eom_sys}, which is passed to the ODE solver of {\tt NaBBODES}. 
 
 The constructor of the class is
 %
 \begin{cpp}
 	template<class LD>
 	mimes::AxionEOM<LD>(LD fa, LD ratio_ini, std::string inputFile, AxionMass<LD> *axionMass)
 \end{cpp}
 %
 The role of the arguments are discussed in section~\refs{sec:cpp_first_example,sec:run_time_input} as well as in table~\ref{tab:run_time-input}. One the instance is created, we the interpolations are constructed by calling the member function
 %
 \begin{cpp}
 	template<class LD> void mimes::AxionMass<LD>::makeInt()
 \end{cpp}
 %
 Then, the temperature as a function of $u=\log a/\ai$, is given via the member function 
 %
 \begin{cpp}
 	template<class LD> LD mimes::AxionMass<LD>::Temperature(LD u)
 \end{cpp}
 %
 Another useful member function is  
 %
 \begin{cpp}
 	template<class LD> LD mimes::AxionMass<LD>::logH2(LD u)
 \end{cpp}
 %
 This function returns $\log H^2$  as a function of $u$. Moreover, its derivative, $\frac{d \log H^2}{du}$, is computed using 
 %
 \begin{cpp}
 	template<class LD> LD mimes::AxionMass<LD>::dlogH2du(LD u)
 \end{cpp}
 %
 It should be noted that the highest interpolation temperature is determined by {\tt ratio\_ini} while the lower interpolation temperature is the one given in the data file {\tt inputFile}. Beyond these limits, all functions are assumed to be constant. Therefore, one should be careful, and choose an appropriate {\tt ratio\_ini}, and provide a lower temperature at which any entropy injection has stopped and the axion has reached its adiabatic evolution.
 
 
 Finally, the actual EOM is given an overloaded call operator
 %
 \begin{cpp}
 	template<class LD> 
 	void mimes::AxionMass<LD>::operator()(std::array<LD,2> &lhs, std::array<LD,2> &y, LD u)
 \end{cpp}
 %
 Here, the inputs are $u=\log a/\ai$, {\tt y[0]}$=\theta$, and {\tt y[1]}$=\zeta$; which are used to calculate the components of the EOM, with {\tt lhs[0]}$=\frac{d \theta}{d u}$ and {\tt lhs[1]}$=\frac{d \zeta}{d u}$.
 
 
 
 \subsection{{\tt Axion} class}
 %
 The \cppin{mimes::Axion<LD,Solver,Method>} class is the class that combines all the others, and actually solves the axion EOM~\ref{eq:eom_sys}. Its header file is {\tt \mimes/src/Axion/AxionSolve.hpp}, and its constructor is
 %
 \begin{cpp}
  	template<class LD, const int Solver, class Method>
 	mimes::Axion< LD, Solver, Method<LD> >(LD theta_i, LD fa, LD umax, LD TSTOP, 
 	LD ratio_ini, unsigned int N_convergence_max, LD convergence_lim, 
 	std::string inputFile, AxionMass<LD> *axionMass, LD initial_step_size=1e-2, 
 	LD minimum_step_size=1e-8, LD maximum_step_size=1e-2, LD absolute_tolerance=1e-8, 
 	LD relative_tolerance=1e-8, LD beta=0.9, LD fac_max=1.2, LD fac_min=0.8, 
 	unsigned int maximum_No_steps=10000000)
 \end{cpp}
 %
 The various arguments are discussed in section~\refs{sec:cpp_first_example,sec:run_time_input}; and outlined in table~\ref{tab:run_time-input}.
 %
 
 The member function responsible for solving the EOM is
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method>
 	void mimes::Axion< LD, Solver, Method<LD> >::solveAxion()
 \end{cpp}
 %
 Once this function finishes, the results are stored in several member variables.
 
 The quantities $a/\ai, \ T, \ \theta, \ \zeta, \rho_a$, at the integration  steps are stored in
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method> 
 	std::vector< std::vector<LD> > mimes::Axion< LD, Solver, Method<LD> >::points
 \end{cpp}
 %
 
 The quantities $a/\ai, \ T, \ \theta, \ \zeta, \rho_a, \ J$, at the peaks of the oscillation are stored in 
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method> 
 	std::vector< std::vector<LD> > mimes::Axion< LD, Solver, Method<LD> >::peaks
 \end{cpp}
 %
 Note that these points are computed using linear interpolation between two integration points with a change in teh sign of $\zeta$.
 
 
 The local integration errors for $\theta$ and $\zeta$ are stored in
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method> 
 	std::vector<LD> mimes::Axion< LD, Solver, Method<LD> >::dtheta
 	
 	template<class LD, const int Solver, class Method> 
 	std::vector<LD> mimes::Axion< LD, Solver, Method<LD> >::dzeta
 \end{cpp}
 %
 
 Moreover, the oscillation temperature, $\Tosc$, and the corresponding values of $a/\ai$ and $\theta$ are given in
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method>
 	LD mimes::Axion< LD, Solver, Method<LD> >::T_osc
 	
 	template<class LD, const int Solver, class Method>
 	LD mimes::Axion< LD, Solver, Method<LD> >::a_osc
 
 	template<class LD, const int Solver, class Method>
 	LD mimes::Axion< LD, Solver, Method<LD> >::theta_osc
 \end{cpp}
 %
 
 Also, the entropy injection between the last peak ($T=T_{\rm peak}$) and today ($T=T_0$), $\gamma$ (defined as in \eqs{eq:entropy_injection_gamma}), is given in 
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method>
 	LD mimes::Axion< LD, Solver, Method<LD> >::gamma
 \end{cpp}
 
 
 
 We can set another initial condition, $\thetai$, using 
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method>
 	void mimes::Axion< LD, Solver, Method<LD> >::setTheta_i(LD theta_i)
 \end{cpp}
 %
 We should note that running this function all variables are cleared. So we lose all information about the last time {\tt axionSolve()} ran. 
 
 In case the mass of the axion is changed, we also need to remake  the interpolation (\ie run \cppin{mimes::AxionEOM::makeInt()}). This is done using
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method>
 	void mimes::Axion< LD, Solver, Method<LD> >::restart()
 \end{cpp}
 %
 Again, this function clears all member variables. So it should be used with caution.
 
 
 Finally, there is static \cppin{mimes::Cosmo<LD>} member variable
 %
 \begin{cpp}
 	template<class LD, const int Solver, class Method>
 	static mimes::Cosmo<LD> mimes::Axion< LD, Solver, Method<LD> >::plasma
 \end{cpp}
 %
 This variable can be used without an instance of the \cppin{mimes::Axion<LD,Solver,Method>} class.
 
 \section{\PY modules}\label{app:modules}
 \setcounter{equation}{0}
-\DK{Describe the modules.}
+
+The various \PY modules, classes, and functions are designed to work exactly in the same way as the ones in \CPP. All the modules are located in \pyin{src/interfacePY}, so it is helpful to add {\pyin src} path to the system path at the top of every script calling \mimes. This is done by adding
+%
+\begin{py}
+	from sys import path as sysPath
+	sysPath.append('path_to_src')
+\end{py}
+%
+After this line, the others can be included.
 
 
 
 \section{Utilities}\label{app:util}
 \setcounter{equation}{0}
 \DK{Describe {\tt FormatFile.sh} and {\tt timeit.sh}.}
 
 \section{Quick guide to user input}\label{app:usr_input}
 \setcounter{equation}{0}
 In table~\ref{tab:input}, we provide a reference for all the user input, and compile-time options.
 %
 \begin{table}[h!]
 	\centering
 	\begin{tabular}{l l}
 		\hline\\[-0.4cm]
 		\multicolumn{2}{c}{\bf User run-time input}  \\
 		\hline\\[-0.4cm]
 
 		{\tt theta\_i} & initial angle.  \\
 		\hline\\[-0.4cm]
 
 		{\tt fa} & the PQ scale.\\
 		\hline\\[-0.4cm]
 
 		{\tt umax } & Once $u=\log a/a_i>${\tt umax}, the integration stops. Typical value: $\sim 500$.\\
 		\hline\\[-0.4cm]
 
 		{\tt TSTOP} & Once $T<${\tt TSTOP}, integration stops. Typical value: $10^{-4}~\GeV$.\\
 		\hline\\[-0.4cm]
 
  		{\tt ratio\_ini}& Integration starts at $u$ with $3H/\maT \approx${\tt ratio\_ini}. Typical value: $\sim 10^{3}$.\\
 		\hline\\[-0.4cm]
 
 		\multirow{1}{4cm}{{\tt N\_convergence\_max} {\tt convergence\_lim}} & \multirow{1}{12cm}{Integration stops when  the relative difference 
 		between two consecutive peaks  is less than {\tt convergence\_lim} for {\tt N\_convergence\_max} 
 		consecutive peaks. } \\ \\ \\ 
 		\hline\\[-0.4cm]
 
 		{\tt inputFile} & \multirow{1}{12cm}{Relative (or absolute) path to a file that describes the cosmology. The columns should be: $u$ $T ~[\GeV]$ $\log H$, with acceding $u$. Entropy injection should have stopped before the lowest temperature of given in {\tt inputFile}.} \\ \\  \\ \\
 		\hline\\[-0.4cm]
 
 		{\tt axionMass} & \multirow{1}{12cm}{ Instance of \cppin{mimes::AxionMass<LD>} class. In \CPP this instance is passed as a pointer to the constructor
 		of the \cppin{mimes::Axion<LD,Solver,Method>} class, while in \PY it is simply passed as a variable.}\\ \\  \\ \\
 		\hline\\[-0.4cm]
 
 		{\tt initial\_stepsize} &  Initial step-size of the solver. Default value: $10^{-2}$.\\ 
 		\hline\\[-0.4cm]
 
 		{\tt minimum\_stepsize} & Lower limit of the step-size. Default value:  $10^{-8}$.\\
 		\hline\\[-0.4cm]
 
 		{\tt maximum\_stepsize} & Upper limit of the step-size. Default value:  $10^{-2}$.\\
 		\hline\\[-0.4cm]
 
 		{\tt absolute\_tolerance} & \multirow{1}{12cm}{Absolute tolerance of the RK solver	(see also table~\ref{tab:RK_mimes_params}).  Default value:  $10^{-8}$.}\\\\
 		\hline\\[-0.4cm]
 
 		{\tt relative\_tolerance} & \multirow{1}{12cm}{Relative tolerance of the RK solver	(see also table~\ref{tab:RK_mimes_params}).  Default value:  $10^{-8}$.}\\\\
 		\hline\\[-0.4cm]
 		
 		{\tt beta} & \multirow{1}{12cm}{Aggressiveness of the adaptation strategy	(see also table~\ref{tab:RK_mimes_params}).  Default value:  $0.9$.}\\\\
 		\hline\\[-0.4cm]
 
 		{\tt fac\_max}, {\tt fac\_min} &\multirow{1}{12cm}{The step-size does not change more than {\tt fac\_max} and less than {\tt fac\_min} within a trial step (see also table~\ref{tab:RK_mimes_params}). Default values: $1.2$ and $0.8$, respectively.} \\ \\ \\ 
 		\hline\\[-0.4cm]
 		
 		{\tt maximum\_No\_steps} & \multirow{1}{12cm}{If integration needs more than {\tt maximum\_No\_steps} integration stops. Default value: $10^7$.}\\\\
 		\hline\\[-0.4cm]
 	\end{tabular}
 	\caption{Table of run-time user input.}
 	\label{tab:run_time-input}
 \end{table}
 		
 \begin{table}[h!]
 	\centering
 	\begin{tabular}{l l}
 		\multicolumn{2}{c}{\bf Required data files, with corresponding variables in {\tt \mimes/Paths.mk}}  \\
 		\hline\\[-0.4cm]
 	
 		{\tt cosmoDat}& \multirow{1}{12cm}{Relative path to data file with $T$ (in $\GeV$), $\heff$, $\geff$. If the path changes one must run
 		{\tt bash configure.sh} and {\tt make}.}\\\\		
 		\hline\\[-0.4cm]
 
 		{\tt axMDat}& \multirow{1}{12cm}{Relative path to data file with $T$ (in $\GeV$), $\chi$ (defined from \eqs{eq:axion_mass_def}). If the path changes one must run {\tt bash configure.sh} and {\tt make}. This variable can be omitted, since the user has to define an AxionMass instance.}\\\\\\\\		
 		\hline\\[-0.4cm]
 		
 		{\tt anFDat}& \multirow{1}{12cm}{Relative path to data file with $\thetamax$, $f(\thetamax)$. If the path changes one must run
 		{\tt bash configure.sh} and {\tt make}.}\\\\		
 		\hline\\[-0.4cm]
 
 
 
 	\end{tabular}
 	\caption{Paths to the required data files.}
 \label{tab:input}
 \end{table}
 
 
 \begin{table}[h!]
 	\centering
 	\begin{tabular}{l l}
 		\multicolumn{2}{c}{\bf User compile-time input}  \\
 		\hline\\[-0.4cm]
 
 		{\tt rootDir}& \multirow{1}{12cm}{The relative path of root directory of \mimes. Relevant only when compiling using {\tt make}. Available in all {\tt Definitions.mk}.}\\\\		
 		\hline\\[-0.4cm]
 		
 		{\tt LONG}& \multirow{1}{12cm}{{\tt long} for {\tt long double }or empty for {\tt double}. This should be a macro at the top of the {\tt .cpp} file. Available in all {\tt Definitions.mk}.}\\\\		
 		\hline\\[-0.4cm]
 
 		{\tt LONGpy}& \multirow{1}{12cm}{{\tt long} or empty. Same as {\tt LONG}, applies in the python modules. Available in {\tt \mimes/Definitions.mk}.}\\\\		
 		\hline\\[-0.4cm]
 
 		{\tt SOLVER}& \multirow{1}{12cm}{In order to use a Rosenbrock method {\tt SOLVER}=$1$. For explicit RK method, {\tt SOLVER}=$1$. 
 		This should be a macro defined before we include {\tt \mimes/src/Axion/AxionSolve.hpp}. Therefore it is preferable to 
 		define it with the compiler flag {\tt -DSOLVER=}$1$ or $2$. The corresponding variable in {\tt \mimes/Definitions.mk} applies to the python modules. The variable in {\tt \mimes/UserSpace/Cpp/Axion/Definitions.mk} applies to the example in the same directory.}\\\\\\\\\\\\\\		
 		\hline\\[-0.4cm]
 
 		{\tt METHOD}& \multirow{1}{12cm}{Depending of the solver, this macro should name one of its available methods. 
 		For {\tt SOLVER}=$1$, {\tt METHOD}=RODASPR2 (fourth order) or ROS34PW2 (third order). 	
 		For {\tt SOLVER}=$1$, {\tt METHOD}=DormandPrince (seventh order). This should be a macro defined before we include {\tt \mimes/src/Axion/AxionSolve.hpp}. Therefore it is preferable to 
 		define it with the compiler flag {\tt -DMETHOD=}RODASPR2, ROS34PW2, or DormandPrince.
 		The corresponding variable in {\tt \mimes/Definitions.mk} applies to the python modules. The variable in {\tt \mimes/UserSpace/Cpp/Axion/Definitions.mk} applies to the example in the same directory.}\\\\\\\\\\\\\\\\\\\\ 		
 		\hline\\[-0.4cm]
 		
 		\multicolumn{2}{c}{\bf Compiler options}  \\
 		\hline\\[-0.4cm]
 		
 		{\tt CC} &  \multirow{1}{12cm}{The preferred \CPP compiler ({\tt g++} by default). Corresponding variable in all {\tt Definitions.mk} files.} \\\\
 		\hline\\[-0.4cm]
 		
 		{\tt OPT} &  \multirow{1}{12cm}{Optimization level of the compiler. Available options {\tt OPT}={\tt O1}, {\tt O2}, {\tt O3} (be default). By Corresponding variable in all {\tt Definitions.mk} files.}   \\\\
 		\hline\\[-0.4cm]
 
 	\end{tabular}
 	\caption{User compile-time input. These are available in the various {\tt Definitions.mk} files, which are used when compiling using {\tt make}.}
 	\label{tab:compile_time-input}
 \end{table}
 
 
 
 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
 \newpage
 \bibliography{refs}{}
 \bibliographystyle{JHEP}                        
 
 \end{document}
diff --git a/UserSpace/JupyterNotebooks/AnharmonicFactor.ipynb b/UserSpace/JupyterNotebooks/AnharmonicFactor.ipynb
index ceb2dfd..8293f86 100755
--- a/UserSpace/JupyterNotebooks/AnharmonicFactor.ipynb
+++ b/UserSpace/JupyterNotebooks/AnharmonicFactor.ipynb
@@ -1,1091 +1,1108 @@
 {
  "cells": [
   {
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
    "outputs": [],
    "source": [
     "import numpy as np#you usually need numpy\n",
     "\n",
     "#---these are for plots---#\n",
     "import matplotlib\n",
     "matplotlib.use('nbAgg')\n",
     "import matplotlib.pyplot as plt\n",
     "\n",
     "plt.rcParams['font.size']=16\n",
     "plt.rcParams['font.family']='dejavu sans'\n",
     "\n",
     "plt.rcParams['mathtext.fontset']='stix'\n",
     "plt.rcParams['mathtext.rm']='custom'\n",
     "plt.rcParams['mathtext.it']='stix:italic'\n",
     "plt.rcParams['mathtext.bf']='stix:bold'\n",
     "#-------------------------#"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 2,
    "metadata": {},
    "outputs": [],
    "source": [
     "#load the module\n",
     "from sys import path as sysPath\n",
-    "from os import path as osPath\n",
-    "sysPath.append(osPath.join(osPath.dirname('./'), '../../src'))\n",
+    "sysPath.append('../../src')\n"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": []
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "#load the module\n",
+    "from sys import path as sysPath\n",
+    "sysPath.append('../../src')\n",
     "\n",
     "from interfacePy.AnharmonicFactor import anharmonicFactor #load the anharmonicFactor function form the AnharmonicFactor module\n",
     "from interfacePy.FT import FT #easy tick formatting\n",
     "\n"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 4,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
        "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "# plot data alongsode the interpolation (if there are large differences there is something wrong)\n",
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "    \n",
     "    \n",
     "    Y=[]\n",
     "    T=np.linspace(0,np.pi,50)\n",
     "    for t in T:\n",
     "        Y.append(anharmonicFactor(t))\n",
     "    sub.plot(T,Y,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')\n",
     "    \n",
     "    \n",
     "    sub.set_xlabel(r'$\\theta_{\\rm peak}$')\n",
     "    sub.xaxis.set_label_coords(0.5, -0.1) \n",
     "#     sub.set_ylabel(r'$ \\dfrac{2\\sqrt{2}}{\\pi \\ \\theta_{\\rm max}^2} \\ \\int_{-\\theta_{\\rm max}}^{\\theta_{\\rm max}} d\\theta \\ \\sqrt{ \\cos(\\theta) - \\cos(\\theta_{\\rm max}) }$')\n",
     "    sub.set_ylabel(r'$ f\\left( \\theta_{\\rm peak} \\right)$')\n",
     "    sub.yaxis.set_label_coords(-0.1,0.5) \n",
     "    #set major ticks\n",
     "    _M_xticks=[ 0,0.5,1,1.5,2,2.5,3]\n",
     "    _M_yticks=[ 0.5,0.6,0.7,0.8,0.9,1 ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[np.pi]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=0,xmax=np.pi,ymin=0.5,ymax=1.005,xscale='linear',yscale='linear')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "    \n",
     "    sub.text(np.pi*0.99,0.46,r'$\\pi$')\n",
     "\n",
     "    \n",
-    "    fig.savefig('anharmonic_factor.pdf',bbox_inches='tight')\n",
+    "#     fig.savefig('anharmonic_factor.pdf',bbox_inches='tight')\n",
     "    fig.show()"
    ]
   },
   {
    "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.9.5"
   }
  },
  "nbformat": 4,
  "nbformat_minor": 4
 }
diff --git a/UserSpace/JupyterNotebooks/Axion.ipynb b/UserSpace/JupyterNotebooks/Axion.ipynb
index 48d6248..8e03150 100755
--- a/UserSpace/JupyterNotebooks/Axion.ipynb
+++ b/UserSpace/JupyterNotebooks/Axion.ipynb
@@ -1,5361 +1,5362 @@
 {
  "cells": [
   {
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
    "outputs": [],
    "source": [
     "import numpy as np#you usually need numpy\n",
     "\n",
     "#---these are for plots---#\n",
     "import matplotlib\n",
     "matplotlib.use('nbAgg')\n",
     "import matplotlib.pyplot as plt\n",
     "\n",
     "plt.rcParams['font.size']=16\n",
     "plt.rcParams['font.family']='dejavu sans'\n",
     "\n",
     "plt.rcParams['mathtext.fontset']='stix'\n",
     "plt.rcParams['mathtext.rm']='custom'\n",
     "plt.rcParams['mathtext.it']='stix:italic'\n",
     "plt.rcParams['mathtext.bf']='stix:bold'\n",
     "#-------------------------#"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 2,
    "metadata": {},
    "outputs": [],
    "source": [
     "#load the module\n",
     "from sys import path as sysPath\n",
-    "from os import path as osPath\n",
-    "sysPath.append(osPath.join(osPath.dirname('./'), '../../src'))\n",
+    "sysPath.append('../../src')\n",
     "\n",
     "from interfacePy.Axion import Axion \n",
     "from interfacePy.AxionMass import AxionMass\n",
     "\n",
-    "from interfacePy.Cosmo import Hubble,rho_crit,h_hub\n",
+    "from interfacePy.Cosmo import Cosmo\n",
     "from interfacePy.FT import FT #easy tick formatting\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 3,
    "metadata": {},
    "outputs": [],
    "source": [
     "theta_i, fa=0.94435, 1e12\n",
     "# theta_i, fa=np.pi, 1e12\n",
     "# theta_i, fa=1e-3, 1e5\n",
     "\n",
     "umax=500\n",
     "TSTOP=1e-4\n",
     "ratio_ini=1e3\n",
     "\n",
     "N_convergence_max, convergence_lim=5, 1e-2 #this is fine, but you can experiment a bit. \n",
     "\n",
     "#radiation dominated example\n",
     "inputFile=\"../InputExamples/RDinput.dat\" \n",
     "\n",
     "# Matter domination example. \n",
     "# the NSC parameters (using the notation of 2012.07202) are:\n",
     "# T_end=1e-2 (GeV), c=3, T_ini=1e12 (GeV), and r=1e-1\n",
     "# inputFile=\"../InputExamples/MatterInput.dat\" \n",
     "\n",
     "\n",
     "# Kination domination example. \n",
     "# the NSC parameters (using the notation of 2012.07202) are:\n",
     "# T_end=0, c=6, T_ini=1e3 (GeV), and r=1e10\n",
     "# inputFile=\"../InputExamples/KinationInput.dat\" \n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 4,
    "metadata": {},
    "outputs": [],
    "source": [
     "#you can define the axion mass using a data file\n",
     "axionMass = AxionMass(r'../../src/data/chi.dat',0,1e5)\n",
     "\n",
     "#you can define the axion mass via a function\n",
     "# def ma2(T,fa):\n",
     "#     TQCD=150*1e-3;\n",
     "#     ma20=3.1575e-05/fa/fa;\n",
     "#     if T<=TQCD:\n",
     "#         return ma20;\n",
     "#     return ma20*pow((TQCD/T),8.16)\n",
     "\n",
     "# axionMass = AxionMass(ma2)\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 5,
    "metadata": {},
    "outputs": [],
    "source": [
     "# options for the solver\n",
     "# These variables are optional. Yoou can use the Axion class without them.\n",
     "initial_step_size=1e-2; #initial step the solver takes. \n",
     "minimum_step_size=1e-8; #This limits the sepsize to an upper limit. \n",
     "maximum_step_size=1e-2; #This limits the sepsize to a lower limit.\n",
     "absolute_tolerance=1e-8; #absolute tolerance of the RK solver\n",
     "relative_tolerance=1e-8; #relative tolerance of the RK solver\n",
     "beta=0.9; #controls how agreesive the adaptation is. Generally, it should be around but less than 1.\n",
     "\n",
     "#The stepsize does not increase more than fac_max, and less than fac_min. \n",
     "#This ensures a better stability. Ideally, fac_max=inf and fac_min=0, but in reality one must \n",
     "#tweak them in order to avoid instabilities.\n",
     "fac_max=1.2; \n",
     "fac_min=0.8;\n",
     "maximum_No_steps=int(1e7); #maximum steps the solver can take Quits if this number is reached even if integration is not finished.\n",
     "\n",
     "\n",
     "# Axion instance\n",
     "ax=Axion(theta_i, fa, umax, TSTOP, ratio_ini, N_convergence_max, convergence_lim, inputFile,axionMass,\n",
     "        initial_step_size,minimum_step_size, maximum_step_size, absolute_tolerance, \n",
     "        relative_tolerance, beta, fac_max, fac_min, maximum_No_steps)\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 6,
    "metadata": {},
    "outputs": [],
    "source": [
     "# Axion instance\n",
     "# you can always run Axion with the default parameters for the solver\n",
     "# ax=Axion(theta_i, fa, umax, TSTOP, ratio_ini, N_convergence_max, convergence_lim, inputFile)"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 7,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "0.019005775451660156"
+       "0.018198728561401367"
       ]
      },
      "execution_count": 7,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "# solve the EOM (this only gives you the relic, T_osc, theta_osc, and a_osc)\n",
     "ax.solveAxion()"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 8,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "(0.11935020970232879, 0.9785452023797905, 0.8975109868818929)"
+       "(0.11934766380256942, 0.978544806529701, 0.8975020887699964)"
       ]
      },
      "execution_count": 8,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "ax.relic, ax.T_osc, ax.theta_osc"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 9,
    "metadata": {},
    "outputs": [],
    "source": [
     "ax.getPeaks()#this gives you the peaks of the oscillation\n",
     "ax.getPoints()#this gives you all the points of integration\n",
     "ax.getErrors()#this gives you local errors of integration"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 10,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "    \n",
     "    #this plot shows the peaks of the oscillation\n",
     "    sub.plot(ax.T_peak,ax.theta_peak,linestyle=':',marker='+',color='xkcd:blue',linewidth=2)\n",
     "\n",
     "    #this plot shows all the points\n",
     "    sub.plot(ax.T,ax.theta,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')\n",
     "\n",
     "    \n",
     "    \n",
     "    sub.set_xlabel(r'$T ~[{\\rm GeV}]$')\n",
     "    sub.xaxis.set_label_coords(0.5, -0.1) \n",
     "    sub.set_ylabel(r'$\\theta$')\n",
     "    sub.yaxis.set_label_coords(-0.1,0.5) \n",
     "\n",
     "    sub.axhline(ax.theta_osc,linestyle=':',color='xkcd:red',linewidth=1.5)\n",
     "    sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)\n",
     "    \n",
     "    \n",
     "    \n",
     "    #set major ticks\n",
     "    _M_xticks=[ round(0.45+i*0.15,4) for i in range(0,15) ]\n",
     "    _M_yticks=[ round(-0.4+i*0.1,3) for i in range(0,20,2)  ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=0.45,xmax=2,ymin=-0.4,ymax=1,xscale='linear',yscale='linear')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "\n",
     "    \n",
     "    sub.text(x=0.92,y=0.1, s=r'$T_{\\rm osc}$',rotation=90)\n",
     "    sub.text(x=1.5,y=0.8, s=r'$\\theta_{\\rm osc}$')\n",
     "    \n",
     "    sub.text(x=0.5,y=0.2, s=r'$\\theta_{\\rm max}$',rotation=20)\n",
     "    \n",
     "    sub.text(x=1.6,y=-0.3,\n",
     "         s=r'$f_{a}=10^{12}~{\\rm GeV}$'+'\\n'+ r'$\\theta_i = 0.94435$')\n",
     "    \n",
     "#     fig.savefig('theta_evolution.pdf',bbox_inches='tight')\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 11,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "    \n",
     "    sub.plot(ax.T,ax.zeta,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')\n",
     "    sub.plot(ax.T_peak,ax.zeta_peak,linestyle=':',marker='+',color='xkcd:blue',linewidth=2)\n",
     "    \n",
     "    sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)\n",
     "\n",
     "    \n",
     "    #set major ticks\n",
     "    _M_xticks=[ round(0.45+i*0.15,4) for i in range(0,15) ]\n",
     "    _M_yticks=[ round(-20+i*5,3) for i in range(0,20,1)  ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=0.45,xmax=2,ymin=-20,ymax=20,xscale='linear',yscale='linear')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "\n",
     "    \n",
     "    sub.text(x=0.92,y=10, s=r'$T_{\\rm osc}$',rotation=90)\n",
     "    \n",
     "    \n",
     "    sub.text(x=1.6,y=-15,\n",
     "         s=r'$f_{a}=10^{12}~{\\rm GeV}$'+'\\n'+ r'$\\theta_i = 0.94435$')\n",
     "\n",
     "    \n",
     "#     fig.savefig('zeta_evolution.pdf',bbox_inches='tight')\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 12,
    "metadata": {},
    "outputs": [
     {
      "name": "stderr",
      "output_type": "stream",
      "text": [
       "<ipython-input-12-b569a0ff9cd6>:7: RuntimeWarning: invalid value encountered in true_divide\n",
       "  sub.plot(ax.T,np.abs(ax.dzeta/ax.zeta),linestyle='-',linewidth=2,alpha=1,c='xkcd:red',label=r'$\\dfrac{\\delta \\zeta}{\\zeta}$')\n"
      ]
     },
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "    \n",
     "    sub.plot(ax.T,np.abs(ax.dtheta/ax.theta),linestyle='-',linewidth=2,alpha=1,c='xkcd:black',label=r'$\\dfrac{\\delta \\theta}{\\theta}$')\n",
     "    sub.plot(ax.T,np.abs(ax.dzeta/ax.zeta),linestyle='-',linewidth=2,alpha=1,c='xkcd:red',label=r'$\\dfrac{\\delta \\zeta}{\\zeta}$')\n",
     "    \n",
     "    sub.set_yscale('log')\n",
     "    sub.set_xscale('linear')\n",
     "    \n",
     "    sub.set_xlabel(r'$T ~[{\\rm GeV}]$')\n",
     "    sub.xaxis.set_label_coords(0.5, -0.1) \n",
     "    sub.set_ylabel(r'local errors')\n",
     "    sub.yaxis.set_label_coords(-0.1,0.5) \n",
     "    \n",
     "    sub.legend(bbox_to_anchor=(0.98, 0.95),borderaxespad=0., \n",
     "           borderpad=0.05,ncol=1,loc='upper right',fontsize=14,framealpha=0)\n",
     "\n",
     "    \n",
     "    \n",
     "    \n",
     "    sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)\n",
     "    \n",
     "    \n",
     "    #set major ticks\n",
     "    _M_xticks=[ round(0.45+i*0.15,4) for i in range(0,15) ]\n",
     "    _M_yticks=[ 10.**i for i in range(-12,5,1)  ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=0.45,xmax=2,ymin=1e-11,ymax=1e-4,xscale='linear',yscale='log')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "\n",
     "    \n",
     "    sub.text(x=0.92,y=1e-6, s=r'$T_{\\rm osc}$',rotation=90)\n",
     "    \n",
     "#     fig.savefig('local_errors.pdf',bbox_inches='tight')\n",
     "\n",
     "    \n",
     "    fig.show()\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 13,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "\n",
     "    sub.hist(ax.T,bins=30,color='xkcd:blue')\n",
     "    \n",
     "    sub.set_yscale('log')\n",
     "    sub.set_xscale('linear')\n",
     "    \n",
     "    sub.set_xlabel(r'$T ~[{\\rm GeV}]$')\n",
     "    sub.xaxis.set_label_coords(0.5, -0.1) \n",
     "    sub.set_ylabel(r'Number of steps')\n",
     "    sub.yaxis.set_label_coords(-0.1,0.5) \n",
     "    \n",
     "\n",
     "    sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)\n",
     "    \n",
     "\n",
     "    sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)\n",
     "    \n",
     "    \n",
     "    #set major ticks\n",
     "    _M_xticks=[ round(0.45+i*0.15,4) for i in range(0,15) ]\n",
     "    _M_yticks=[ 10.**i for i in range(-12,5,1)  ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=0.45,xmax=2,ymin=1e0,ymax=1e3,xscale='linear',yscale='log')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "\n",
     "    \n",
     "    sub.text(x=0.92,y=1e2, s=r'$T_{\\rm osc}$',rotation=90)\n",
     "\n",
     "#     fig.savefig('histogram.pdf',bbox_inches='tight')\n",
     "\n",
     "    fig.show()\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 14,
    "metadata": {},
+   "outputs": [],
+   "source": [
+    "cosmo=Cosmo()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "    \n",
-    "    sub.plot(ax.T,ax.rho_axion/rho_crit,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')\n",
-    "    sub.plot(ax.T_peak,ax.rho_axion_peak/rho_crit,linestyle=':',linewidth=2,alpha=1,c='xkcd:blue')\n",
+    "    sub.plot(ax.T,ax.rho_axion/cosmo.rho_crit,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')\n",
+    "    sub.plot(ax.T_peak,ax.rho_axion_peak/cosmo.rho_crit,linestyle=':',linewidth=2,alpha=1,c='xkcd:blue')\n",
     "\n",
     "    \n",
     "    sub.set_xlabel(r'$T ~[{\\rm GeV}]$')\n",
     "    sub.xaxis.set_label_coords(0.5, -0.1) \n",
     "    sub.set_ylabel(r'$\\dfrac{\\rho_{a}(T)}{\\rho_{\\rm crit}}$')\n",
     "    sub.yaxis.set_label_coords(-0.1,0.5) \n",
     "    \n",
     "    \n",
     "    sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)\n",
     "    #set major ticks\n",
     "    _M_xticks=[ round(0.45+i*0.15,4) for i in range(0,15) ]\n",
     "    _M_yticks=[ 10.**i for i in range(32,42,1)  ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=0.45,xmax=2,ymin=1e32,ymax=1e36,xscale='linear',yscale='log')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "\n",
     "    \n",
     "    sub.text(x=0.92,y=1e33, s=r'$T_{\\rm osc}$',rotation=90)\n",
     "\n",
     "#     fig.savefig('axion_energy_density.pdf',bbox_inches='tight')\n",
     "\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 16,
    "metadata": {},
    "outputs": [],
    "source": [
     "#run the destructor\n",
     "del ax"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": null,
    "metadata": {},
    "outputs": [],
    "source": []
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": []
   }
  ],
  "metadata": {
   "kernelspec": {
-   "display_name": "Python 3 (ipykernel)",
+   "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.8.8"
+   "version": "3.9.5"
   }
  },
  "nbformat": 4,
  "nbformat_minor": 2
 }
diff --git a/UserSpace/JupyterNotebooks/AxionMass.ipynb b/UserSpace/JupyterNotebooks/AxionMass.ipynb
index 52f7605..cfae66c 100755
--- a/UserSpace/JupyterNotebooks/AxionMass.ipynb
+++ b/UserSpace/JupyterNotebooks/AxionMass.ipynb
@@ -1,1113 +1,1125 @@
 {
  "cells": [
   {
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
    "outputs": [],
    "source": [
     "import numpy as np#you usually need numpy\n",
     "\n",
     "#---these are for plots---#\n",
     "import matplotlib\n",
     "matplotlib.use('nbAgg')\n",
     "import matplotlib.pyplot as plt\n",
     "\n",
     "plt.rcParams['font.size']=16\n",
     "plt.rcParams['font.family']='dejavu sans'\n",
     "\n",
     "plt.rcParams['mathtext.fontset']='stix'\n",
     "plt.rcParams['mathtext.rm']='custom'\n",
     "plt.rcParams['mathtext.it']='stix:italic'\n",
     "plt.rcParams['mathtext.bf']='stix:bold'\n",
     "#-------------------------#"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 2,
    "metadata": {},
    "outputs": [],
    "source": [
     "#load the module\n",
     "from sys import path as sysPath\n",
-    "from os import path as osPath\n",
-    "sysPath.append(osPath.join(osPath.dirname('./'), '../../src'))\n",
+    "sysPath.append('../../src')\n",
     "\n",
     "from interfacePy.AxionMass import AxionMass\n",
     "from interfacePy.FT import FT #easy tick formatting"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 3,
    "metadata": {},
    "outputs": [],
    "source": [
     "#you can define the axion mass using a data file\n",
-    "axionMassInterpolation = AxionMass(r'../../src/data/chi.dat',0,1e5)\n",
+    "axionMassInterpolation = AxionMass(r'../../src/data/chi.dat',0,10)\n",
+    "\n",
+    "def ma2_MAX(T,fa):\n",
+    "    TMAX=axionMassInterpolation.getTMax()\n",
+    "    chiMAX=axionMassInterpolation.getChiMax()\n",
+    "    return chiMAX/fa/fa/pow(T/TMAX,8.16)\n",
+    "#set this in order to get the correct scaling (for the dataset we use here)\n",
+    "axionMassInterpolation.set_ma2_MAX( ma2_MAX )\n",
+    "\n",
+    "\n",
     "\n",
     "#you can define the axion mass via a function\n",
     "def ma2(T,fa):\n",
     "    TQCD=150*1e-3;\n",
     "    ma20=3.1575e-05/fa/fa;\n",
     "    if T<=TQCD:\n",
     "        return ma20;\n",
     "    return ma20*pow((TQCD/T),8.16)\n",
     "\n",
-    "axionMassFunction = AxionMass(ma2)\n"
+    "axionMassFunction = AxionMass(ma2)\n",
+    "\n",
+    "\n",
+    "\n",
+    "\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 4,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
        "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "    \n",
     "    X=np.logspace(-2,1,500)\n",
     "    Y=[]\n",
     "    for t in X:\n",
     "        Y.append(axionMassInterpolation.ma2(t,1e12)**0.5*1e15)\n",
     "    sub.plot(X,Y,linestyle='-',linewidth=1,alpha=1,c='xkcd:black',label='interpolation')\n",
     "    \n",
     "    Y=[]\n",
     "    for t in X:\n",
     "        Y.append(axionMassFunction.ma2(t,1e12)**0.5*1e15)\n",
     "    sub.plot(X,Y,linestyle='--',linewidth=1,alpha=1,c='xkcd:red',label='function')\n",
     "    \n",
     "    sub.set_xlabel(r'$T ~[{\\rm GeV}]$')\n",
     "    sub.xaxis.set_label_coords(0.5, -0.1) \n",
     "    sub.set_ylabel(r'$ m_a(T) ~[{\\rm \\mu eV}]$')\n",
     "    sub.yaxis.set_label_coords(-0.1,0.5) \n",
     "   \n",
     "    sub.legend(bbox_to_anchor=(1, 0.5),borderaxespad=0., \n",
     "               borderpad=0.05,ncol=1,loc='lower right',fontsize=14,framealpha=0)\n",
     "\n",
     "    #set major ticks\n",
     "    _M_xticks=[ 10.**i for i in range(-2,2) ]\n",
     "    _M_yticks=[ 10.**i for i in range(-7,2) ]\n",
     "\n",
     "    #set major ticks that will not have a label\n",
     "    _M_xticks_exception=[]\n",
     "    _M_yticks_exception=[]\n",
     "\n",
     "    _m_xticks=[j*10.**i for i in range(-2,2) for j in range(1,10)]\n",
     "    _m_yticks=[]  \n",
     "    ft=FT(_M_xticks,_M_yticks,\n",
     "                 _M_xticks_exception,_M_yticks_exception,\n",
     "                 _m_xticks,_m_yticks,\n",
     "                 xmin=1e-2,xmax=1e1,ymin=1e-7,ymax=1e1,xscale='log',yscale='log')\n",
     "\n",
     "    ft.format_ticks(plt,sub)    \n",
     "\n",
     "#     fig.savefig('axion_mass.pdf',bbox_inches='tight')\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "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.9.5"
   }
  },
  "nbformat": 4,
  "nbformat_minor": 2
 }
diff --git a/UserSpace/JupyterNotebooks/Cosmo.ipynb b/UserSpace/JupyterNotebooks/Cosmo.ipynb
index 86d46cf..dd166c6 100755
--- a/UserSpace/JupyterNotebooks/Cosmo.ipynb
+++ b/UserSpace/JupyterNotebooks/Cosmo.ipynb
@@ -1,5058 +1,5066 @@
 {
  "cells": [
   {
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
    "outputs": [],
    "source": [
     "import numpy as np#you usually need numpy\n",
     "\n",
     "#---these are for plots---#\n",
     "import matplotlib\n",
     "matplotlib.use('nbAgg')\n",
     "import matplotlib.pyplot as plt\n",
     "\n",
     "plt.rcParams['font.size']=16\n",
     "plt.rcParams['font.family']='dejavu sans'\n",
     "\n",
     "plt.rcParams['mathtext.fontset']='stix'\n",
     "plt.rcParams['mathtext.rm']='custom'\n",
     "plt.rcParams['mathtext.it']='stix:italic'\n",
     "plt.rcParams['mathtext.bf']='stix:bold'\n",
     "#-------------------------#"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 2,
    "metadata": {},
    "outputs": [],
    "source": [
     "#load the module\n",
     "from sys import path as sysPath\n",
-    "from os import path as osPath\n",
-    "sysPath.append(osPath.join(osPath.dirname('./'), '../../src'))\n",
+    "sysPath.append('../../src')\n",
     "\n",
     "import interfacePy.Cosmo as Cosmo\n",
     "from interfacePy.FT import FT #easy tick formatting"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 3,
    "metadata": {},
+   "outputs": [],
+   "source": [
+    "cosmo=Cosmo()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "\n",
     "    T=np.logspace(-5,5,500)\n",
-    "    gt=[Cosmo.geff(i) for i in T]\n",
-    "    ht=[Cosmo.heff(i) for i in T]\n",
+    "    gt=[cosmo.geff(i) for i in T]\n",
+    "    ht=[cosmo.heff(i) for i in T]\n",
     "\n",
     "    sub.plot(T,gt,linestyle='--',c='xkcd:red',label=r\"$g_{\\rm eff} (T)$\")\n",
     "    sub.plot(T,ht,linestyle=':',c='xkcd:black',label=r\"$h_{\\rm eff} (T)$\")\n",
     "\n",
     "    sub.set_xlabel(r'$T ~ [{\\rm GeV}]$')\n",
     "    sub.set_ylabel(r'rel. dof')\n",
     "\n",
     "    sub.legend(bbox_to_anchor=(1, 0.0),borderaxespad=0., \n",
     "               borderpad=0.05,ncol=1,loc='lower right',fontsize=14,framealpha=0)\n",
     "    sub.set_yscale('log')\n",
     "    sub.set_xscale('log')\n",
     "\n",
     "\n",
     "\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 5,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "\n",
     "    T=np.logspace(-5,5,500)\n",
-    "    dg=[Cosmo.dgeffdT (i) for i in T]\n",
-    "    dh=[Cosmo.dheffdT(i) for i in T]\n",
+    "    dg=[cosmo.dgeffdT (i) for i in T]\n",
+    "    dh=[cosmo.dheffdT(i) for i in T]\n",
     "\n",
     "    sub.plot(T,dg,linestyle='--',c='xkcd:red',label=r\"$\\dfrac{d g_{\\rm eff}}{dT} (T)$\")\n",
     "    sub.plot(T,dh,linestyle=':',c='xkcd:black',label=r\"$\\dfrac{d h_{\\rm eff}}{dT} (T)$\")\n",
     "\n",
     "    sub.set_xlabel(r'$T ~ [{\\rm GeV}]$')\n",
     "\n",
     "    sub.legend(bbox_to_anchor=(1, 0.5),borderaxespad=0., \n",
     "               borderpad=0.05,ncol=1,loc='lower right',fontsize=14,framealpha=0)\n",
     "    sub.set_yscale('symlog')\n",
     "    sub.set_xscale('log')\n",
     "\n",
     "\n",
     "\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 6,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "\n",
     "    T=np.logspace(-5,5,500)\n",
-    "    dht=[Cosmo.dh(i) for i in T]\n",
+    "    dht=[cosmo.dh(i) for i in T]\n",
     "    sub.plot(T,dht,linestyle='-',c='xkcd:black') \n",
     "    sub.set_xlabel(r'$T ~ [{\\rm GeV}]$')\n",
     "    sub.set_ylabel(r'$\\delta_h = 1 + \\dfrac{1}{3} \\dfrac{d \\log h_{\\rm eff} }{d \\log T}$')\n",
     "    sub.set_yscale('linear')\n",
     "    sub.set_xscale('log')\n",
     "\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 7,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<img src=\"\" width=\"900\">"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "fig=plt.figure(figsize=(9,4))\n",
     "fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "fig.suptitle('')\n",
     "\n",
     "sub = fig.add_subplot(1,1,1)\n",
     "\n",
     "T=np.logspace(-5,5,500)\n",
-    "H=[Cosmo.Hubble(i) for i in T]\n",
+    "H=[cosmo.Hubble(i) for i in T]\n",
     "sub.plot(T,H,linestyle='-',c='xkcd:black')\n",
     "sub.set_ylabel(r'$H ~ [{\\rm GeV}]$')\n",
     "sub.set_yscale('log')\n",
     "sub.set_xscale('log')\n",
     "\n",
     "sub.set_xlabel(r'$T \\; [{\\rm GeV}]$')\n",
     "\n",
     "fig.show()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 8,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "application/javascript": [
        "/* Put everything inside the global mpl namespace */\n",
        "/* global mpl */\n",
        "window.mpl = {};\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(\n",
        "            '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",
        "\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 = document.createElement('div');\n",
        "    this.root.setAttribute('style', 'display: inline-block');\n",
        "    this._root_extra_style(this.root);\n",
        "\n",
        "    parent_element.appendChild(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 (fig.ratio !== 1) {\n",
        "            fig.send_message('set_dpi_ratio', { dpi_ratio: fig.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 = document.createElement('div');\n",
        "    titlebar.classList =\n",
        "        'ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix';\n",
        "    var titletext = document.createElement('div');\n",
        "    titletext.classList = 'ui-dialog-title';\n",
        "    titletext.setAttribute(\n",
        "        'style',\n",
        "        'width: 100%; text-align: center; padding: 3px;'\n",
        "    );\n",
        "    titlebar.appendChild(titletext);\n",
        "    this.root.appendChild(titlebar);\n",
        "    this.header = titletext;\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (_canvas_div) {};\n",
        "\n",
        "mpl.figure.prototype._init_canvas = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var canvas_div = (this.canvas_div = document.createElement('div'));\n",
        "    canvas_div.setAttribute(\n",
        "        'style',\n",
        "        'border: 1px solid #ddd;' +\n",
        "            'box-sizing: content-box;' +\n",
        "            'clear: both;' +\n",
        "            'min-height: 1px;' +\n",
        "            'min-width: 1px;' +\n",
        "            'outline: 0;' +\n",
        "            'overflow: hidden;' +\n",
        "            'position: relative;' +\n",
        "            'resize: both;'\n",
        "    );\n",
        "\n",
        "    function on_keyboard_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.key_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    canvas_div.addEventListener(\n",
        "        'keydown',\n",
        "        on_keyboard_event_closure('key_press')\n",
        "    );\n",
        "    canvas_div.addEventListener(\n",
        "        'keyup',\n",
        "        on_keyboard_event_closure('key_release')\n",
        "    );\n",
        "\n",
        "    this._canvas_extra_style(canvas_div);\n",
        "    this.root.appendChild(canvas_div);\n",
        "\n",
        "    var canvas = (this.canvas = document.createElement('canvas'));\n",
        "    canvas.classList.add('mpl-canvas');\n",
        "    canvas.setAttribute('style', 'box-sizing: content-box;');\n",
        "\n",
        "    this.context = canvas.getContext('2d');\n",
        "\n",
        "    var backingStore =\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        this.context.webkitBackingStorePixelRatio ||\n",
        "        this.context.mozBackingStorePixelRatio ||\n",
        "        this.context.msBackingStorePixelRatio ||\n",
        "        this.context.oBackingStorePixelRatio ||\n",
        "        this.context.backingStorePixelRatio ||\n",
        "        1;\n",
        "\n",
        "    this.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
        "\n",
        "    var rubberband_canvas = (this.rubberband_canvas = document.createElement(\n",
        "        'canvas'\n",
        "    ));\n",
        "    rubberband_canvas.setAttribute(\n",
        "        'style',\n",
        "        'box-sizing: content-box; position: absolute; left: 0; top: 0; z-index: 1;'\n",
        "    );\n",
        "\n",
        "    // Apply a ponyfill if ResizeObserver is not implemented by browser.\n",
        "    if (this.ResizeObserver === undefined) {\n",
        "        if (window.ResizeObserver !== undefined) {\n",
        "            this.ResizeObserver = window.ResizeObserver;\n",
        "        } else {\n",
        "            var obs = _JSXTOOLS_RESIZE_OBSERVER({});\n",
        "            this.ResizeObserver = obs.ResizeObserver;\n",
        "        }\n",
        "    }\n",
        "\n",
        "    this.resizeObserverInstance = new this.ResizeObserver(function (entries) {\n",
        "        var nentries = entries.length;\n",
        "        for (var i = 0; i < nentries; i++) {\n",
        "            var entry = entries[i];\n",
        "            var width, height;\n",
        "            if (entry.contentBoxSize) {\n",
        "                if (entry.contentBoxSize instanceof Array) {\n",
        "                    // Chrome 84 implements new version of spec.\n",
        "                    width = entry.contentBoxSize[0].inlineSize;\n",
        "                    height = entry.contentBoxSize[0].blockSize;\n",
        "                } else {\n",
        "                    // Firefox implements old version of spec.\n",
        "                    width = entry.contentBoxSize.inlineSize;\n",
        "                    height = entry.contentBoxSize.blockSize;\n",
        "                }\n",
        "            } else {\n",
        "                // Chrome <84 implements even older version of spec.\n",
        "                width = entry.contentRect.width;\n",
        "                height = entry.contentRect.height;\n",
        "            }\n",
        "\n",
        "            // Keep the size of the canvas and rubber band canvas in sync with\n",
        "            // the canvas container.\n",
        "            if (entry.devicePixelContentBoxSize) {\n",
        "                // Chrome 84 implements new version of spec.\n",
        "                canvas.setAttribute(\n",
        "                    'width',\n",
        "                    entry.devicePixelContentBoxSize[0].inlineSize\n",
        "                );\n",
        "                canvas.setAttribute(\n",
        "                    'height',\n",
        "                    entry.devicePixelContentBoxSize[0].blockSize\n",
        "                );\n",
        "            } else {\n",
        "                canvas.setAttribute('width', width * fig.ratio);\n",
        "                canvas.setAttribute('height', height * fig.ratio);\n",
        "            }\n",
        "            canvas.setAttribute(\n",
        "                'style',\n",
        "                'width: ' + width + 'px; height: ' + height + 'px;'\n",
        "            );\n",
        "\n",
        "            rubberband_canvas.setAttribute('width', width);\n",
        "            rubberband_canvas.setAttribute('height', height);\n",
        "\n",
        "            // And update the size in Python. We ignore the initial 0/0 size\n",
        "            // that occurs as the element is placed into the DOM, which should\n",
        "            // otherwise not happen due to the minimum size styling.\n",
        "            if (fig.ws.readyState == 1 && width != 0 && height != 0) {\n",
        "                fig.request_resize(width, height);\n",
        "            }\n",
        "        }\n",
        "    });\n",
        "    this.resizeObserverInstance.observe(canvas_div);\n",
        "\n",
        "    function on_mouse_event_closure(name) {\n",
        "        return function (event) {\n",
        "            return fig.mouse_event(event, name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousedown',\n",
        "        on_mouse_event_closure('button_press')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseup',\n",
        "        on_mouse_event_closure('button_release')\n",
        "    );\n",
        "    // Throttle sequential mouse events to 1 every 20ms.\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mousemove',\n",
        "        on_mouse_event_closure('motion_notify')\n",
        "    );\n",
        "\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseenter',\n",
        "        on_mouse_event_closure('figure_enter')\n",
        "    );\n",
        "    rubberband_canvas.addEventListener(\n",
        "        'mouseleave',\n",
        "        on_mouse_event_closure('figure_leave')\n",
        "    );\n",
        "\n",
        "    canvas_div.addEventListener('wheel', function (event) {\n",
        "        if (event.deltaY < 0) {\n",
        "            event.step = 1;\n",
        "        } else {\n",
        "            event.step = -1;\n",
        "        }\n",
        "        on_mouse_event_closure('scroll')(event);\n",
        "    });\n",
        "\n",
        "    canvas_div.appendChild(canvas);\n",
        "    canvas_div.appendChild(rubberband_canvas);\n",
        "\n",
        "    this.rubberband_context = rubberband_canvas.getContext('2d');\n",
        "    this.rubberband_context.strokeStyle = '#000000';\n",
        "\n",
        "    this._resize_canvas = function (width, height, forward) {\n",
        "        if (forward) {\n",
        "            canvas_div.style.width = width + 'px';\n",
        "            canvas_div.style.height = height + 'px';\n",
        "        }\n",
        "    };\n",
        "\n",
        "    // Disable right mouse context menu.\n",
        "    this.rubberband_canvas.addEventListener('contextmenu', function (_e) {\n",
        "        event.preventDefault();\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 toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'mpl-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'mpl-button-group';\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'mpl-button-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        var button = (fig.buttons[name] = document.createElement('button'));\n",
        "        button.classList = 'mpl-widget';\n",
        "        button.setAttribute('role', 'button');\n",
        "        button.setAttribute('aria-disabled', 'false');\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "\n",
        "        var icon_img = document.createElement('img');\n",
        "        icon_img.src = '_images/' + image + '.png';\n",
        "        icon_img.srcset = '_images/' + image + '_large.png 2x';\n",
        "        icon_img.alt = tooltip;\n",
        "        button.appendChild(icon_img);\n",
        "\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    var fmt_picker = document.createElement('select');\n",
        "    fmt_picker.classList = 'mpl-widget';\n",
        "    toolbar.appendChild(fmt_picker);\n",
        "    this.format_dropdown = fmt_picker;\n",
        "\n",
        "    for (var ind in mpl.extensions) {\n",
        "        var fmt = mpl.extensions[ind];\n",
        "        var option = document.createElement('option');\n",
        "        option.selected = fmt === mpl.default_extension;\n",
        "        option.innerHTML = fmt;\n",
        "        fmt_picker.appendChild(option);\n",
        "    }\n",
        "\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\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",
        "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",
        "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], msg['forward']);\n",
        "        fig.send_message('refresh', {});\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_rubberband = function (fig, msg) {\n",
        "    var x0 = msg['x0'] / fig.ratio;\n",
        "    var y0 = (fig.canvas.height - msg['y0']) / fig.ratio;\n",
        "    var x1 = msg['x1'] / fig.ratio;\n",
        "    var y1 = (fig.canvas.height - msg['y1']) / fig.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,\n",
        "        0,\n",
        "        fig.canvas.width / fig.ratio,\n",
        "        fig.canvas.height / fig.ratio\n",
        "    );\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",
        "        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.handle_history_buttons = function (fig, msg) {\n",
        "    for (var key in msg) {\n",
        "        if (!(key in fig.buttons)) {\n",
        "            continue;\n",
        "        }\n",
        "        fig.buttons[key].disabled = !msg[key];\n",
        "        fig.buttons[key].setAttribute('aria-disabled', !msg[key]);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_navigate_mode = function (fig, msg) {\n",
        "    if (msg['mode'] === 'PAN') {\n",
        "        fig.buttons['Pan'].classList.add('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    } else if (msg['mode'] === 'ZOOM') {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.add('active');\n",
        "    } else {\n",
        "        fig.buttons['Pan'].classList.remove('active');\n",
        "        fig.buttons['Zoom'].classList.remove('active');\n",
        "    }\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",
        "\n",
        "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
        "                evt.data\n",
        "            );\n",
        "            fig.updated_canvas_event();\n",
        "            fig.waiting = false;\n",
        "            return;\n",
        "        } else if (\n",
        "            typeof evt.data === 'string' &&\n",
        "            evt.data.slice(0, 21) === 'data:image/png;base64'\n",
        "        ) {\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(\n",
        "                \"No handler for the '\" + msg_type + \"' message type: \",\n",
        "                msg\n",
        "            );\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(\n",
        "                    \"Exception inside the 'handler_\" + msg_type + \"' callback:\",\n",
        "                    e,\n",
        "                    e.stack,\n",
        "                    msg\n",
        "                );\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",
        "    }\n",
        "    if (e.target) {\n",
        "        targ = e.target;\n",
        "    } else if (e.srcElement) {\n",
        "        targ = e.srcElement;\n",
        "    }\n",
        "    if (targ.nodeType === 3) {\n",
        "        // defeat Safari bug\n",
        "        targ = targ.parentNode;\n",
        "    }\n",
        "\n",
        "    // pageX,Y are the mouse positions relative to the document\n",
        "    var boundingRect = targ.getBoundingClientRect();\n",
        "    var x = e.pageX - (boundingRect.left + document.body.scrollLeft);\n",
        "    var y = e.pageY - (boundingRect.top + document.body.scrollTop);\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",
        "        }\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",
        "        this.canvas.focus();\n",
        "        this.canvas_div.focus();\n",
        "    }\n",
        "\n",
        "    var x = canvas_pos.x * this.ratio;\n",
        "    var y = canvas_pos.y * this.ratio;\n",
        "\n",
        "    this.send_message(name, {\n",
        "        x: x,\n",
        "        y: y,\n",
        "        button: event.button,\n",
        "        step: event.step,\n",
        "        guiEvent: simpleKeys(event),\n",
        "    });\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",
        "    // Prevent repeat events\n",
        "    if (name === 'key_press') {\n",
        "        if (event.which === this._key) {\n",
        "            return;\n",
        "        } else {\n",
        "            this._key = event.which;\n",
        "        }\n",
        "    }\n",
        "    if (name === 'key_release') {\n",
        "        this._key = null;\n",
        "    }\n",
        "\n",
        "    var value = '';\n",
        "    if (event.ctrlKey && event.which !== 17) {\n",
        "        value += 'ctrl+';\n",
        "    }\n",
        "    if (event.altKey && event.which !== 18) {\n",
        "        value += 'alt+';\n",
        "    }\n",
        "    if (event.shiftKey && event.which !== 16) {\n",
        "        value += 'shift+';\n",
        "    }\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, 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",
        "\n",
        "///////////////// REMAINING CONTENT GENERATED BY embed_js.py /////////////////\n",
        "// prettier-ignore\n",
        "var _JSXTOOLS_RESIZE_OBSERVER=function(A){var t,i=new WeakMap,n=new WeakMap,a=new WeakMap,r=new WeakMap,o=new Set;function s(e){if(!(this instanceof s))throw new TypeError(\"Constructor requires 'new' operator\");i.set(this,e)}function h(){throw new TypeError(\"Function is not a constructor\")}function c(e,t,i,n){e=0 in arguments?Number(arguments[0]):0,t=1 in arguments?Number(arguments[1]):0,i=2 in arguments?Number(arguments[2]):0,n=3 in arguments?Number(arguments[3]):0,this.right=(this.x=this.left=e)+(this.width=i),this.bottom=(this.y=this.top=t)+(this.height=n),Object.freeze(this)}function d(){t=requestAnimationFrame(d);var s=new WeakMap,p=new Set;o.forEach((function(t){r.get(t).forEach((function(i){var r=t instanceof window.SVGElement,o=a.get(t),d=r?0:parseFloat(o.paddingTop),f=r?0:parseFloat(o.paddingRight),l=r?0:parseFloat(o.paddingBottom),u=r?0:parseFloat(o.paddingLeft),g=r?0:parseFloat(o.borderTopWidth),m=r?0:parseFloat(o.borderRightWidth),w=r?0:parseFloat(o.borderBottomWidth),b=u+f,F=d+l,v=(r?0:parseFloat(o.borderLeftWidth))+m,W=g+w,y=r?0:t.offsetHeight-W-t.clientHeight,E=r?0:t.offsetWidth-v-t.clientWidth,R=b+v,z=F+W,M=r?t.width:parseFloat(o.width)-R-E,O=r?t.height:parseFloat(o.height)-z-y;if(n.has(t)){var k=n.get(t);if(k[0]===M&&k[1]===O)return}n.set(t,[M,O]);var S=Object.create(h.prototype);S.target=t,S.contentRect=new c(u,d,M,O),s.has(i)||(s.set(i,[]),p.add(i)),s.get(i).push(S)}))})),p.forEach((function(e){i.get(e).call(e,s.get(e),e)}))}return s.prototype.observe=function(i){if(i instanceof window.Element){r.has(i)||(r.set(i,new Set),o.add(i),a.set(i,window.getComputedStyle(i)));var n=r.get(i);n.has(this)||n.add(this),cancelAnimationFrame(t),t=requestAnimationFrame(d)}},s.prototype.unobserve=function(i){if(i instanceof window.Element&&r.has(i)){var n=r.get(i);n.has(this)&&(n.delete(this),n.size||(r.delete(i),o.delete(i))),n.size||r.delete(i),o.size||cancelAnimationFrame(t)}},A.DOMRectReadOnly=c,A.ResizeObserver=s,A.ResizeObserverEntry=h,A}; // eslint-disable-line\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\", \"Left button pans, Right button zooms\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\\nx/y fixes axis, CTRL fixes aspect\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
        "\n",
        "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
        "\n",
        "mpl.default_extension = \"png\";/* global mpl */\n",
        "\n",
        "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 = document.getElementById(id);\n",
        "    var ws_proxy = comm_websocket_adapter(comm);\n",
        "\n",
        "    function ondownload(figure, _format) {\n",
        "        window.open(figure.canvas.toDataURL());\n",
        "    }\n",
        "\n",
        "    var fig = new mpl.figure(id, ws_proxy, ondownload, element);\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;\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",
        "    fig.cell_info[0].output_area.element.on(\n",
        "        'cleared',\n",
        "        { fig: fig },\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_close = function (fig, msg) {\n",
        "    var width = fig.canvas.width / fig.ratio;\n",
        "    fig.cell_info[0].output_area.element.off(\n",
        "        'cleared',\n",
        "        fig._remove_fig_handler\n",
        "    );\n",
        "    fig.resizeObserverInstance.unobserve(fig.canvas_div);\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.innerHTML =\n",
        "        '<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 / this.ratio;\n",
        "    var dataURL = this.canvas.toDataURL();\n",
        "    this.cell_info[1]['text/html'] =\n",
        "        '<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 () {\n",
        "        fig.push_to_output();\n",
        "    }, 1000);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._init_toolbar = function () {\n",
        "    var fig = this;\n",
        "\n",
        "    var toolbar = document.createElement('div');\n",
        "    toolbar.classList = 'btn-toolbar';\n",
        "    this.root.appendChild(toolbar);\n",
        "\n",
        "    function on_click_closure(name) {\n",
        "        return function (_event) {\n",
        "            return fig.toolbar_button_onclick(name);\n",
        "        };\n",
        "    }\n",
        "\n",
        "    function on_mouseover_closure(tooltip) {\n",
        "        return function (event) {\n",
        "            if (!event.currentTarget.disabled) {\n",
        "                return fig.toolbar_button_onmouseover(tooltip);\n",
        "            }\n",
        "        };\n",
        "    }\n",
        "\n",
        "    fig.buttons = {};\n",
        "    var buttonGroup = document.createElement('div');\n",
        "    buttonGroup.classList = 'btn-group';\n",
        "    var button;\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",
        "            /* Instead of a spacer, we start a new button group. */\n",
        "            if (buttonGroup.hasChildNodes()) {\n",
        "                toolbar.appendChild(buttonGroup);\n",
        "            }\n",
        "            buttonGroup = document.createElement('div');\n",
        "            buttonGroup.classList = 'btn-group';\n",
        "            continue;\n",
        "        }\n",
        "\n",
        "        button = fig.buttons[name] = document.createElement('button');\n",
        "        button.classList = 'btn btn-default';\n",
        "        button.href = '#';\n",
        "        button.title = name;\n",
        "        button.innerHTML = '<i class=\"fa ' + image + ' fa-lg\"></i>';\n",
        "        button.addEventListener('click', on_click_closure(method_name));\n",
        "        button.addEventListener('mouseover', on_mouseover_closure(tooltip));\n",
        "        buttonGroup.appendChild(button);\n",
        "    }\n",
        "\n",
        "    if (buttonGroup.hasChildNodes()) {\n",
        "        toolbar.appendChild(buttonGroup);\n",
        "    }\n",
        "\n",
        "    // Add the status bar.\n",
        "    var status_bar = document.createElement('span');\n",
        "    status_bar.classList = 'mpl-message pull-right';\n",
        "    toolbar.appendChild(status_bar);\n",
        "    this.message = status_bar;\n",
        "\n",
        "    // Add the close button to the window.\n",
        "    var buttongrp = document.createElement('div');\n",
        "    buttongrp.classList = 'btn-group inline pull-right';\n",
        "    button = document.createElement('button');\n",
        "    button.classList = 'btn btn-mini btn-primary';\n",
        "    button.href = '#';\n",
        "    button.title = 'Stop Interaction';\n",
        "    button.innerHTML = '<i class=\"fa fa-power-off icon-remove icon-large\"></i>';\n",
        "    button.addEventListener('click', function (_evt) {\n",
        "        fig.handle_close(fig, {});\n",
        "    });\n",
        "    button.addEventListener(\n",
        "        'mouseover',\n",
        "        on_mouseover_closure('Stop Interaction')\n",
        "    );\n",
        "    buttongrp.appendChild(button);\n",
        "    var titlebar = this.root.querySelector('.ui-dialog-titlebar');\n",
        "    titlebar.insertBefore(buttongrp, titlebar.firstChild);\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._remove_fig_handler = function (event) {\n",
        "    var fig = event.data.fig;\n",
        "    if (event.target !== this) {\n",
        "        // Ignore bubbled events from children.\n",
        "        return;\n",
        "    }\n",
        "    fig.close_ws(fig, {});\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._root_extra_style = function (el) {\n",
        "    el.style.boxSizing = 'content-box'; // override notebook setting of border-box.\n",
        "};\n",
        "\n",
        "mpl.figure.prototype._canvas_extra_style = function (el) {\n",
        "    // this is important to make the div 'focusable\n",
        "    el.setAttribute('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",
        "    } else {\n",
        "        // location in version 2\n",
        "        IPython.keyboard_manager.register_events(el);\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",
        "\n",
        "    // Check for shift+enter\n",
        "    if (event.shiftKey && event.which === 13) {\n",
        "        this.canvas_div.blur();\n",
        "        // select the cell after this one\n",
        "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
        "        IPython.notebook.select(index + 1);\n",
        "    }\n",
        "};\n",
        "\n",
        "mpl.figure.prototype.handle_save = function (fig, _msg) {\n",
        "    fig.ondownload(fig, null);\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(\n",
        "        'matplotlib',\n",
        "        mpl.mpl_figure_comm\n",
        "    );\n",
        "}\n"
       ],
       "text/plain": [
        "<IPython.core.display.Javascript object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     },
     {
      "data": {
       "text/html": [
-       "<img src=\"\" width=\"900\">"
+       "<div id='95d66bfb-a49f-488f-9966-6c19cdd13b4a'></div>"
       ],
       "text/plain": [
        "<IPython.core.display.HTML object>"
       ]
      },
      "metadata": {},
      "output_type": "display_data"
     }
    ],
    "source": [
     "if True:\n",
     "    fig=plt.figure(figsize=(9,4))\n",
     "    fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)\n",
     "    sub = fig.add_subplot(1,1,1)\n",
     "\n",
     "    T=np.logspace(-5,5,500)\n",
-    "    s=[Cosmo.s(i) for i in T]\n",
+    "    s=[cosmo.s(i) for i in T]\n",
     "    sub.plot(T,s,linestyle='-',c='xkcd:black')#,label=r\"$g_{\\rm eff} (T)$\")\n",
     "    sub.set_ylabel(r'$s ~ [{\\rm GeV}^3]$')\n",
     "    sub.set_yscale('log')\n",
     "    sub.set_xscale('log')\n",
     "\n",
     "    sub.set_xlabel(r'$T \\; [{\\rm GeV}]$')\n",
     "\n",
     "    fig.show()"
    ]
   },
   {
    "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.9.5"
   }
  },
  "nbformat": 4,
  "nbformat_minor": 2
 }
diff --git a/UserSpace/JupyterNotebooks/WKB.ipynb b/UserSpace/JupyterNotebooks/WKB.ipynb
index 41c1729..d2274c3 100755
--- a/UserSpace/JupyterNotebooks/WKB.ipynb
+++ b/UserSpace/JupyterNotebooks/WKB.ipynb
@@ -1,204 +1,133 @@
 {
  "cells": [
   {
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
    "outputs": [],
    "source": [
     "import numpy as np#you usually need numpy\n",
     "\n",
     "#---these are for plots---#\n",
     "import matplotlib\n",
     "matplotlib.use('nbAgg')\n",
     "import matplotlib.pyplot as plt\n",
     "\n",
     "plt.rcParams['font.size']=16\n",
     "plt.rcParams['font.family']='dejavu sans'\n",
     "\n",
     "plt.rcParams['mathtext.fontset']='stix'\n",
     "plt.rcParams['mathtext.rm']='custom'\n",
     "plt.rcParams['mathtext.it']='stix:italic'\n",
     "plt.rcParams['mathtext.bf']='stix:bold'\n",
     "#-------------------------#"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 2,
    "metadata": {
     "scrolled": true
    },
    "outputs": [],
    "source": [
     "#load the module\n",
     "from sys import path as sysPath\n",
-    "from os import path as osPath\n",
-    "sysPath.append(osPath.join(osPath.dirname('./'), '../../src'))\n",
+    "sysPath.append('../../src')\n",
     "\n",
     "from interfacePy.WKB import WKB\n",
     "from interfacePy.Axion import Axion \n",
     "from interfacePy.AxionMass import AxionMass \n",
-    "from interfacePy.Cosmo import s \n",
+    "from interfacePy.Cosmo import Cosmo\n",
     "\n",
     "from interfacePy.FT import FT #easy tick formatting\n"
    ]
   },
   {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "One can use the WKB approximation, in order to estimate the relic abundance using ```WKB.relic```, which takes as\n",
-    "$T_{\\rm osc}$, $\\theta_{\\rm osc}$, and $\\gamma$ (the entropy injection from $T_{\\rm osc}$ until today).\n",
-    "\n",
-    "For $\\theta_{\\rm osc}$ one usually uses $\\theta_{\\rm ini}$, or can use ```WKB.theta_osc``` for a firs order approximation. This function takes \n",
-    "\n",
-    "$T_{\\rm ini}$, $3H/m_a$ at $T=T_{\\rm ini}$, $T_{\\rm osc}$, $\\theta_{\\rm ini}$, $\\gamma_{\\rm osc}$ (the entropy injection between $T_{\\rm ini}$ and $T_{\\rm osc}$).\n"
-   ]
-  },
-  {
    "cell_type": "code",
    "execution_count": 3,
    "metadata": {},
    "outputs": [],
    "source": [
     "theta_ini, fa=0.94435, 1e12\n",
     "\n",
     "CosmoFile='../InputExamples/RDinput.dat'\n",
     "# CosmoFile='../InputExamples/MatterInput.dat'\n",
     "# CosmoFile='../InputExamples/KinationInput.dat'\n",
     "\n",
     "\n",
     "axionMass = AxionMass(r'../../src/data/chi.dat',0,1e5)\n",
     "#------------------------------------------------------------------------------#\n",
     "# this is the axion mass squared beyond the interpolation limits for the current data \n",
     "# if yo don't specify it, the axion mass is taken to be constant beyond these limits\n",
     "TMax=axionMass.getTMax() \n",
     "chiMax=axionMass.getChiMax()\n",
     "TMin=axionMass.getTMin() \n",
     "chiMin=axionMass.getChiMin()\n",
     "\n",
     "axionMass.set_ma2_MAX( lambda T,fa: chiMax/fa/fa*pow(T/TMax,-8.16) )\n",
     "axionMass.set_ma2_MIN( lambda T,fa: chiMin/fa/fa )\n",
     "#------------------------------------------------------------------------------#\n",
-    "\n",
-    "\n",
-    "gamma_osc, gamma, Tosc, Tini,ratio_ini=WKB.getPoints(5,1.5,fa,axionMass.ma2,CosmoFile)"
+    "\n"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 4,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "0.11935022004089332\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
-    "#numerical result\n",
-    "ax =Axion(theta_ini,fa,500,1e-4,1e3,5,1e-2,CosmoFile,axionMass)\n",
-    "ax.solveAxion()\n",
-    "print(ax.relic)\n",
-    "MiMeS_theta_osc=ax.theta_osc\n",
-    "del ax"
+    "gamma_osc, gamma, Tosc, Tini, ratio_ini = WKB.getPoints(10,5,fa,axionMass.ma2,CosmoFile)"
    ]
   },
   {
    "cell_type": "code",
    "execution_count": 5,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
        "0.07152932609370595"
       ]
      },
      "execution_count": 5,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
-    "#just WKB\n",
+    "#WKB\n",
     "WKB.relic(Tosc,theta_ini,axionMass.ma2,gamma)"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "0.06841468669903171"
-      ]
-     },
-     "execution_count": 6,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "#WKB with approximate theta_osc\n",
-    "theta_osc=WKB.theta_osc(Tini, ratio_ini, Tosc, theta_ini, gamma_osc)\n",
-    "WKB.relic(Tosc,theta_osc,axionMass.ma2,gamma)"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 7,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "0.0646097119884361"
-      ]
-     },
-     "execution_count": 7,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "#WKB with numerical result for theta_osc\n",
-    "WKB.relic(Tosc,MiMeS_theta_osc,axionMass.ma2,gamma)"
-   ]
-  },
-  {
-   "cell_type": "code",
    "execution_count": null,
    "metadata": {},
    "outputs": [],
    "source": []
   }
  ],
  "metadata": {
   "kernelspec": {
-   "display_name": "Python 3 (ipykernel)",
+   "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.8.8"
+   "version": "3.9.5"
   }
  },
  "nbformat": 4,
  "nbformat_minor": 2
 }
diff --git a/UserSpace/Python/Axion.py b/UserSpace/Python/Axion.py
index 66a0143..11ad702 100755
--- a/UserSpace/Python/Axion.py
+++ b/UserSpace/Python/Axion.py
@@ -1,193 +1,197 @@
 from time import time
 from sys import stderr
 
 from sys import path as sysPath
 from os import path as osPath
 sysPath.append(osPath.join(osPath.dirname(__file__), '../../src'))
 
 #load the module
 from interfacePy.Axion import Axion 
 from interfacePy.AxionMass import AxionMass 
-from interfacePy.Cosmo import Hubble,mP
+from interfacePy.Cosmo import Cosmo
+
+
+Hubble=Cosmo().Hubble
+mP=Cosmo().mP
 
 theta_i, fa=0.94435, 1e12
 
 umax=500
 TSTOP=1e-4
 ratio_ini=1e3
 
 N_convergence_max, convergence_lim=5, 1e-2 #this is fine, but you can experiment a bit. 
 
 #radiation dominated example
 inputFile="../InputExamples/RDinput.dat" 
 
 # Matter Domination example. 
 # the NSC parameters (using the notation of 2012.07202) are:
 # T_end=1e-2 (GeV), c=3, T_ini=1e12 (GeV), and r=1e-1
 # inputFile="../InputExamples/MatterInput.dat" 
 
 # Kination Domination example. 
 # the NSC parameters (using the notation of 2012.07202) are:
 # T_end=0, c=6, T_ini=1e3 (GeV), and r=1e10
 # inputFile="../InputExamples/KinationInput.dat" 
 
 
 
 # options for the solver
 # These variables are optional. Yoou can use the Axion class without them.
 initial_step_size=1e-2; #initial step the solver takes. 
 minimum_step_size=1e-8; #This limits the sepsize to an upper limit. 
 maximum_step_size=1e-2; #This limits the sepsize to a lower limit.
 absolute_tolerance=1e-8; #absolute tolerance of the RK solver
 relative_tolerance=1e-8; #relative tolerance of the RK solver
 beta=0.9; #controls how agreesive the adaptation is. Generally, it should be around but less than 1.
 
 #The stepsize does not increase more than fac_max, and less than fac_min. 
 #This ensures a better stability. Ideally, fac_max=inf and fac_min=0, but in reality one must 
 #tweak them in order to avoid instabilities.
 fac_max=1.2; 
 fac_min=0.8;
 maximum_No_steps=int(1e7); #maximum steps the solver can take Quits if this number is reached even if integration is not finished.
 
 _=time()
 # AxionMass instance
 axionMass = AxionMass(r'../../src/data/chi.dat',0,mP)
 #------------------------------------------------------------------------------#
 # this is the axion mass squared beyond the interpolation limits for the current data 
 # if yo don't specify it, the axion mass is taken to be constant beyond these limits
 TMax=axionMass.getTMax() 
 chiMax=axionMass.getChiMax()
 TMin=axionMass.getTMin() 
 chiMin=axionMass.getChiMin()
 
 axionMass.set_ma2_MAX( lambda T,fa: chiMax/fa/fa*pow(T/TMax,-8.16) )
 axionMass.set_ma2_MIN( lambda T,fa: chiMin/fa/fa )
 #------------------------------------------------------------------------------#
 # def ma2(T,fa):
 #     TQCD=150*1e-3
 #     ma20=3.1575e-05/fa/fa
 #     if T<=TQCD:
 #         return ma20
 #     return ma20*pow((TQCD/T),8.16)
 # axionMass = AxionMass(ma2)
 
 # Axion instance
 ax=Axion(theta_i, fa, umax, TSTOP, ratio_ini, N_convergence_max, convergence_lim, inputFile, axionMass,
         initial_step_size,minimum_step_size, maximum_step_size, absolute_tolerance, 
         relative_tolerance, beta, fac_max, fac_min, maximum_No_steps)
 
 
 # solve the EOM (this only gives you the relic, T_osc, theta_osc, and a_osc)
 ax.solveAxion()
 
 print(ax.theta_i, ax.fa, ax.theta_osc, ax.T_osc ,ax.relic)
 print(round(time()-_,3),file=stderr)
 
 
 
 # change to True in order to mkae the plots
 if False:
     import matplotlib.pyplot as plt
     from numpy import array as np_array
     from numpy import abs as np_abs
 
     ax.getPeaks()#this gives you the peaks of the oscillation
     ax.getPoints()#this gives you all the points of integration
     ax.getErrors()#this gives you local errors of integration
 
 
     #########-----\theta-----#########
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)
     sub = fig.add_subplot(1,1,1)
     
     #this plot shows the peaks of the oscillation
     sub.plot(ax.T_peak,ax.theta_peak,linestyle=':',marker='+',color='xkcd:blue',linewidth=2)
 
     #this plot shows all the points
     sub.plot(ax.T,ax.theta,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')
 
     
     sub.set_yscale('linear')
     sub.set_xscale('linear')
     
     sub.set_xlabel(r'$T ~[{\rm GeV}]$')
     sub.xaxis.set_label_coords(0.5, -0.1) 
     sub.set_ylabel(r'$\theta$')
     sub.yaxis.set_label_coords(-0.1,0.5) 
 
     sub.axhline(ax.theta_osc,linestyle=':',color='xkcd:red',linewidth=1.5)
     sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)
    
     fig.savefig('theta-T_examplePlot.pdf',bbox_inches='tight')
 
 
 
     #########-----\zeta-----#########
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)
     sub = fig.add_subplot(1,1,1)
     
     sub.plot(ax.T,ax.zeta,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')
     sub.plot(ax.T_peak,ax.zeta_peak,linestyle=':',marker='+',color='xkcd:blue',linewidth=2)
     
     sub.set_yscale('linear')
     sub.set_xscale('linear')
     
     sub.set_xlabel(r'$T ~[{\rm GeV}]$')
     sub.xaxis.set_label_coords(0.5, -0.1) 
     sub.set_ylabel(r'$\zeta $')
     sub.yaxis.set_label_coords(-0.1,0.5) 
 
     sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)
     fig.savefig('zeta-T_examplePlot.pdf',bbox_inches='tight')
 
 
     #########-----errors-----#########
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)
     sub = fig.add_subplot(1,1,1)
     
     sub.plot(ax.T,np_abs(ax.dtheta/ax.theta),linestyle='-',linewidth=2,alpha=1,c='xkcd:black',label=r'$\dfrac{\delta \theta}{\theta}$')
     sub.plot(ax.T,np_abs(ax.dzeta/ax.zeta),linestyle='-',linewidth=2,alpha=1,c='xkcd:red',label=r'$\dfrac{\delta \zeta}{\zeta}$')
     
     sub.set_yscale('log')
     sub.set_xscale('linear')
     
     sub.set_xlabel(r'$T ~[{\rm GeV}]$')
     sub.xaxis.set_label_coords(0.5, -0.1) 
     sub.set_ylabel(r'local errors')
     sub.yaxis.set_label_coords(-0.1,0.5) 
     sub.legend(bbox_to_anchor=(1, 0.0),borderaxespad=0., 
            borderpad=0.05,ncol=1,loc='lower right',fontsize=14,framealpha=0)
 
     sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)
     fig.savefig('errors_examplePlot.pdf',bbox_inches='tight')
 
 
 
     #########-----rho_a-----#########    
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.9, right=0.9,wspace=0.0,hspace=0.25)
     sub = fig.add_subplot(1,1,1)
     
     sub.plot(ax.T,ax.rho_axion,linestyle='-',linewidth=2,alpha=1,c='xkcd:black')
     sub.plot(ax.T_peak,ax.rho_axion_peak,linestyle=':',linewidth=2,alpha=1,c='xkcd:blue')
 
     sub.set_yscale('log')
     sub.set_xscale('linear')
     
     sub.set_xlabel(r'$T ~[{\rm GeV}]$')
     sub.xaxis.set_label_coords(0.5, -0.1) 
     sub.set_ylabel(r'$\rho_{a}(T) ~[{\rm GeV}^{4}]$')
     sub.yaxis.set_label_coords(-0.1,0.5) 
     
     
     sub.axvline(ax.T_osc,linestyle='--',color='xkcd:gray',linewidth=1.5)
     fig.savefig('rho_a-T_examplePlot.pdf',bbox_inches='tight')
 
 
 
 
 #don't forget the destructor
 del ax
\ No newline at end of file
diff --git a/UserSpace/Python/Cosmo.py b/UserSpace/Python/Cosmo.py
index 9c550de..5cb19c8 100755
--- a/UserSpace/Python/Cosmo.py
+++ b/UserSpace/Python/Cosmo.py
@@ -1,94 +1,95 @@
 from numpy import logspace
 
 
 from sys import path as sysPath
 from os import path as osPath
 sysPath.append(osPath.join(osPath.dirname(__file__), '../../src'))
 
 #load the module
-import interfacePy.Cosmo as Cosmo
+from interfacePy import Cosmo 
 
+cosmo=Cosmo()
 
 
 for T in logspace(-5,5,50):
     print(
         'T=',T,'GeV\t',
-        'H=',Cosmo.Hubble(T),'GeV\t',
-        'h_eff=',Cosmo.heff(T),'\t',
-        'g_eff=',Cosmo.geff(T),'\t',
-        's=',Cosmo.s(T),'GeV^3\t',
+        'H=',cosmo.Hubble(T),'GeV\t',
+        'h_eff=',cosmo.heff(T),'\t',
+        'g_eff=',cosmo.geff(T),'\t',
+        's=',cosmo.s(T),'GeV^3\t',
     )
 
 
 
 if False:
     import matplotlib.pyplot as plt
 
     #########-----g_eff and h_eff-----#########
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)
     fig.suptitle('')
 
     sub = fig.add_subplot(1,1,1)
 
     T=logspace(-5,5,500)
-    gt=[Cosmo.geff(i) for i in T]
-    ht=[Cosmo.heff(i) for i in T]
+    gt=[cosmo.geff(i) for i in T]
+    ht=[cosmo.heff(i) for i in T]
 
     sub.plot(T,gt,linestyle='--',c='xkcd:red',label=r"$g_{\rm eff} (T)$")
     sub.plot(T,ht,linestyle=':',c='xkcd:black',label=r"$h_{\rm eff} (T)$")
 
     sub.set_xlabel(r'$T ~ [{\rm GeV}]$')
     sub.set_ylabel(r'rel. dof')
 
     sub.legend(bbox_to_anchor=(1, 0.0),borderaxespad=0., 
             borderpad=0.05,ncol=1,loc='lower right',fontsize=14,framealpha=0)
     sub.set_yscale('log')
     sub.set_xscale('log')
 
     fig.savefig('rdofs-T_examplePlot.pdf',bbox_inches='tight')
 
 
 
     #########-----dg_effdT and dh_effdT-----#########
 
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)
     fig.suptitle('')
 
     sub = fig.add_subplot(1,1,1)
 
     T=logspace(-5,5,500)
-    dg=[Cosmo.dgeffdT (i) for i in T]
-    dh=[Cosmo.dheffdT(i) for i in T]
+    dg=[cosmo.dgeffdT (i) for i in T]
+    dh=[cosmo.dheffdT(i) for i in T]
 
     sub.plot(T,dg,linestyle='--',c='xkcd:red',label=r"$\dfrac{d g_{\rm eff}}{dT} (T)$")
     sub.plot(T,dh,linestyle=':',c='xkcd:black',label=r"$\dfrac{d h_{\rm eff}}{dT} (T)$")
 
     sub.set_xlabel(r'$T ~ [{\rm GeV}]$')
 
     sub.legend(bbox_to_anchor=(1, 0.5),borderaxespad=0., 
             borderpad=0.05,ncol=1,loc='lower right',fontsize=14,framealpha=0)
     sub.set_yscale('symlog')
     sub.set_xscale('log')
 
     fig.savefig('drdofsdT-T_examplePlot.pdf',bbox_inches='tight')
 
 
     #########-----dh-----#########
 
     fig=plt.figure(figsize=(9,4))
     fig.subplots_adjust(bottom=0.15, left=0.15, top = 0.95, right=0.9,wspace=0.0,hspace=0.0)
     fig.suptitle('')
 
     sub = fig.add_subplot(1,1,1)
 
     T=logspace(-5,5,500)
-    dht=[Cosmo.dh(i) for i in T]
+    dht=[cosmo.dh(i) for i in T]
     sub.plot(T,dht,linestyle='-',c='xkcd:black') 
     sub.set_xlabel(r'$T ~ [{\rm GeV}]$')
     sub.set_ylabel(r'$\delta_h = 1 + \dfrac{1}{3} \dfrac{d \log h_{\rm eff} }{d \log T}$')
     sub.set_yscale('linear')
     sub.set_xscale('log')
 
     fig.savefig('dh-T_examplePlot.pdf',bbox_inches='tight')
diff --git a/UserSpace/scan/AxionScan/AxionScan.py b/UserSpace/scan/AxionScan/AxionScan.py
index 7175a6e..7ce23f1 100755
--- a/UserSpace/scan/AxionScan/AxionScan.py
+++ b/UserSpace/scan/AxionScan/AxionScan.py
@@ -1,47 +1,47 @@
 ###-----------------------------------------------------------------------------###
 ###-------------------Example of  interfacePy.ScanScript.Scan-------------------###
 ###-----------------------------------------------------------------------------###
 
 #------------------------------------Note:------------------------------------#
 ###########-Scan: scans for the entire theta_i and f_a one provides-###########
 #-----------------------------------------------------------------------------#
 
 from sys import path as sysPath
 from os import path as osPath
 
 sysPath.append(osPath.join(osPath.dirname(__file__), '../../../src'))
 
 from interfacePy.ScanScript import Scan 
 
 
 from numpy import logspace,linspace,log10,pi
 
 scan=Scan(
     cpus=8,
-    table_fa= logspace(10,19,50),
-    table_theta_i=linspace(0.1,2,50),
+    table_fa= logspace(10,19,150),
+    table_theta_i=linspace(0.1,2,150),
     umax=500,
     TSTOP=1e-4,
     ratio_ini=1e3,
     N_convergence_max=5,
     convergence_lim=1e-2,
     inputFile="../../InputExamples/RDinput.dat", 
     # inputFile="../../InputExamples/MatterInput.dat", 
     # inputFile="../../InputExamples/KinationInput.dat",
     PathToCppExecutable=r"../../Cpp/Axion/Axion.run",
     break_after=60*60*3,
     break_time=60,
     break_command='',
     initial_step_size=1e-2, 
     minimum_step_size=1e-8,
     maximum_step_size=1e-2,
     absolute_tolerance=1e-8,
     relative_tolerance=1e-8,
     beta=0.9,
     fac_max=1.2, 
     fac_min=0.8,
     maximum_No_steps=int(1e7)
 )
 
 
 scan.run()
\ No newline at end of file
diff --git a/UserSpace/scan/AxionScanObs/AxionScanObs.py b/UserSpace/scan/AxionScanObs/AxionScanObs.py
index e9f980d..e4a5a1c 100755
--- a/UserSpace/scan/AxionScanObs/AxionScanObs.py
+++ b/UserSpace/scan/AxionScanObs/AxionScanObs.py
@@ -1,77 +1,80 @@
 ###-----------------------------------------------------------------------------###
 ###-------------------Example of interfacePy.ScanScript.ScanObs-----------------###
 ###-----------------------------------------------------------------------------###
 
 #--------------------------------Note:--------------------------------#
 ###################-The scan proceeds as follows:-#####################
 #-For each value of $f_a$, we find $\theta_i^{approx}$ such that------#
 #-$Omega h^2 = 0.12$, assuming that $\theta_i^{approx} \ll 1$.--------#
 #-Then, it scans values of $\theta_i$ close to $\theta_i^{approx}$.---# 
 #-This results in a range of values of $\theta_i$ for each $f_a$, with# 
 #-$\Omega h^2$ close to the observed value.---------------------------#
 #---------------------------------------------------------------------#
 
 
 
 from sys import path as sysPath
 from os import path as osPath
 
 sysPath.append(osPath.join(osPath.dirname(__file__), '../../../src'))
 
 from interfacePy.ScanScript import ScanObs 
-from interfacePy.Cosmo import relicDM_obs,mP 
+from interfacePy.Cosmo import Cosmo 
 from interfacePy.AxionMass import AxionMass 
 
 
 
 from numpy import logspace
 
+cosmo=Cosmo()
+mP=cosmo.mP
+relicDM_obs=cosmo.relicDM_obs
 
 # AxionMass instance
 axionMass = AxionMass(r'../../../src/data/chi.dat',0,mP)
 #------------------------------------------------------------------------------#
 # this is the axion mass squared beyond the interpolation limits for the current data 
 # if yo don't specify it, the axion mass is taken to be constant beyond these limits
 TMax=axionMass.getTMax() 
 chiMax=axionMass.getChiMax()
 TMin=axionMass.getTMin() 
 chiMin=axionMass.getChiMin()
 
 axionMass.set_ma2_MAX( lambda T,fa: chiMax/fa/fa*pow(T/TMax,-8.16) )
 axionMass.set_ma2_MIN( lambda T,fa: chiMin/fa/fa )
 
 
 
 scan=ScanObs(
     cpus=8,
     table_fa= logspace(10,20,150),
-    len_theta=350,
+    len_theta=150,
     umax=500,
     TSTOP=1e-4,
     ratio_ini=1e3,
     N_convergence_max=10,
     convergence_lim=1e-1,
     inputFile="../../InputExamples/RDinput.dat", 
     # inputFile="../../InputExamples/MatterInput.dat", 
     # inputFile="../../InputExamples/KinationInput.dat",
     axionMass=axionMass,
     PathToCppExecutable=r"../../Cpp/Axion/Axion.run",
     relic_obs=relicDM_obs,
     relic_err_up=0.01,
     relic_err_low=0.01,
     break_after=60*60*3,
     break_time=60,
     break_command='',
     initial_step_size=1e-2, 
     minimum_step_size=1e-8,
     maximum_step_size=1e-2,
     absolute_tolerance=1e-8,
     relative_tolerance=1e-8,
     beta=0.9,
     fac_max=1.2, 
     fac_min=0.8,
     maximum_No_steps=int(1e7)
 )
 
 
 scan.run()
\ No newline at end of file
diff --git a/src/interfacePy/Cosmo/Cosmo.py b/src/interfacePy/Cosmo/Cosmo.py
index a93ef7f..3adb3d1 100644
--- a/src/interfacePy/Cosmo/Cosmo.py
+++ b/src/interfacePy/Cosmo/Cosmo.py
@@ -1,79 +1,95 @@
 from  ctypes import CDLL, c_longdouble, c_double
 
 from sys import path as sysPath
 from os import path as osPath
 
 sysPath.append(osPath.join(osPath.dirname(__file__), '../../'))
 
 from misc_dir.path import _PATH_
 from misc_dir.type import cdouble
 
 #load the library
 func = CDLL(_PATH_+'/lib/libCosmo.so')
 
 CppFunc=func.heff,func.geff,func.dgeffdT,func.dheffdT,func.dh,func.rhoR,func.Hubble,func.s  
 
 #specify the argument types
 for i,f in enumerate (CppFunc): 
     f.argtypes = cdouble,
     #specify the return type
     f.restype = cdouble
 ###############################################
 
-def heff(T):
-    '''h_eff at temperature T [GeV]'''
-    return func.heff(T)
-
-def geff(T):
-    '''g_eff at temperature T [GeV] '''
-    return func.geff(T)
-
-def dgeffdT(T):
-    '''\\dfrac{dg_eff}{dT} at temperature T [GeV]'''
-    return func.dgeffdT(T)
-
-def dheffdT(T):
-    '''\\dfrac{dh_eff}{dT} at temperature T [GeV]'''
-    return func.dheffdT(T)
-
-def dh(T):
-    '''\\delta_h(T)=1+1/3 \frac{d log h_eff}{d log T} at temperature T [GeV]'''
-    return func.dh(T)
-
-def rhoR(T):
-    '''energy density of the plasma at temperature T [GeV]'''
-    return func.rhoR(T)
-
-def Hubble(T): 
-    '''H at temperature T [GeV]'''
-    return func.Hubble(T)
-
-def s(T): 
-    '''s (entropy density of the plasma) at temperature T [GeV]'''
-    return func.s(T)
 
 func.getT0.argtypes =None
 func.getT0.restype =cdouble
 func.geth_hub.argtypes =None
 func.geth_hub.restype =cdouble
 func.getrho_crit.argtypes =None
 func.getrho_crit.restype =cdouble
 func.getrelicDM.argtypes =None
 func.getrelicDM.restype =cdouble
 func.getMP.argtypes =None
 func.getMP.restype =cdouble
 
-# CMB temperature today in GeV
-T0=func.getT0()
-# critical density today in GeV^4
-rho_crit=func.getrho_crit()
-# dimensionless hubble parameter
-h_hub=func.geth_hub()
-# central value of Omega h^2 according to Planck 
-relicDM_obs=func.getrelicDM()
-# Planck mass
-mP=func.getMP()
+
+
+class Cosmo:
+    '''Class that contains various cosmological parameters and functions.
+    The available functions are:
+    heff, geff, dgeffdT, dheffdT, dh, rhoR, Hubble, s. 
+    All are functions of the temperature (in GeV), and correspond to the standard comsological scenario.
+    The available variables are:
+    T0: CMB temperature today in GeV
+    rho_crit: critical density today in GeV^4
+    h_hub: dimensionless hubble parameter
+    relicDM_obs: central value of Omega h^2 according to Planck
+    mP: Planck mass in GeV
+    '''
+    def __init__(self):
+        # CMB temperature today in GeV
+        self.T0=func.getT0()
+        # critical density today in GeV^4
+        self.rho_crit=func.getrho_crit()
+        # dimensionless hubble parameter
+        self.h_hub=func.geth_hub()
+        # central value of Omega h^2 according to Planck 
+        self.relicDM_obs=func.getrelicDM()
+        # Planck mass in GeV
+        self.mP=func.getMP()
+
+    def heff(self,T):
+        '''h_eff at temperature T [GeV]'''
+        return func.heff(T)
+
+    def geff(self,T):
+        '''g_eff at temperature T [GeV] '''
+        return func.geff(T)
+
+    def dgeffdT(self,T):
+        '''\\dfrac{dg_eff}{dT} at temperature T [GeV]'''
+        return func.dgeffdT(T)
+
+    def dheffdT(self,T):
+        '''\\dfrac{dh_eff}{dT} at temperature T [GeV]'''
+        return func.dheffdT(T)
+
+    def dh(self,T):
+        '''\\delta_h(T)=1+1/3 \frac{d log h_eff}{d log T} at temperature T [GeV]'''
+        return func.dh(T)
+
+    def rhoR(self,T):
+        '''energy density of the plasma at temperature T [GeV]'''
+        return func.rhoR(T)
+
+    def Hubble(self,T): 
+        '''H at temperature T [GeV]'''
+        return func.Hubble(T)
+
+    def s(self,T): 
+        '''s (entropy density of the plasma) at temperature T [GeV]'''
+        return func.s(T)
 
 
 if __name__=="__main__":
     pass
\ No newline at end of file
diff --git a/src/interfacePy/Cosmo/__init__.py b/src/interfacePy/Cosmo/__init__.py
index 206c9ac..790951b 100644
--- a/src/interfacePy/Cosmo/__init__.py
+++ b/src/interfacePy/Cosmo/__init__.py
@@ -1,3 +1,3 @@
-from .Cosmo import heff, geff, dgeffdT, dheffdT, dh, rhoR, Hubble, s, T0, rho_crit, h_hub, relicDM_obs, mP
+from .Cosmo import Cosmo
 
 
diff --git a/src/interfacePy/ScanScript/ScanObs.py b/src/interfacePy/ScanScript/ScanObs.py
index 4989310..2b4bd15 100644
--- a/src/interfacePy/ScanScript/ScanObs.py
+++ b/src/interfacePy/ScanScript/ScanObs.py
@@ -1,267 +1,273 @@
 # Scan for axion in radiation dominated Uinverse.
 # 
 #------------------Note:------------------#
 # The scan proceeds as follows:
 #  The for each value of $f_a$, we find $\theta_i^{\rm approx}$ such that 
 # $\Omega h^2 = 0.12$, assuming that $sin(\theta_i^{\rm approx}) \approx \theta_i^{\rm approx}$. 
 # Then, we scan values of $\theta_i$ close to $\theta_i^{\rm approx}$. 
 # This results in a range of values of $\theta_i$ for each $f_a$, 
 # with $\Omega h^2$ close to the observed value.
 #-----------------------------------------#
 
 
 from sys import path as sysPath
 from os import path as osPath
 sysPath.append(osPath.join(osPath.dirname(__file__), '../'))
 sysPath.append(osPath.join(osPath.dirname(__file__), '../../src'))
 from misc_dir.path import _PATH_
 
 from interfacePy.Axion import Axion 
-from interfacePy.Cosmo import h_hub,T0,rho_crit,s 
+from interfacePy.Cosmo import Cosmo
+
+cosmo=Cosmo()
+h_hub=cosmo.h_hub
+T0=cosmo.T0
+rho_crit=cosmo.rho_crit
+s=cosmo.s
 
 
 
 from numpy import pi, sqrt, abs, array, float64, argmin, savetxt, linspace 
 
 
 from subprocess import check_output as subprocess_check_output
 from sys import stdout as sys_stdout
 from time import time
 from datetime import datetime
 from datetime import timedelta
 
 from os import remove as os_remove
 from os import system as os_system
 from pathlib import Path
 
 
 
 parallelScan=_PATH_+r"/src/util/parallel_scan.sh"
 
 
 class ScanObs:
     def __init__(self,cpus,table_fa,len_theta,umax,TSTOP,ratio_ini,N_convergence_max,convergence_lim,inputFile,axionMass,
                 PathToCppExecutable, relic_obs,relic_err_up,relic_err_low,break_after,break_time,break_command='',
                 initial_step_size=1e-2, minimum_step_size=1e-8, maximum_step_size=1e-2, 
                 absolute_tolerance=1e-8, relative_tolerance=1e-8,
                 beta=0.9, fac_max=1.2, fac_min=0.8, maximum_No_steps=int(1e7)):
         '''
         scan for different values of fa (in table_fa) and find the theta_i closer to relic_obs.
         The result file is timecoded (so it would be difficult to write over it), and the columns correspond to
         theta_i fa [GeV] theta_osc T_osc [GeV] relic (Omega h^2) 
         
         Comment 1: The way it works is the following:
         1) we solve for theta_i=1e-5.
         2) based on this, rescale it in order to find an appropriate theta_i such that Omegah^2 = relic_obs
         3) use this rescaled theta_i as initial point, and scan for len_theta between 
             np.linspace(min([theta_i*0.85,1]),min([theta_i*1.2,pi]),len_theta)
         4) finally, the point with Omegah^2 closer to relic_obs is stored.
         
         Comment 2: If the scan exits before it finishes, it will continue from the point it stopped.
         In order to start from the beginning, delete the file "count.dat".
         
         
         cpus: number of points to run simultaneously (No of cpus available). 
         table_fa: table of fa to scan
         len_theta: number of points to search for theta closest to relic_obs. This search happens in 
         np.linspace(min([theta_i*0.85,1]),min([theta_i*1.2,pi]),self.len_theta), with theta_i is the angle
         that results in Omega h^2=relic_obs assuming theta_i<<1.
 
         umax: if u>umax the integration stops (rempember that u=log(a/a_i))
         TSTOP: if the temperature drops below this, integration stops
         ratio_ini: integration starts when 3H/m_a<~ratio_ini (this is passed to AxionEOM, 
         in order to make the interpolations start at this point)
 
         N_convergence_max and convergence_lim: integration stops when the relative difference 
         between two consecutive peaks is less than convergence_lim for N_convergence_max 
         consecutive peaks
         
         inputFile: file that describes the cosmology. the columns should be: u T[GeV] logH     
         
         axionMass: instance of AxionMass
 
         PathToCppExecutable: path to an executable that takes "theta_i  fa  umax  TSTOP  ratio_ini  N_convergence_max convergence_lim inputFile"
         and prints "theta_i fa theta_osc T_osc relic"
 
         relic_obs: central value of the relic (example, relic_obs=0.12)
         relic_err_up: upper error bar around relic_obs (example, relic_err_up=0.001)
         relic_err_low: lower error bar around relic_obs (example, relic_err_low=0.001)
 
         break_after,break_time: take a break after break_after seconds for break_time seconds
         (optional) break_command: before it takes a break, run this system command (this may be a script to send the results
         to an e-mail, or back them up)
 
         -----------Optional arguments------------------------
         initial_stepsize: initial step the solver takes. 
 
         maximum_stepsize: This limits the sepsize to an upper limit. 
         minimum_stepsize: This limits the sepsize to a lower limit. 
         
         absolute_tolerance: absolute tolerance of the RK solver
 
         relative_tolerance: relative tolerance of the RK solver
         Note:
         Generally, both absolute and relative tolerances should be 1e-8. 
         In some cases, however, one may need more accurate result (eg if f_a is extremely high, 
         the oscillations happen violently, and the ODE destabilizes). Whatever the case, if the  
         tolerances are below 1e-8, long doubles *must* be used.
 
         beta: controls how agreesive the adaptation is. Generally, it should be around but less than 1.
         
         fac_max,  fac_min: the stepsize does not increase more than fac_max, and less than fac_min. 
         This ensures a better stability. Ideally, fac_max=inf and fac_min=0, but in reality one must 
         tweak them in order to avoid instabilities.
 
         maximum_No_steps: maximum steps the solver can take Quits if this number is reached even if integration
         is not finished. 
         '''
         
         self.cpus=cpus
         self.Table_fa=table_fa
         self.len_theta=len_theta
         
         self.umax=umax
         self.TSTOP=TSTOP
         self.ratio_ini=ratio_ini
         self.Npeaks=N_convergence_max
         self.conv=convergence_lim
         self.inputFile=inputFile
         self.axionMass=axionMass
         
         self.PathToCppExecutable=PathToCppExecutable
 
         self.relic_obs=relic_obs
         self.relic_err_up,self.relic_err_low = relic_err_up,relic_err_low
         
         self.break_after=break_after        
         self.break_time=break_time
         self.break_command=break_command
         
         self.initial_step_size=initial_step_size
         self.minimum_step_size=minimum_step_size
         self.maximum_step_size= maximum_step_size
         self.absolute_tolerance=absolute_tolerance
         self.relative_tolerance=relative_tolerance
         self.beta=beta
         self.fac_max=fac_max
         self.fac_min=fac_min
         self.maximum_No_steps=maximum_No_steps
 
 
         self.FileDate=datetime.now()
         self.FileName = "{}".format(self.FileDate.strftime('%d-%m-%Y_%H-%M-%S')) 
         self._p = Path(self.FileName+'.dat')
         
         
         self.in_file="in._mimes_"
         self.count_file="count._mimes_"
     def run_batch(self):
         '''
         run a batch.
         '''
         # get the result     
         points=subprocess_check_output( [parallelScan,  self.PathToCppExecutable, str(self.cpus),self.in_file]).decode(sys_stdout.encoding)
         points=points.split('\n')
         
 
         points=array([array(i.split(),float64)  for i in points[:-1] ])
         points=array(sorted(points, key=lambda arr: arr[0]))
         absDiff=argmin(abs(points[:,4]-self.relic_obs))
         
         _=points[absDiff]
         if _[4]<=self.relic_obs+self.relic_err_up and _[4]>=self.relic_obs-self.relic_err_low:  
             File= self._p.open('ab')            
             savetxt(File,array([_]))
             File.close()
 
     def run(self):
 
         try:
             with open(self.count_file,'r') as _:
                 last_batch=int(_.read())
         except:
             last_batch=0
             
         Total_batches=len(self.Table_fa)-last_batch
             
         totalT=0
         meanT=0
         ETA=0
         batch=0
         
         sleepTimer=0
         for fa in self.Table_fa[last_batch:]:
             time0=time()
 
             ########################---find theta_i (assuming theta_i<<1) such that Omega h^=0.12---########################
 
             theta_small=1e-3
             ax=Axion(theta_small,fa,self.umax,self.TSTOP,
             self.ratio_ini,self.Npeaks,self.conv,self.inputFile, self.axionMass, 
             self.initial_step_size, self.minimum_step_size, self.maximum_step_size, 
             self.absolute_tolerance, self.relative_tolerance, self.beta,
             self.fac_max, self.fac_min, self.maximum_No_steps)
 
             ax.solveAxion()
             ax.getPeaks()
             
             T=ax.T_peak[-1]
             theta=ax.theta_peak[-1]
 
             relic=ax.relic
             theta_obs=sqrt(theta**2/relic*self.relic_obs)
 
             relic=s(T0)/s(T)*0.5*sqrt(self.axionMass.ma2(0,1)*self.axionMass.ma2(T,1))*theta_obs**2*h_hub**2/rho_crit
 
             theta_small_i=theta_small*theta_obs/theta
             
             if theta_small_i < theta_small:
                 Table_theta_i = array([theta_small_i])
             else:
                 if theta_small_i>pi:
                     theta_small_i=pi
                 Table_theta_i=linspace(min([theta_small_i*0.85,1]),min([theta_small_i*1.2,pi]),self.len_theta)
                             
             del ax
             ########################---END---########################
             file=open(self.in_file , 'w')
             for theta_i in Table_theta_i :
                 file.write( '{0} {1} {2} {3} {4} {5} {6} {7} {8} {9} {10} {11} {12} {13} {14} {15} {16} \n'.
                 format(theta_i, fa, self.umax, self.TSTOP, self.ratio_ini, self.Npeaks, self.conv, self.inputFile,
                 self.initial_step_size, self.minimum_step_size, self.maximum_step_size, self.absolute_tolerance, self.relative_tolerance, self.beta,
                 self.fac_max, self.fac_min, self.maximum_No_steps) )
             file.close()
             
             self.run_batch()
 
             
             batch+=1
             totalT+=time()-time0
             meanT=totalT/batch
 
             if self.break_after>0:
                 sleepTimer+=time()-time0
                 if sleepTimer>self.break_after:
                     os_system(self.break_command)
                     print("taking a break")
                     time.sleep(self.break_time)
                     
                     sleepTimer=0
                 
             ETA=meanT*(Total_batches-batch)
 
             print('======================================== \n',
             'Completed batches:  ',batch,' out of ', Total_batches ,'\n',
             'Running for:     {0:.50s}'.format( str(timedelta(seconds=totalT))  ), '\n', 
             'ETA:             {0:.50s}'.format(str(timedelta(seconds=ETA) )  ),'\n', 
             'Time per batch:  {0:.3f} sec'.format(meanT),
             '\n========================================' )
             
             
             with open(self.count_file,'w') as _:
                 _.write(str(batch+last_batch))
 
         os_remove(self.in_file)
         os_remove(self.count_file)
         print('Done!')
     
\ No newline at end of file
diff --git a/src/interfacePy/WKB/WKB.py b/src/interfacePy/WKB/WKB.py
index 313a088..6d0a6da 100644
--- a/src/interfacePy/WKB/WKB.py
+++ b/src/interfacePy/WKB/WKB.py
@@ -1,66 +1,53 @@
-from ..Cosmo import Hubble,s,T0, rho_crit, h_hub,heff,geff
+from ..Cosmo import Cosmo
 from ..AxionMass import AxionMass
 from numpy import sqrt,loadtxt,vectorize,array,exp
 
 
 
+cosmo=Cosmo()
+
 def relic(Tosc,theta_osc,ma2,gamma=1.):
     '''
     The axion relic abundance using the WKB approximation.
     Tosc: the oscillation temperature
     theta_osc:the  angle at T_osc (usually one uses theta_osc = theta_ini)
     gamma: entropy ratio betweem Tosc and today (gamma = S(T0)/S(Tosc)) 
     '''
 
     correction=(3/4)**(0.5)#this factor gives more acurate form of WKB result
-    return   s(T0)/s(Tosc)/gamma*0.5*sqrt(ma2(0,1)*ma2(Tosc,1))*theta_osc**2*h_hub**2/rho_crit *correction
-
-
-
-def theta_osc(Tini,ratio_ini,Tosc,theta_ini,gamma_osc=1):
-    '''
-    The first approximation for theta_osc.
-    Tini=initial temperature . In this approximation it has to be as close to T_osc as possible, 
-          while dtheta/dt ~= 0.
-    ratio_ini: 3H/ma at T=Tini.
-    Tosc: the oscillation temperature
-    theta_ini:the angle at Tini
-    gamma_osc: entropy ratio betweem Tini and Tosc (gamma_osc = S(Tosc)/S(Tini)) 
-    '''
-    return theta_ini*(1- (ratio_ini/3.)**(-2.)*( 1-Tini/Tosc*( heff(Tini)/heff(Tosc)*gamma_osc )**(1/3.) )**2)#work in progress
-
+    return   cosmo.s(cosmo.T0)/cosmo.s(Tosc)/gamma*0.5*sqrt(ma2(0,1)*ma2(Tosc,1))*theta_osc**2*cosmo.h_hub**2/cosmo.rho_crit *correction
 
 
 
 
 def getPoints(T_start,ratio_ini,fa,ma2,inputFile):
     '''find the points you need for Tosc, gamma, and gamma osc 
     T_start: some initial temperature (this just help to start searching for an appropriate Tini)
     ratio_ini: 3H/ma at Tini (it has to be close to 1 for the theta_osc approximation to work). This is used to find Tini
     fa: PQ scale
     inputFile: a file that contains u,T,logH forthe cosmology of interest
     
     this function returns gamma_osc, gamma, Tosc, Tini, and ratio_ini (this is close to the inut's value, but corresponts to the closest point in inputFile)
     '''
     _=loadtxt(inputFile)
     cosmology=_[_[:,1]<T_start]
     u=cosmology[:,0]
     T=cosmology[:,1]
     logH=cosmology[:,2]
     ma=vectorize(lambda T:ma2(T,fa)**0.5)
     ratio=3*exp(logH)/ma(T)
 
     ch=ratio<=ratio_ini
     tmp=cosmology[ch][0]
     Tini=tmp[1]
     uini=tmp[0]
     logHini=tmp[2]
     ratio_ini=3*exp(logHini)/ma(Tini)
     
     ch=ratio<=1
     tmp=cosmology[ch][0]
     Tosc=tmp[1]
     uosc=tmp[0]
     
     
-    return s(Tosc)/s(Tini)*exp(3*(uosc-uini)),s(cosmology[-1][1])/s(Tosc)*exp(3*(cosmology[-1][0]-uosc)),Tosc,Tini,ratio_ini         
+    return cosmo.s(Tosc)/cosmo.s(Tini)*exp(3*(uosc-uini)),cosmo.s(cosmology[-1][1])/cosmo.s(Tosc)*exp(3*(cosmology[-1][0]-uosc)),Tosc,Tini,ratio_ini         
diff --git a/src/interfacePy/WKB/__init__.py b/src/interfacePy/WKB/__init__.py
index dd9a788..f82160a 100644
--- a/src/interfacePy/WKB/__init__.py
+++ b/src/interfacePy/WKB/__init__.py
@@ -1,2 +1,2 @@
-from .WKB import relic, theta_osc, getPoints
+from .WKB import relic, getPoints
 
diff --git a/src/interfacePy/__init__.py b/src/interfacePy/__init__.py
index e69de29..227e109 100644
--- a/src/interfacePy/__init__.py
+++ b/src/interfacePy/__init__.py
@@ -0,0 +1,10 @@
+from .Axion import Axion 
+from .AxionMass import AxionMass 
+from .AnharmonicFactor import anharmonicFactor
+from .Cosmo import Cosmo
+from .WKB import relic, getPoints
+
+from .ScanScript.ScanObs import ScanObs
+from .ScanScript.Scan import Scan
+
+from FT import FT
\ No newline at end of file