{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "# 7. Orbit Distributions\n", "\n", "In this notebook we'll take a closer look at orbit distributions. To run this notebook, you will already need to have created some models for the example galaxy NGC6278. To do this, first run the notebook `2_quickstart` and/or `3_model_iterations_and_plots`." ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "import dynamite as dyn\n", "\n", "import numpy as np\n", "import matplotlib.pyplot as plt\n", "\n", "print('DYNAMITE version', dyn.__version__)\n", "#print(' installed at ', dyn.__path__) # Uncomment to print the complete DYNAMITE installation path\n", "\n", "fname = 'NGC6278_config.yaml'\n", "c = dyn.config_reader.Configuration(fname, reset_logging=True)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## 1. Get the dynamite `Model` object\n", "\n", "First let's pick a model whose orbit distribution we want to construct. Let's look at a list of all models:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "c.all_models.table" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Find the row index of the best model," ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "best_model_idx = c.all_models.get_best_n_models_idx(1)[0]\n", "print(f'Index of best model: {best_model_idx}')" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "and then get a dynamite `Model` object corresponding to that row:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "mod = c.all_models.get_model_from_row(best_model_idx)\n", "mod" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Let's then load the orbit library and (already solved for) orbit weights for this model" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "orblib = mod.get_orblib()\n", "\n", "weight_solver = mod.get_weights(orblib)\n", "weights = mod.weights" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## 3. Orbit distribution plots" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Let's run the plotting routine to make the orbit distribution plot:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "plotter = dyn.plotter.Plotter(c)\n", "\n", "# define settings\n", "minr, maxr = 0.1, 10. # radius limits in kpc (to be binned in log r)\n", "nr, nl = 6, 7 # use 6 bins in radius, 7 bins in circularity lambda\n", "\n", "fig = plotter.orbit_distribution(mod, nr=nr, nl=nl, minr=minr, maxr=maxr)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "This plot shows the orbit-distributions split into four panels corresponding to four orbit-types. Each distribution shows a density in log radius (y-axis) against a circularity parameter $\\lambda$ (x-axis). Which component of the circularity is shown depends on the orbit type:\n", "- long-axis tube: $\\lambda_x$\n", "- intermediate-axis tube: $\\lambda_y$\n", "- short-axis tube: $\\lambda_z$\n", "- box: $\\lambda_\\mathrm{tot}$, using the total angular momentum (which should be zero for box orbits).\n", "\n", "The main difference between this orbit-distribution plot and previous ones, which showed all orbit types in $(r,\\lambda_z)$ space, is that by splitting by orbit-type we can distinguish long-axis tubes from box orbits. Restricting to $(r,\\lambda_z)$ space overlays these two orbit-types since they both have $\\lambda_z=0$. Intermediate-axis orbits are unstable, and therefore should form a negligible contribution; they are left in this plot mainly as a diagnostic for the orbit classification, more details of which are given below.\n", "\n", "Finally, note that if you want to limit which orbit types are shown, you can use the `subset`keyword argument of `plotter.orbit_distribution` along with many other options to change its behaivour.\n", "\n", "## 3. Recreating orbit distributions using `orblib.projection_tensor` \n", "\n", "To make the orbit distribution plot, when we call `plotter.orbit_distribution`, a method called `orblib.get_projection_tensor` is run internally. Let's run this ourselves in order to get access to the `orblib.projection_tensor` object which we can use to recreate the orbit distributions shown in the plots above:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "orblib.get_projection_tensor(nr=nr, nl=nl, minr=minr, maxr=maxr) # use same settings as before\n", "orblib.projection_tensor" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The `orblib.projection_tensor` is stored as a sparse object array to speed up some calculations. For simplicity, let's convert it to a regular `numpy` array using the `todense()` method:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "projection_tensor = orblib.projection_tensor.todense()\n", "projection_tensor.shape" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The `projection_tensor` has dimension 4 and shape (4, 6, 7, 360). These 4 dimensions correspond to:\n", "\n", "- 4: orbit-types $t$ (long-axis tube, intermediate-axis tube, short-axis tube, box)\n", "- 6: bins in radius $r$\n", "- 7: bins in circularity (aka lambda) $l$\n", "- 360: orbit bundles $b$\n", "\n", "We'll refer to this tensor as $P_{trlb}$ where $(t,r,l,b)$ index over these four dimensions. $P_{trlb}$ is defined as the fraction of orbit bundle $b$ that is of orbit-type $t$ and that resides in radius bin $r$ and lambda bin $l$.\n", "\n", "To demonstrate how we might use this object, say we want to know the fraction of different orbit-types per orbit-bundle. To do this, we can sum over all radius and lambda bins, " ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "orbit_fractions = np.sum(projection_tensor, (1,2))\n", "orbit_fractions.shape" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "then plot the resulting fractions:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "fig, ax = plt.subplots(1, 4, figsize=(9,3), sharey=True)\n", "titles = ['long-axis tube', 'int-axis tube', 'short-axis tube', 'box']\n", "for i, (ax0, tit0) in enumerate(zip(ax, titles)):\n", " ax0.plot(orbit_fractions[i], '.')\n", " ax0.set_title(tit0)\n", " ax0.set_xlabel('Orbit Bundle Index')\n", "ax[0].set_ylabel('Fraction')\n", "fig.tight_layout()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "In this test example (with dithering=1) each bundle is entirely made up of one orbit type i.e. all orbit fractions are 0 or 1. For dithering>1, note that this isn't the case: you can have orbit-bundles with mixtures of different types of orbits within them.\n", "\n", "To get the orbit distributions in (radius, lambda) space, we take the dot product of the projection tensor with the orbit weights," ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "orbit_distributions = orblib.projection_tensor.dot(weights)\n", "orbit_distributions.shape # (orbit types, radius bins, lambda bins)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "We could then re-create the short-axis distribution plot as follows:" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "plt.imshow(\n", " orbit_distributions[2], # use index 2 for short axis tubes\n", " origin='lower',\n", " extent=(-1, 1, np.log10(minr), np.log10(maxr)),\n", " aspect='auto'\n", " )\n", "plt.gca().set_xlabel('Circularity $\\lambda_z$')\n", "plt.gca().set_ylabel('log (radius / kpc)')\n", "plt.gca().set_title('Short-axis tube distribution')" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## 4. Orbit classification\n", "\n", "To do the orbit classification, `orblib.get_projection_tensor` internally runs the method `orblib.classify_orbits`. To see what this does let's run this ourselves, now with the `make_diagnostic_plots` option set," ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "orblib.classify_orbits(make_diagnostic_plots=True)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "This shows the distribution of angular momenta of the orbits in the orbit library. This is shown for the three components of angular momentum $(L_x, L_y, L_z)$ represented by the three coloured histrograms. The dotted vertical lines show a threshold angular momentum $dL$ which is used for orbit classification. By default, this is set to a (somewhat arbitrary!) choice of $dL= 10^{17}$.\n", "\n", "Given this choice of $dL$, the classification is done as follows:\n", "- if $L_x < dL$, $L_y < dL$ and $L_z < dL$, then it's a box orbit \n", "- otherwise, it's a tube orbit, with the largest component of $L$ determining the axis (e.g. if $L_z$ is largest then it's a short-axis tube orbit)\n", "\n", "In theory, for tube-orbits we expect one and only one component of the time-averaged angular momentum to be non-zero. The output which is printed when we ran `orblib.classify_orbits` tells us what fraction of tube-orbits this is actually true for, when $dL$ is used as a threshold for \"non-zero\". Copying the relevant lines here:\n", "\n", "```\n", "[INFO] 23:46:28 - dynamite.orblib.LegacyOrbitLibrary - Amongst tubes, % with only one nonzero component of L:\n", "[INFO] 23:46:28 - dynamite.orblib.LegacyOrbitLibrary - - 22.9% of x-tubes\n", "[INFO] 23:46:28 - dynamite.orblib.LegacyOrbitLibrary - - 35.7% of y-tubes\n", "[INFO] 23:46:28 - dynamite.orblib.LegacyOrbitLibrary - - 72.0% of z-tubes\n", "```\n", "\n", "e.g. this says that only 72% of orbits we have classified as short-axis tubes have both $L_x