{ "cells": [ { "cell_type": "markdown", "id": "dffc232a", "metadata": {}, "source": [ "# Intro: KOMO - Motion Optimization\n", "\n", "KOMO is a framework for designing motion by formulating optimization problems. Inverse kinematics (IK) is the special case of optimizing only over a single configuration rather than a path. Formulating KOMO problems is key to realizing motion in `rai`.\n", "\n", "The [Script:Inverse Kinematics](https://marctoussaint.github.io/robotics-course/script/script.html#general-concept-of-differentiable-features) and the [Appendix:NLP Interface](https://marctoussaint.github.io/robotics-course/script/script.html#nlp-interface) provide the mathematical background on inverse kinematics and especially the convention of how NLPs can be defined by adding objectives.\n", "\n", "This tutorial shows how IK, rough waypoint optimization, and fine path optimization can be formulated as non-linear mathematical program (NLP) using KOMO. Essentially, the `addObjective` allows to add costs or constraints over any `Feature` to the NLP (same features that can be evaluated with `C.eval`)." ] }, { "cell_type": "markdown", "id": "a177972b", "metadata": {}, "source": [ "## Minimal IK example" ] }, { "cell_type": "code", "execution_count": null, "id": "8e07bf36", "metadata": {}, "outputs": [], "source": [ "import robotic as ry\n", "import numpy as np\n", "import time" ] }, { "cell_type": "code", "execution_count": null, "id": "059a8ee7", "metadata": {}, "outputs": [], "source": [ "C = ry.Config()\n", "C.addFile(ry.raiPath('scenarios/pandaSingle.g'))\n", "C.view()" ] }, { "cell_type": "code", "execution_count": null, "id": "582b68ba", "metadata": {}, "outputs": [], "source": [ "C.addFrame('box') \\\n", " .setPosition([-.25,.1,1.]) \\\n", " .setShape(ry.ST.ssBox, size=[.06,.06,.06,.005]) \\\n", " .setColor([1,.5,0]) \\\n", " .setContact(1)\n", "C.view()" ] }, { "cell_type": "markdown", "id": "ac059dc2", "metadata": {}, "source": [ "The following defines an optimization problem over a single configuration. The KOMO object essentially contains:\n", "1. Copies of the configuration(s) over which we optimize\n", "2. The list of objectives (=costs & constraints) that define the optimization problem.\n", "\n", "The constructor declares over how many configurations (single, waypoints, path..) we optimize. The addObjective methods add costs or constraints:" ] }, { "cell_type": "code", "execution_count": null, "id": "bccb7b55", "metadata": {}, "outputs": [], "source": [ "qHome = C.getJointState()\n", "\n", "komo = ry.KOMO(C, phases=1, slicesPerPhase=1, kOrder=0, enableCollisions=False)\n", "komo.addObjective(\n", " times=[], \n", " feature=ry.FS.jointState, \n", " frames=[],\n", " type=ry.OT.sos, \n", " scale=[1e-1], \n", " target=qHome\n", ")\n", "komo.addObjective([], ry.FS.positionDiff, ['l_gripper', 'box'], ry.OT.eq, [1e1])" ] }, { "cell_type": "markdown", "id": "76895850", "metadata": {}, "source": [ "We explain the KOMO constructor arguments later. (The above defines an IK problem.)\n", "\n", "The `addObjective` method has signature\n", "\n", "* `times`: the time intervals (subset of configurations in a path) over which this feature is active (irrelevant for IK)\n", "* `feature`: the feature symbol (see advanced `Feature` tutorial)\n", "* `frames`: the frames for which the feature is computed, given as list of frame names\n", "* `type`: whether this is a sum-of-squares (sos) cost, or eq or ineq constraint\n", "* `scale`: the matrix(!) by which the feature is multiplied\n", "* `target`: the offset which is substracted from the feature (before scaling)\n", "\n", "Please see more formal details [here](https://marctoussaint.github.io/robotics-course/script/script.html#nlp-interface)." ] }, { "cell_type": "markdown", "id": "9e27cfa8", "metadata": {}, "source": [ "Given this definition of an optimization problem, we can call a generic NLP solver:" ] }, { "cell_type": "code", "execution_count": null, "id": "178e3d42", "metadata": {}, "outputs": [], "source": [ "ret = ry.NLP_Solver(komo.nlp(), verbose=4) .solve()\n", "print(ret)" ] }, { "cell_type": "markdown", "id": "5481e8b8", "metadata": {}, "source": [ "With this high verbosity, individual newton steps and Augmented Lagrangian outer loops are displayed (we need only very few steps here).\n", "\n", "The KOMO view displays the optimized configuration(s) stored by KOMO. (For paths, this is an overlay of many configurations. For IK, just one.):" ] }, { "cell_type": "code", "execution_count": null, "id": "b727e37c", "metadata": {}, "outputs": [], "source": [ "komo.view(False, \"IK solution\")" ] }, { "cell_type": "markdown", "id": "10840e1f", "metadata": {}, "source": [ "We can get the sequence of joint state vectors for the optimized configuration(s) with `getPath`. Since this is only an IK problem, the sequence contains only the joint state vector for the single optimized configuration:" ] }, { "cell_type": "code", "execution_count": null, "id": "5adc15f8", "metadata": {}, "outputs": [], "source": [ "q = komo.getPath()\n", "print(type(q), len(q))" ] }, { "cell_type": "markdown", "id": "9f92e896", "metadata": {}, "source": [ "We're done with KOMO and can destroy it. Then set the optimal joint state in C and view it:" ] }, { "cell_type": "code", "execution_count": null, "id": "b20fc581", "metadata": {}, "outputs": [], "source": [ "del komo #also closes komo view\n", "C.setJointState(q[0])\n", "C.view()" ] }, { "cell_type": "markdown", "id": "57ccf739", "metadata": {}, "source": [ "## Example for more constraints: box grasping IK\n", "\n", "The key to design motions is to add clever constraints. Here is an example for more realistic box grasping:" ] }, { "cell_type": "code", "execution_count": null, "id": "bdbbbe7b", "metadata": {}, "outputs": [], "source": [ "komo = ry.KOMO(C, 1,1,0, True)\n", "komo.addObjective([], ry.FS.jointState, [], ry.OT.sos, [1e-1], qHome)\n", "komo.addObjective([], ry.FS.accumulatedCollisions, [], ry.OT.eq)\n", "komo.addObjective([], ry.FS.jointLimits, [], ry.OT.ineq)\n", "komo.addObjective([], ry.FS.positionDiff, ['l_gripper', 'box'], ry.OT.eq, [1e1])\n", "komo.addObjective([], ry.FS.scalarProductXX, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "komo.addObjective([], ry.FS.scalarProductXZ, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "komo.addObjective([], ry.FS.distance, ['l_palm', 'box'], ry.OT.ineq, [1e1])" ] }, { "cell_type": "markdown", "id": "de8fe5b5", "metadata": {}, "source": [ "The two `scalarProduct` feature state that the gripper x-axis (which is the axis connecting the fingers) should be orthogonal to the object x- and z-axes. That implies fingers to normally oppose the object's y-planes.\n", "\n", "Note that grasping could also be opposing the object x- or z- planes -- see below. Let solve it and then set the joint state to the solution:" ] }, { "cell_type": "code", "execution_count": null, "id": "dab4fbee", "metadata": {}, "outputs": [], "source": [ "ret = ry.NLP_Solver(komo.nlp(), verbose=0 ) .solve()\n", "print(ret)\n", "if ret.feasible:\n", " print('-- Always check feasibility flag of NLP solver return')\n", "else:\n", " print('-- THIS IS INFEASIBLE!')" ] }, { "cell_type": "code", "execution_count": null, "id": "f1970bb1", "metadata": {}, "outputs": [], "source": [ "q = komo.getPath()\n", "C.setJointState(q[0])\n", "C.view(False, \"IK solution\")" ] }, { "cell_type": "markdown", "id": "bef1a139", "metadata": {}, "source": [ "Reusing the KOMO instance is ok if some aspect of the configuration changes and you want to resolve the same problem:" ] }, { "cell_type": "code", "execution_count": null, "id": "a2d386d6", "metadata": {}, "outputs": [], "source": [ "box = C.getFrame('box')\n", "box.setPosition([-.25,.1,1.])\n", "p0 = box.getPosition() # memory the start box position\n", "\n", "for t in range(10):\n", " box.setPosition(p0 + .2 * np.random.randn(3)) # randomize box position\n", " komo.updateRootObjects(C) # only works for root objects (the 'box' is one)\n", " ret = ry.NLP_Solver(komo.nlp(), verbose=0 ) .solve()\n", " print(ret)\n", " q = komo.getPath()\n", " C.setJointState(q[0])\n", " C.view(False, 'IK solution - ' + ('*** INFEASIBLE ***' if not ret.feasible else 'feasible'))\n", " time.sleep(1.)" ] }, { "cell_type": "markdown", "id": "c0e78a35", "metadata": {}, "source": [ "So the solver finds feasible grasps and exploits the null space of the constraints (grasps from different directions, but always opposing the y-planes).\n", "\n", "To make this proper, we should actually test all three possible grasps - so let's define 3 IK problems, solve each, and pick the best:" ] }, { "cell_type": "code", "execution_count": null, "id": "8222658d", "metadata": {}, "outputs": [], "source": [ "del komo\n", "komo = []\n", "for k in range(3):\n", " komo.append(ry.KOMO(C, 1,1,0, True))\n", " komo[k].addObjective([], ry.FS.jointState, [], ry.OT.sos, [1e-1], qHome)\n", " komo[k].addObjective([], ry.FS.accumulatedCollisions, [], ry.OT.eq)\n", " komo[k].addObjective([], ry.FS.jointLimits, [], ry.OT.ineq)\n", " komo[k].addObjective([], ry.FS.positionDiff, ['l_gripper', 'box'], ry.OT.eq, [1e1])\n", " komo[k].addObjective([], ry.FS.distance, ['l_palm', 'box'], ry.OT.ineq, [1e1])\n", "\n", "komo[0].addObjective([], ry.FS.scalarProductXY, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "komo[0].addObjective([], ry.FS.scalarProductXZ, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "\n", "komo[1].addObjective([], ry.FS.scalarProductXX, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "komo[1].addObjective([], ry.FS.scalarProductXZ, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "\n", "komo[2].addObjective([], ry.FS.scalarProductXX, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])\n", "komo[2].addObjective([], ry.FS.scalarProductXY, ['l_gripper', 'box'], ry.OT.eq, [1e1], [0])" ] }, { "cell_type": "code", "execution_count": null, "id": "09c7a057", "metadata": {}, "outputs": [], "source": [ "for t in range(10):\n", " box.setPosition(p0 + .2 * np.random.randn(3))\n", " box.setQuaternion(np.random.randn(4)) # also set random orientation (quaternions get internally normalized)\n", " \n", " score = []\n", " for k in range(3):\n", " komo[k].updateRootObjects(C)\n", " ret = ry.NLP_Solver(komo[k].nlp(), verbose=0 ) .solve()\n", " score.append( 100.*(ret.eq+ret.ineq) + ret.sos )\n", " \n", " k = np.argmin(score)\n", " C.setJointState(komo[k].getPath()[0])\n", " C.view(False, f'IK solution {k} - ' + ('*** INFEASIBLE ***' if not ret.feasible else 'feasible'))\n", " time.sleep(1.)" ] }, { "cell_type": "code", "execution_count": null, "id": "cda905f5", "metadata": {}, "outputs": [], "source": [ "del komo\n", "del C" ] }, { "cell_type": "markdown", "id": "f7d69b02", "metadata": {}, "source": [ "## Waypoints example\n", "\n", "Motion design can often be done by computing waypoints, i.e. a coarse-resolution sequence of poses. The BotOp interface can then spline-interpolate between them when executing them.\n", "\n", "Let's define a configuration where the desired gripper waypoints are pre-defined as marker frames. (That's a common pattern: Simplify defining constraints by adding helper reference frames in the configuration.)" ] }, { "cell_type": "code", "execution_count": null, "id": "a6da9bda", "metadata": {}, "outputs": [], "source": [ "C = ry.Config()\n", "C.addFile(ry.raiPath('scenarios/pandaSingle.g'))\n", "C.addFrame('way1'). setShape(ry.ST.marker, [.1]) .setPosition([.4, .2, 1.])\n", "C.addFrame('way2'). setShape(ry.ST.marker, [.1]) .setPosition([.4, .2, 1.4])\n", "C.addFrame('way3'). setShape(ry.ST.marker, [.1]) .setPosition([-.4, .2, 1.])\n", "C.addFrame('way4'). setShape(ry.ST.marker, [.1]) .setPosition([-.4, .2, 1.4])\n", "qHome = C.getJointState()\n", "C.view()" ] }, { "cell_type": "markdown", "id": "53a2b1c8", "metadata": {}, "source": [ "Now we can define a KOMO problem over 4 configurations, where at each configuration we impose position equality between gripper and a waypoint:" ] }, { "cell_type": "code", "execution_count": null, "id": "7c3a74d2", "metadata": {}, "outputs": [], "source": [ "komo = ry.KOMO(C, phases=4, slicesPerPhase=1, kOrder=1, enableCollisions=False)\n", "komo.addControlObjective([], 0, 1e-1)\n", "komo.addControlObjective([], 1, 1e0)\n", "komo.addObjective([1], ry.FS.positionDiff, ['l_gripper', 'way1'], ry.OT.eq, [1e1])\n", "komo.addObjective([2], ry.FS.positionDiff, ['l_gripper', 'way2'], ry.OT.eq, [1e1])\n", "komo.addObjective([3], ry.FS.positionDiff, ['l_gripper', 'way3'], ry.OT.eq, [1e1])\n", "komo.addObjective([4], ry.FS.positionDiff, ['l_gripper', 'way4'], ry.OT.eq, [1e1])\n", "\n", "ret = ry.NLP_Solver(komo.nlp(), verbose=0 ) .solve()\n", "print(ret)\n", "q = komo.getPath()\n", "print(q)\n", "\n", "for t in range(len(q)):\n", " C.setJointState(q[t])\n", " C.view(False, f'waypoint {t}')\n", " time.sleep(1)" ] }, { "cell_type": "markdown", "id": "f6263d5c", "metadata": {}, "source": [ "The `KOMO constructor` has arguments:\n", "\n", "* `config`: the configuration, which is copied once (for IK) or many times (for waypoints/paths) to be the optimization variable\n", "* `phases`: the number $P$ of phases (which essentially defines the real-valued interval $[0, P]$ over which objectives can be formulated)\n", "* `slicesPerPhase`: the discretizations per phase -> in total we have $\\texttt{phases} \\cdot \\texttt{slicesPerPhases}$ configurations which form the path and over which we optimize\n", "* `kOrder`: the \"Markov-order\", i.e., maximal tuple of configurations over which we formulate features (e.g. take finite differences)\n", "* enableCollisions: if True, KOMO runs a broadphase collision check (using libFCL) in each optimization step -- only then accumulative collision/penetration features will correctly evaluate to non-zero. But this is costly.\n", "\n", "In our waypoint case: We have 4 phases, one for each waypoint. We don't sub-sample the motion between waypoints, which is why we have slicesPerPhase=1. We formulate this as a 1-order problem: Some features take the finite difference between consecutive configurations (namely, to penalize velocities).\n", "\n", "The `addControlObjective` is /almost/ the same as adding a `FS.jointState` objective: It penalizes distances in joint space. It has three arguments:\n", "\n", "* `times`: (as for `addObjective`) the phase-interval in which this objective holds; [] means all times\n", "* `order`: Do we penalize the jointState directly (order=0: penalizing sqr distance to qHome, order=1: penalizing sqr distances between consecutive configurations (velocities), order=2: penalizing accelerations across 3 configurations)\n", "* `scale`: as usual, but modulated by a factor \"sqrt(delta t)\" that somehow ensures total control costs in approximately independent of the choice of stepsPerPhase\n", "\n", "In our waypoint case: We add control costs for both: homing (order 0, ensuring to stay close to homing), and velocities (order 1, penalizing movement between waypoints)\n", "\n", "And the `addObjective` method now makes use of `times` argument: Specifying [1] means that this objective only holds in the interval [1,1], i.e. at phase-time 1 only." ] }, { "cell_type": "markdown", "id": "132f82a0", "metadata": {}, "source": [ "## Path example\n", "\n", "Let's do almost the same, but for a fine path. First order=1, leading to zig-zag, then order=2, leading to smooth path." ] }, { "cell_type": "code", "execution_count": null, "id": "dd21ae9e", "metadata": {}, "outputs": [], "source": [ "# Note, the slicesPerPhase=10 is the only difference to above\n", "C.setJointState(qHome)\n", "komo = ry.KOMO(C, 4, 10, 1, False)\n", "komo.addControlObjective([], 0, 1e-1) # what happens if you change weighting to 1e0? why?\n", "komo.addControlObjective([], 1, 1e0)\n", "komo.addObjective([1], ry.FS.positionDiff, ['l_gripper', 'way1'], ry.OT.eq, [1e1])\n", "komo.addObjective([2], ry.FS.positionDiff, ['l_gripper', 'way2'], ry.OT.eq, [1e1])\n", "komo.addObjective([3], ry.FS.positionDiff, ['l_gripper', 'way3'], ry.OT.eq, [1e1])\n", "komo.addObjective([4], ry.FS.positionDiff, ['l_gripper', 'way4'], ry.OT.eq, [1e1])\n", "\n", "ret = ry.NLP_Solver(komo.nlp(), verbose=0 ) .solve()\n", "print(ret)\n", "q = komo.getPath()\n", "print('size of path:', q.shape)\n", "\n", "for t in range(q.shape[0]):\n", " C.setJointState(q[t])\n", " C.view(False, f'waypoint {t}')\n", " time.sleep(.1)" ] }, { "cell_type": "code", "execution_count": null, "id": "40341e34", "metadata": {}, "outputs": [], "source": [ "# only differences: the kOrder=2, control objective order 2, constrain final jointState velocity to zero\n", "C.setJointState(qHome)\n", "komo = ry.KOMO(C, 4, 10, 2, False)\n", "komo.addControlObjective([], 0, 1e-1) # what happens if you change weighting to 1e0? why?\n", "komo.addControlObjective([], 2, 1e0)\n", "komo.addObjective([1], ry.FS.positionDiff, ['l_gripper', 'way1'], ry.OT.eq, [1e1])\n", "komo.addObjective([2], ry.FS.positionDiff, ['l_gripper', 'way2'], ry.OT.eq, [1e1])\n", "komo.addObjective([3], ry.FS.positionDiff, ['l_gripper', 'way3'], ry.OT.eq, [1e1])\n", "komo.addObjective([4], ry.FS.positionDiff, ['l_gripper', 'way4'], ry.OT.eq, [1e1])\n", "komo.addObjective([4], ry.FS.jointState, [], ry.OT.eq, [1e1], [], order=1)\n", "\n", "ret = ry.NLP_Solver(komo.nlp(), verbose=0 ) .solve()\n", "print(ret)\n", "q = komo.getPath()\n", "print('size of path:', q.shape)\n", "\n", "for t in range(q.shape[0]):\n", " C.setJointState(q[t])\n", " C.view(False, f'waypoint {t}')\n", " time.sleep(.1)" ] }, { "cell_type": "markdown", "id": "154ea039", "metadata": {}, "source": [ "Notice the new last objective! Without it, *final velocity* would not be zero. The last objective constrains the order=1 (i.e. velocity!) of the jointState feature to be zero.\n", "\n", "Let's plot the trajectory:" ] }, { "cell_type": "code", "execution_count": null, "id": "3d47d887", "metadata": {}, "outputs": [], "source": [ "from matplotlib import pyplot as plt\n", "\n", "print(q.shape)\n", "fig, ax = plt.subplots()\n", "ax.plot(q)\n", "ax.set_xlabel(\"time slice (0=first-to-be-optimized)\")\n", "ax.set_ylabel(\"joint angles\")\n", "fig.tight_layout()\n", "plt.show()" ] }, { "cell_type": "markdown", "id": "748b6a95", "metadata": {}, "source": [ "Let's plot the errors/costs of the objectives over time:" ] }, { "cell_type": "code", "execution_count": null, "id": "add05d33", "metadata": {}, "outputs": [], "source": [ "err = komo.info_objectiveErrorTraces()\n", "labels = komo.info_objectiveNames()\n", "\n", "fig, ax = plt.subplots()\n", "ax.plot(err, label=labels)\n", "ax.legend(loc=\"upper left\", prop={'size': 6})\n", "ax.set_xlabel(\"time slice (0=first-to-be-optimized)\")\n", "ax.set_ylabel(\"objectiv error\")\n", "fig.tight_layout()\n", "plt.show()" ] }, { "cell_type": "markdown", "id": "6b6968ec", "metadata": {}, "source": [ "And here is an interactive widget to slide through the solution. See the KOMO Viewer (raise to top). It follows the slider and displays the top error/cost objectives for the current time slice:" ] }, { "cell_type": "code", "execution_count": null, "id": "3eff36e4", "metadata": {}, "outputs": [], "source": [ "%matplotlib widget\n", "from matplotlib.widgets import Slider\n", "\n", "T = komo.getT()\n", "komo.view()\n", "\n", "fig, ax = plt.subplots()\n", "fig.canvas.toolbar_visible = False\n", "fig.canvas.header_visible = False\n", "fig.canvas.footer_visible = False\n", "fig.subplots_adjust(left=.1, bottom=.2, right=.95, top=.9)\n", "\n", "ax.set_title('Use the slider to control view; raise KOMO Viewer \"Always on Top\"')\n", "ax.plot(err, label=labels)\n", "line, = ax.plot([0,0], ax.get_ylim(), linewidth=1.5, color='#b3b3b3')\n", "ax.legend(loc=\"upper left\", prop={'size': 6})\n", "ax.set_xlabel(\"time slice (0=first-to-be-optimized)\")\n", "ax.set_ylabel(\"objective error\")\n", "\n", "def slider_callback(val):\n", " slice = max(-1, min(T-1, int(val)))\n", " if slice==-1:\n", " komo.view()\n", " else:\n", " komo.view_slice(slice, False)\n", " line.set_xdata([slice, slice])\n", " fig.canvas.draw_idle()\n", "\n", "axslider = fig.add_axes([0.05, 0.05, 0.9, 0.03])\n", "slider = Slider(ax=axslider, label='t', valmin=-1, valmax=T-1, valinit=0)\n", "slider.on_changed(slider_callback)\n", "\n", "plt.show()" ] }, { "cell_type": "code", "execution_count": null, "id": "2c39b8cb", "metadata": {}, "outputs": [], "source": [ "plt.close('all')" ] }, { "cell_type": "code", "execution_count": null, "id": "08aacd1a", "metadata": {}, "outputs": [], "source": [ "del komo\n", "del C" ] }, { "cell_type": "code", "execution_count": null, "id": "a2a93b87", "metadata": {}, "outputs": [], "source": [] } ], "metadata": { "kernelspec": { "display_name": "Python 3 (ipykernel)", "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.12.3" }, "widgets": { "application/vnd.jupyter.widget-state+json": { "state": { "2c8ccc7ef4654d8d81986391543e4263": { "model_module": "jupyter-matplotlib", "model_module_version": "^0.11", "model_name": "ToolbarModel", "state": { "_current_action": "", "_dom_classes": [], "_model_module": "jupyter-matplotlib", "_model_module_version": "^0.11", "_model_name": "ToolbarModel", "_view_count": null, "_view_module": "jupyter-matplotlib", "_view_module_version": "^0.11", "_view_name": "ToolbarView", "button_style": "", "collapsed": true, "layout": "IPY_MODEL_39ba6c379bdd461a9905edb6371926c3", "orientation": "vertical", "tabbable": null, "toolitems": [ [ "Home", "Reset original view", "home", "home" ], [ "Back", "Back to previous view", "arrow-left", "back" ], [ "Forward", "Forward to next view", "arrow-right", "forward" ], [ "Pan", "Left button pans, Right button zooms\nx/y fixes axis, CTRL fixes aspect", "arrows", "pan" ], [ "Zoom", "Zoom to rectangle\nx/y fixes axis", "square-o", "zoom" ], [ "Download", "Download plot", "floppy-o", "save_figure" ] ], "tooltip": null } }, "39ba6c379bdd461a9905edb6371926c3": { "model_module": "@jupyter-widgets/base", "model_module_version": "2.0.0", "model_name": "LayoutModel", "state": { "_model_module": "@jupyter-widgets/base", "_model_module_version": "2.0.0", "_model_name": "LayoutModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "2.0.0", "_view_name": "LayoutView", "align_content": null, "align_items": null, "align_self": null, "border_bottom": null, "border_left": null, "border_right": null, "border_top": null, "bottom": null, "display": null, "flex": null, "flex_flow": null, "grid_area": null, "grid_auto_columns": null, "grid_auto_flow": null, "grid_auto_rows": null, "grid_column": null, "grid_gap": null, "grid_row": null, "grid_template_areas": null, "grid_template_columns": null, "grid_template_rows": null, "height": null, "justify_content": null, "justify_items": null, "left": null, "margin": null, "max_height": null, "max_width": null, "min_height": null, "min_width": null, "object_fit": null, "object_position": null, "order": null, "overflow": null, "padding": null, "right": null, "top": null, "visibility": null, "width": null } }, "49529751f16e4439bb1100893e980492": { "model_module": "@jupyter-widgets/base", "model_module_version": "2.0.0", "model_name": "LayoutModel", "state": { "_model_module": "@jupyter-widgets/base", "_model_module_version": "2.0.0", "_model_name": "LayoutModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "2.0.0", "_view_name": "LayoutView", "align_content": null, "align_items": null, "align_self": null, "border_bottom": null, "border_left": null, "border_right": null, "border_top": null, "bottom": null, "display": null, "flex": null, "flex_flow": null, "grid_area": null, "grid_auto_columns": null, "grid_auto_flow": null, "grid_auto_rows": null, "grid_column": null, "grid_gap": null, "grid_row": null, "grid_template_areas": null, "grid_template_columns": null, "grid_template_rows": null, "height": null, "justify_content": null, "justify_items": null, "left": null, "margin": null, "max_height": null, "max_width": null, "min_height": null, "min_width": null, "object_fit": null, "object_position": null, "order": null, "overflow": null, "padding": null, "right": null, "top": null, "visibility": null, "width": null } }, "702d5a46d474492199b7b041d85d0b59": { "model_module": "jupyter-matplotlib", "model_module_version": "^0.11", "model_name": "MPLCanvasModel", "state": { "_cursor": "pointer", "_data_url": "", "_dom_classes": [], "_figure_label": "Figure", "_image_mode": "full", "_message": "", "_model_module": "jupyter-matplotlib", "_model_module_version": "^0.11", "_model_name": "MPLCanvasModel", "_rubberband_height": 0, "_rubberband_width": 0, "_rubberband_x": 0, "_rubberband_y": 0, "_size": [ 640.0, 480.0 ], "_view_count": null, "_view_module": "jupyter-matplotlib", "_view_module_version": "^0.11", "_view_name": "MPLCanvasView", "capture_scroll": false, "footer_visible": false, "header_visible": false, "layout": "IPY_MODEL_49529751f16e4439bb1100893e980492", "pan_zoom_throttle": 33.0, "resizable": true, "tabbable": null, "toolbar": "IPY_MODEL_2c8ccc7ef4654d8d81986391543e4263", "toolbar_position": "left", "toolbar_visible": false, "tooltip": null } } }, "version_major": 2, "version_minor": 0 } } }, "nbformat": 4, "nbformat_minor": 5 }