{
  "instance_id": "matplotlib__matplotlib-26011",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-05-30T13:45:49Z",
  "problem_statement": "xlim_changed not emitted on shared axis\n<!--To help us understand and resolve your issue, please fill out the form to the best of your ability.-->\r\n<!--You can feel free to delete the sections that do not apply.-->\r\n\r\n### Bug report\r\n\r\n**Bug summary**\r\n\r\nWhen an axis is shared with another its registered \"xlim_changed\" callbacks does not get called when the change is induced by a shared axis (via sharex=). \r\n\r\nIn _base.py the set_xlim for sibling axis are called with emit=False:\r\n\r\n```\r\nmatplotlib/lib/matplotlib/axes/_base.py:\r\n\r\n/.../\r\ndef set_xlim(...)\r\n/.../\r\n        if emit:\r\n            self.callbacks.process('xlim_changed', self)\r\n            # Call all of the other x-axes that are shared with this one\r\n            for other in self._shared_x_axes.get_siblings(self):\r\n                if other is not self:\r\n                    other.set_xlim(self.viewLim.intervalx,\r\n                                   emit=False, auto=auto)\r\n```\r\n\r\nI'm very new to matplotlib, so perhaps there is a good reason for this? emit=False seems to disable both continued \"inheritance\" of axis (why?) and triggering of change callbacks (looking at the code above).\r\n\r\nIt seems like one would at least want to trigger the xlim_changed callbacks as they would be intended to react to any change in axis limits.\r\n\r\nEdit: Setting emit=True seems to introduce a recursion issue (not sure why but as inheritance seems to be passed along anyway it doesn't really matter). Moving the callback call to outside of the \"if emit:\"-statement seems to solve the issue as far as I can see when trying it out. Any reason to keep it inside the if-statement? \r\n\n",
  "patch": "diff --git a/lib/matplotlib/axis.py b/lib/matplotlib/axis.py\n--- a/lib/matplotlib/axis.py\n+++ b/lib/matplotlib/axis.py\n@@ -1241,11 +1241,13 @@ def _set_lim(self, v0, v1, *, emit=True, auto):\n             self.axes.callbacks.process(f\"{name}lim_changed\", self.axes)\n             # Call all of the other axes that are shared with this one\n             for other in self._get_shared_axes():\n-                if other is not self.axes:\n-                    other._axis_map[name]._set_lim(\n-                        v0, v1, emit=False, auto=auto)\n-                    if other.figure != self.figure:\n-                        other.figure.canvas.draw_idle()\n+                if other is self.axes:\n+                    continue\n+                other._axis_map[name]._set_lim(v0, v1, emit=False, auto=auto)\n+                if emit:\n+                    other.callbacks.process(f\"{name}lim_changed\", other)\n+                if other.figure != self.figure:\n+                    other.figure.canvas.draw_idle()\n \n         self.stale = True\n         return v0, v1\n",
  "similar_bug_items": [
    {
      "pr_number": 10142,
      "pr_title": "FIX: prevent recursive draws with plt.ion not in IPython with qt5agg",
      "pr_body": "## PR Summary\r\n\r\nIn 9b8a944db954c8240e14413e517e1b781881b80b the latch logic was moved\r\nabove the call to `self.draw` to de-latch the call to\r\n`__draw_idle_agg` in `paintEvent` to protect against `blit` calls\r\nduring a draw.\r\n\r\nHowever, this now fails to properly latch the base draw so with\r\n`plt.ion` (not in IPython) the stale callback will cause a (possibly\r\ninfinite) recursion in the draw due to stale being set as part of the\r\ndraw call (which will re-trigger the draw).\r\n\r\nThis adds a second flag to track if we are currently rendering and\r\nbail.\r\n\r\ncloses #10140\r\n\r\n\r\nI suspect a better fix is to re-think how the stale callbacks propagate or how the non-ipython stale callback works, but I think this is the minimal change which fixes a critical issue.\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
      "issue_id": 10140,
      "issue_title": "Qt5Agg eats 100% CPU when plotting with block=True in interactive mode",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\nWhen using Qt5Agg, Matplotlib consumes a full CPU core while an interactive window is open.  The cursor repeatedly shows a \"spinner\" when I mouse over the canvas.  This didn't used to be the case.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nplt.plot([1, 2], [3, 4])\r\nplt.ion()\r\nplt.show(block=True)\r\n```\r\n\r\n**Actual outcome**\r\n\r\nOne of the CPU cores is pegged at 100%.  Interface is very slow and barely responsive.\r\n\r\n**Expected outcome**\r\n\r\nShould not use up 100% of a CPU core when nothing is happening.\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating system: Linux 4.14.8\r\n  * Matplotlib version: 2.1.1+864.g92a93d698.dirty\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: 3.6.4\r\n\r\nThis was reproduced on both XMonad and XFCE4.  Affects both `python-matplotlib` (2.1.1) from official Arch repos and `python-matplotlib-git` from AUR.",
      "issue_closed_at": "2018-01-04T20:28:55Z",
      "base_commit": "008da385fe73614059430073a722d674ade55014",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def __init__(self, figure):\n        super(FigureCanvasQTAggBase, self).__init__(figure=figure)\n        self.setAttribute(QtCore.Qt.WA_OpaquePaintEvent)\n        self._agg_draw_pending = False\n        self._bbox_queue = []\n        self._drawRect = None"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "draw",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def draw(self):\n        \"\"\"Draw the figure with Agg, and queue a request for a Qt draw.\n        \"\"\"\n        # The Agg draw is done here; delaying causes problems with code that\n        # uses the result of the draw() to update plot elements.\n        super(FigureCanvasQTAggBase, self).draw()\n        self.update()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "draw_idle",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def draw_idle(self):\n        \"\"\"Queue redraw of the Agg buffer and request Qt paintEvent.\n        \"\"\"\n        # The Agg draw needs to be handled by the same thread matplotlib\n        # modifies the scene graph from. Post Agg draw request to the\n        # current event loop in order to ensure thread affinity and to\n        # accumulate multiple draw requests from event handling.\n        # TODO: queued signal connection might be safer than singleShot\n        if not self._agg_draw_pending:\n            self._agg_draw_pending = True\n            QtCore.QTimer.singleShot(0, self.__draw_idle_agg)"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "__draw_idle_agg",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def __draw_idle_agg(self, *args):\n        # if nothing to do, bail\n        if not self._agg_draw_pending:\n            return\n        # we have now tried this function at least once, do not run\n        # again until re-armed.  Doing this here rather than after\n        # protects against recursive calls triggered through self.draw\n        self._agg_draw_pending = False\n        # if negative size, bail\n        if self.height() < 0 or self.width() < 0:\n            return\n        try:\n            # actually do the drawing\n            self.draw()\n        except Exception:\n            # Uncaught exceptions are fatal for PyQt5, so catch them instead.\n            traceback.print_exc()"
        }
      ]
    },
    {
      "pr_number": 25433,
      "pr_title": "Release mouse grabs when owning Axes is removed",
      "pr_body": "## PR Summary\r\n\r\nOtherwise, _nothing_ will remove the grab, and no other Axes can grab the mouse (either raising the `RuntimeError` exception, or in the case of widgets, ignoring mouse events altogether.)\r\n\r\nFixes #25345\r\n\r\n## PR Checklist\r\n\r\n**Documentation and Tests**\r\n- [x] Has pytest style unit tests (and `pytest` passes)\r\n- [n/a] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [n/a] New plotting related features are documented with examples.\r\n\r\n**Release Notes**\r\n- [n/a] New features are marked with a `.. versionadded::` directive in the docstring and documented in `doc/users/next_whats_new/`\r\n- [n/a] API changes are marked with a `.. versionchanged::` directive in the docstring and documented in `doc/api/next_api_changes/`\r\n- [n/a] Release notes conform with instructions in  `next_whats_new/README.rst` or `next_api_changes/README.rst`",
      "issue_id": 25345,
      "issue_title": "[Bug]: using clf and pyplot.draw in range slider on_changed callback blocks input to widgets",
      "issue_body": "### Bug summary\n\nWhen using clear figure, adding new widgets and then redrawing the current figure in the on_changed callback of a range slider the inputs to all the widgets in the figure are blocked. When doing the same in the button callback on_clicked, everything works fine.\n\n### Code for reproduction\n\n```python\nimport matplotlib.pyplot as pyplot\r\nimport matplotlib.widgets as widgets\r\n\r\ndef onchanged(values):\r\n    print(\"on changed\")\r\n    print(values)\r\n    pyplot.clf()\r\n    addElements()\r\n    pyplot.draw()\r\n\r\ndef onclick(e):\r\n    print(\"on click\")\r\n    pyplot.clf()\r\n    addElements()\r\n    pyplot.draw()\r\n\r\ndef addElements():\r\n    ax = pyplot.axes([0.1, 0.45, 0.8, 0.1])\r\n    global slider\r\n    slider = widgets.RangeSlider(ax, \"Test\", valmin=1, valmax=10, valinit=(1, 10))\r\n    slider.on_changed(onchanged)\r\n    ax = pyplot.axes([0.1, 0.30, 0.8, 0.1])\r\n    global button\r\n    button = widgets.Button(ax, \"Test\")\r\n    button.on_clicked(onclick)\r\n\r\naddElements()\r\n\r\npyplot.show()\n```\n\n\n### Actual outcome\n\nThe widgets can't receive any input from a mouse click, when redrawing in the on_changed callback of a range Slider. \r\nWhen using a button, there is no problem.\n\n### Expected outcome\n\nThe range slider callback on_changed behaves the same as the button callback on_clicked.\n\n### Additional information\n\nThe problem also occurred on Manjaro with:\r\n- Python version: 3.10.9\r\n- Matplotlib version: 3.6.2\r\n- Matplotlib backend: QtAgg\r\n- Installation of matplotlib via Linux package manager\r\n\n\n### Operating system\n\nWindows 10\n\n### Matplotlib Version\n\n3.6.2\n\n### Matplotlib Backend\n\nTkAgg\n\n### Python version\n\n3.11.0\n\n### Jupyter version\n\n_No response_\n\n### Installation\n\npip",
      "issue_closed_at": "2023-03-14T06:22:33Z",
      "base_commit": "7eafdd8af3c523c1c77b027d378fb337dd489f18",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_break_share_link",
          "class_name": "FigureBase",
          "code": "def _break_share_link(ax, grouper):\n            siblings = grouper.get_siblings(ax)\n            if len(siblings) > 1:\n                grouper.remove(ax)\n                for last_ax in siblings:\n                    if ax is not last_ax:\n                        return last_ax\n            return None"
        }
      ]
    },
    {
      "pr_number": 14068,
      "pr_title": "Alternative fix for passing iterator as frames to FuncAnimation",
      "pr_body": "## PR Summary\r\n\r\nFixes #13676. Replaces #13679.\r\n\r\nThis was proposed in https://github.com/matplotlib/matplotlib/pull/13679#issuecomment-478905310 it's clearly the better approach as it's simpler and can handle generators. Thanks @anntzer.\r\n\r\nI do not really know how to add a test for this. Can't have an infinite loop in a unittest :smile:. Works fine as a standalone test with the following code\r\n\r\n~~~\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nfrom matplotlib.animation import FuncAnimation\r\n\r\nfig, ax = plt.subplots()\r\ns = ax.set_title('0')\r\nplt.plot([1,2,3],[2,4,3])\r\ndef update(frame):\r\n    print(frame)\r\n    s.set_text(str(frame))\r\n    return []\r\nanimation = FuncAnimation(fig, update, frames=(i for i in range(5)), blit=True, interval=100)\r\nplt.show()\r\n~~~\r\n",
      "issue_id": 13676,
      "issue_title": "FuncAnimation with generator causes crash on StopIteration",
      "issue_body": "### Bug report\r\n\r\nUsing FuncAnimation with a generator causes a crash when the generator raises a StopIteration.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nfrom matplotlib.animation import FuncAnimation\r\n\r\nfig, ax = plt.subplots()\r\nplt.plot([1,2,3],[2,4,3])\r\ndef update(frame):\r\n\tprint(frame)\r\n\treturn []\r\nanimation = FuncAnimation(fig, update, frames=iter(range(10)), blit=True, interval=100)\r\nplt.show()\r\n```\r\n\r\nOutput:\r\n\r\n```\r\n$ python animation_crash.py \r\n0\r\n1\r\n2\r\n3\r\n4\r\n5\r\n6\r\n7\r\n8\r\n9\r\nTraceback (most recent call last):\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/backend_bases.py\", line 1238, in _on_timer\r\n    ret = func(*args, **kwargs)\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/animation.py\", line 1462, in _step\r\n    self._init_draw()\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/animation.py\", line 1740, in _init_draw\r\n    self._draw_frame(next(self.new_frame_seq()))\r\nStopIteration\r\nAbgebrochen (Speicherabzug geschrieben)\r\n```\r\n\r\n**Matplotlib version**\r\nOperating System: Arch linux, 64-bit\r\nMatplotlib version: 3.0.3-1 (using pacman)\r\nMatplotlib backend: Qt5Agg\r\n\r\nCoredump:\r\n```\r\n[Current thread is 1 (Thread 0x7f2d24492600 (LWP 25813))]\r\n(gdb) info stack\r\n#0  0x00007f2d249d0d7f in raise () at /usr/lib/libc.so.6\r\n#1  0x00007f2d249bb672 in abort () at /usr/lib/libc.so.6\r\n#2  0x00007f2d1f79a7fc in  () at /usr/lib/libQt5Core.so.5\r\n#3  0x00007f2d1fd2b048 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#4  0x00007f2d1fe8b216 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#5  0x00007f2d1fe8bcd0 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#6  0x00007f2d1f9bb94d in QMetaObject::activate(QObject*, int, int, void**) () at /usr/lib/libQt5Core.so.5\r\n#7  0x00007f2d1f9c7ab8 in QTimer::timeout(QTimer::QPrivateSignal) () at /usr/lib/libQt5Core.so.5\r\n#8  0x00007f2d1fe34c9c in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#9  0x00007f2d1f9bc10b in QObject::event(QEvent*) () at /usr/lib/libQt5Core.so.5\r\n#10 0x00007f2d1fe34b4c in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#11 0x00007f2d1ba68e24 in QApplicationPrivate::notify_helper(QObject*, QEvent*) () at /usr/lib/libQt5Widgets.so.5\r\n#12 0x00007f2d1ba706e1 in QApplication::notify(QObject*, QEvent*) () at /usr/lib/libQt5Widgets.so.5\r\n#13 0x00007f2d1c114b3f in  () at /usr/lib/python3.7/site-packages/PyQt5/QtWidgets.so\r\n#14 0x00007f2d1f990e99 in QCoreApplication::notifyInternal2(QObject*, QEvent*) () at /usr/lib/libQt5Core.so.5\r\n#15 0x00007f2d1f9e6095 in QTimerInfoList::activateTimers() () at /usr/lib/libQt5Core.so.5\r\n#16 0x00007f2d1f9e6922 in  () at /usr/lib/libQt5Core.so.5\r\n#17 0x00007f2d20dd3a2f in g_main_context_dispatch () at /usr/lib/libglib-2.0.so.0\r\n#18 0x00007f2d20dd55e9 in  () at /usr/lib/libglib-2.0.so.0\r\n#19 0x00007f2d20dd562e in g_main_context_iteration () at /usr/lib/libglib-2.0.so.0\r\n#20 0x00007f2d1f9e6ce9 in QEventDispatcherGlib::processEvents(QFlags<QEventLoop::ProcessEventsFlag>) ()\r\n    at /usr/lib/libQt5Core.so.5\r\n#21 0x00007f2d1f98fb2c in QEventLoop::exec(QFlags<QEventLoop::ProcessEventsFlag>) () at /usr/lib/libQt5Core.so.5\r\n#22 0x00007f2d1f997e36 in QCoreApplication::exec() () at /usr/lib/libQt5Core.so.5\r\n#23 0x00007f2d1c1135cd in  () at /usr/lib/python3.7/site-packages/PyQt5/QtWidgets.so\r\n#24 0x00007f2d24775a78 in _PyMethodDef_RawFastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#25 0x00007f2d24775d11 in _PyCFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#26 0x00007f2d247ec8eb in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#27 0x00007f2d247752eb in _PyFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#28 0x00007f2d247ec2b3 in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n--Type <RET> for more, q to quit, c to continue without paging--\r\n#29 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#30 0x00007f2d2472fdec in _PyFunction_FastCallDict () at /usr/lib/libpython3.7m.so.1.0\r\n#31 0x00007f2d2473f908 in _PyObject_Call_Prepend () at /usr/lib/libpython3.7m.so.1.0\r\n#32 0x00007f2d2473033b in PyObject_Call () at /usr/lib/libpython3.7m.so.1.0\r\n#33 0x00007f2d247e943c in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#34 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#35 0x00007f2d24775492 in _PyFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#36 0x00007f2d247ec2b3 in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#37 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#38 0x00007f2d2472fab4 in PyEval_EvalCodeEx () at /usr/lib/libpython3.7m.so.1.0\r\n#39 0x00007f2d2472fadc in PyEval_EvalCode () at /usr/lib/libpython3.7m.so.1.0\r\n#40 0x00007f2d24859c94 in  () at /usr/lib/libpython3.7m.so.1.0\r\n#41 0x00007f2d2485b8be in PyRun_FileExFlags () at /usr/lib/libpython3.7m.so.1.0\r\n#42 0x00007f2d2485cc75 in PyRun_SimpleFileExFlags () at /usr/lib/libpython3.7m.so.1.0\r\n#43 0x00007f2d2485eeb7 in  () at /usr/lib/libpython3.7m.so.1.0\r\n#44 0x00007f2d2485f0fc in _Py_UnixMain () at /usr/lib/libpython3.7m.so.1.0\r\n#45 0x00007f2d249bd223 in __libc_start_main () at /usr/lib/libc.so.6\r\n#46 0x0000561e1056f05e in _start ()\r\n```",
      "issue_closed_at": "2019-05-22T21:21:43Z",
      "base_commit": "2753cf5679cbb943e1219433bbaf784556b8bda3",
      "changes": [
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FuncAnimation",
          "code": "def __init__(self, fig, func, frames=None, init_func=None, fargs=None,\n                 save_count=None, *, cache_frame_data=True, **kwargs):\n        if fargs:\n            self._args = fargs\n        else:\n            self._args = ()\n        self._func = func\n        self._init_func = init_func\n\n        # Amount of framedata to keep around for saving movies. This is only\n        # used if we don't know how many frames there will be: in the case\n        # of no generator or in the case of a callable.\n        self.save_count = save_count\n        # Set up a function that creates a new iterable when needed. If nothing\n        # is passed in for frames, just use itertools.count, which will just\n        # keep counting from 0. A callable passed in for frames is assumed to\n        # be a generator. An iterable will be used as is, and anything else\n        # will be treated as a number of frames.\n        if frames is None:\n            self._iter_gen = itertools.count\n        elif callable(frames):\n            self._iter_gen = frames\n        elif np.iterable(frames):\n            self._iter_gen = lambda: iter(frames)\n            if hasattr(frames, '__len__'):\n                self.save_count = len(frames)\n        else:\n            self._iter_gen = lambda: iter(range(frames))\n            self.save_count = frames\n\n        if self.save_count is None:\n            # If we're passed in and using the default, set save_count to 100.\n            self.save_count = 100\n        else:\n            # itertools.islice returns an error when passed a numpy int instead\n            # of a native python int (http://bugs.python.org/issue30537).\n            # As a workaround, convert save_count to a native python int.\n            self.save_count = int(self.save_count)\n\n        self._cache_frame_data = cache_frame_data\n\n        # Needs to be initialized so the draw functions work without checking\n        self._save_seq = []\n\n        TimedAnimation.__init__(self, fig, **kwargs)\n\n        # Need to reset the saved seq, since right now it will contain data\n        # for a single frame from init, which is not what we want.\n        self._save_seq = []"
        }
      ]
    },
    {
      "pr_number": 6431,
      "pr_title": "Merge from v2.x",
      "pr_body": "This merge from v2.x into master also required a little editing, and it involves many changes, hence this PR.  I resolved conflicts in .travis.yml and lib/matplotlib/tests/test_colors.py.\n",
      "issue_id": 6142,
      "issue_title": "matplotlib.ticker.LinearLocator view_limits algorithm improvement?",
      "issue_body": "Inspecting the code of matplotlib.ticker.LinearLocator\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1161\n\nyou can see that the view limits are chosen such that the difference between vmin and vmax is an interger multiple of scale, which is itself a power of 10. Therefore, the range  will be nicely divisible when divided by 10 (11 tickmarks).\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1213\n\nTherefore, the view_limits function determines the limits assuming there are 11 ticks. This assumption is implicit in two lines:\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1224\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1227\n\nCode is repeated here:\n\n```\nexponent, remainder = divmod(math.log10(vmax - vmin), 1)\nif remainder < 0.5:\n    exponent -= 1\nscale = 10 ** (-exponent)\nvmin = math.floor(scale * vmin) / scale\nvmax = math.ceil(scale * vmax) / scale\n```\n\nSince we know the number of ticks, from `self.num_ticks`, we can generalize the current algorithm to be better suited for any number of ticks. Suggested generalized algorithm:\n\n```\nexponent, remainder = divmod(math.log10(vmax - vmin), math.log10(self.num_ticks-1))\nif remainder < 0.5:\n    exponent -= 1\nscale = (self.num_ticks-1) ** (-exponent)\nvmin = math.floor(scale * vmin) / scale\nvmax = math.ceil(scale * vmax) / scale\n```\n\nThis generalized expression reduces to the current form when `self.num_ticks==11` (which is the current default). For other cases, here is an example:\nwhen num_ticks = 10, vmin = 20, vmax=90\nCurrent algorithm returns vmin = 20, vmax=90, corresponding to ticks spaced by 7.77778.\n\nThe proposed algorithm returns vmin = 18, vmax = 90, corresponding to ticks spaced by 8.\n\nIs this something worth doing? The patch is trivial -- just changing two lines of code. I can turn this in to a pull request to illustrate if it is helpful.\n",
      "issue_closed_at": "2016-05-04T00:26:00Z",
      "base_commit": "22a7b955a0b9dc4dea8adf155041498dd355e4df",
      "changes": [
        {
          "file": "lib/matplotlib/animation.py",
          "type": "line",
          "name": "line 36",
          "code": "import abc\nimport contextlib\nimport tempfile\nfrom matplotlib.cbook import iterable, is_string_like\nfrom matplotlib.compat import subprocess\nfrom matplotlib import verbose\nfrom matplotlib import rcParams, rcParamsDefault\n\n# Process creation flag for subprocess to prevent it raising a terminal\n# window. See for example:"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "class",
          "name": "MovieWriter",
          "code": "class MovieWriter(AbstractMovieWriter):\n    '''\n    Base class for writing movies. Fundamentally, what a MovieWriter does\n    is provide is a way to grab frames by calling grab_frame(). setup()\n    is called to start the process and finish() is called afterwards.\n    This class is set up to provide for writing movie frame data to a pipe.\n    saving() is provided as a context manager to facilitate this process as::\n\n      with moviewriter.saving(fig, outfile='myfile.mp4', dpi=100):\n          # Iterate over frames\n          moviewriter.grab_frame(**savefig_kwargs)\n\n    The use of the context manager ensures that setup and cleanup are\n    performed as necessary.\n\n    frame_format: string\n        The format used in writing frame data, defaults to 'rgba'\n    '''\n    def __init__(self, fps=5, codec=None, bitrate=None, extra_args=None,\n                 metadata=None):\n        '''\n        Construct a new MovieWriter object.\n\n        fps: int\n            Framerate for movie.\n        codec: string or None, optional\n            The codec to use. If None (the default) the setting in the\n            rcParam `animation.codec` is used.\n        bitrate: int or None, optional\n            The bitrate for the saved movie file, which is one way to control\n            the output file size and quality. The default value is None,\n            which uses the value stored in the rcParam `animation.bitrate`.\n            A value of -1 implies that the bitrate should be determined\n            automatically by the underlying utility.\n        extra_args: list of strings or None\n            A list of extra string arguments to be passed to the underlying\n            movie utiltiy. The default is None, which passes the additional\n            argurments in the 'animation.extra_args' rcParam.\n        metadata: dict of string:string or None\n            A dictionary of keys and values for metadata to include in the\n            output file. Some keys that may be of use include:\n            title, artist, genre, subject, copyright, srcform, comment.\n        '''\n        self.fps = fps\n        self.frame_format = 'rgba'\n\n        if codec is None:\n            self.codec = rcParams['animation.codec']\n        else:\n            self.codec = codec\n\n        if bitrate is None:\n            self.bitrate = rcParams['animation.bitrate']\n        else:\n            self.bitrate = bitrate\n\n        if extra_args is None:\n            self.extra_args = list(rcParams[self.args_key])\n        else:\n            self.extra_args = extra_args\n\n        if metadata is None:\n            self.metadata = dict()\n        else:\n            self.metadata = metadata\n\n    @property\n    def frame_size(self):\n        'A tuple (width,height) in pixels of a movie frame.'\n        width_inches, height_inches = self.fig.get_size_inches()\n        return width_inches * self.dpi, height_inches * self.dpi\n\n    def setup(self, fig, outfile, dpi):\n        '''\n        Perform setup for writing the movie file.\n\n        fig: `matplotlib.Figure` instance\n            The figure object that contains the information for frames\n        outfile: string\n            The filename of the resulting movie file\n        dpi: int\n            The DPI (or resolution) for the file.  This controls the size\n            in pixels of the resulting movie file.\n        '''\n        self.outfile = outfile\n        self.fig = fig\n        self.dpi = dpi\n\n        # Run here so that grab_frame() can write the data to a pipe. This\n        # eliminates the need for temp files.\n        self._run()\n\n    def _run(self):\n        # Uses subprocess to call the program for assembling frames into a\n        # movie file.  *args* returns the sequence of command line arguments\n        # from a few configuration options.\n        command = self._args()\n        if verbose.ge('debug'):\n            output = sys.stdout\n        else:\n            output = subprocess.PIPE\n        verbose.report('MovieWriter.run: running command: %s' %\n                       ' '.join(command))\n        self._proc = subprocess.Popen(command, shell=False,\n                                      stdout=output, stderr=output,\n                                      stdin=subprocess.PIPE,\n                                      creationflags=subprocess_creation_flags)\n\n    def finish(self):\n        'Finish any processing for writing the movie.'\n        self.cleanup()\n\n    def grab_frame(self, **savefig_kwargs):\n        '''\n        Grab the image information from the figure and save as a movie frame.\n        All keyword arguments in savefig_kwargs are passed on to the 'savefig'\n        command that saves the figure.\n        '''\n        verbose.report('MovieWriter.grab_frame: Grabbing frame.',\n                       level='debug')\n        try:\n            # Tell the figure to save its data to the sink, using the\n            # frame format and dpi.\n            self.fig.savefig(self._frame_sink(), format=self.frame_format,\n                             dpi=self.dpi, **savefig_kwargs)\n        except RuntimeError:\n            out, err = self._proc.communicate()\n            verbose.report('MovieWriter -- Error '\n                           'running proc:\\n%s\\n%s' % (out,\n                                                      err), level='helpful')\n            raise\n\n    def _frame_sink(self):\n        'Returns the place to which frames should be written.'\n        return self._proc.stdin\n\n    def _args(self):\n        'Assemble list of utility-specific command-line arguments.'\n        return NotImplementedError(\"args needs to be implemented by subclass.\")\n\n    def cleanup(self):\n        'Clean-up and collect the process used to write the movie file.'\n        out, err = self._proc.communicate()\n        self._frame_sink().close()\n        verbose.report('MovieWriter -- '\n                       'Command stdout:\\n%s' % out, level='debug')\n        verbose.report('MovieWriter -- '\n                       'Command stderr:\\n%s' % err, level='debug')\n\n    @classmethod\n    def bin_path(cls):\n        '''\n        Returns the binary path to the commandline tool used by a specific\n        subclass. This is a class method so that the tool can be looked for\n        before making a particular MovieWriter subclass available.\n        '''\n        return rcParams[cls.exec_key]\n\n    @classmethod\n    def isAvailable(cls):\n        '''\n        Check to see if a MovieWriter subclass is actually available by\n        running the commandline tool.\n        '''\n        bin_path = cls.bin_path()\n        if not bin_path:\n            return False\n        try:\n            p = subprocess.Popen(bin_path,\n                             shell=False,\n                             stdout=subprocess.PIPE,\n                             stderr=subprocess.PIPE,\n                             creationflags=subprocess_creation_flags)\n            p.communicate()\n            return True\n        except OSError:\n            return False"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FuncAnimation",
          "code": "def __init__(self, fig, func, frames=None, init_func=None, fargs=None,\n                 save_count=None, **kwargs):\n        if fargs:\n            self._args = fargs\n        else:\n            self._args = ()\n        self._func = func\n\n        # Amount of framedata to keep around for saving movies. This is only\n        # used if we don't know how many frames there will be: in the case\n        # of no generator or in the case of a callable.\n        self.save_count = save_count\n\n        # Set up a function that creates a new iterable when needed. If nothing\n        # is passed in for frames, just use itertools.count, which will just\n        # keep counting from 0. A callable passed in for frames is assumed to\n        # be a generator. An iterable will be used as is, and anything else\n        # will be treated as a number of frames.\n        if frames is None:\n            self._iter_gen = itertools.count\n        elif six.callable(frames):\n            self._iter_gen = frames\n        elif iterable(frames):\n            self._iter_gen = lambda: iter(frames)\n            if hasattr(frames, '__len__'):\n                self.save_count = len(frames)\n        else:\n            self._iter_gen = lambda: xrange(frames).__iter__()\n            self.save_count = frames\n\n        # If we're passed in and using the default, set it to 100.\n        if self.save_count is None:\n            self.save_count = 100\n\n        self._init_func = init_func\n\n        # Needs to be initialized so the draw functions work without checking\n        self._save_seq = []\n\n        TimedAnimation.__init__(self, fig, **kwargs)\n\n        # Need to reset the saved seq, since right now it will contain data\n        # for a single frame from init, which is not what we want.\n        self._save_seq = []"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "isAvailable",
          "class_name": "ImageMagickBase",
          "code": "def isAvailable(cls):\n        '''\n        Check to see if a ImageMagickWriter is actually available\n\n        Done by first checking the windows registry (if applicable) and then\n        running the commandline tool.\n        '''\n        bin_path = cls.bin_path()\n        if bin_path == \"convert\":\n            cls._init_from_registry()\n        return super(ImageMagickBase, cls).isAvailable()"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "class",
          "name": "FFMpegBase",
          "code": "class FFMpegBase(object):\n    exec_key = 'animation.ffmpeg_path'\n    args_key = 'animation.ffmpeg_args'\n\n    @property\n    def output_args(self):\n        # The %dk adds 'k' as a suffix so that ffmpeg treats our bitrate as in\n        # kbps\n        args = ['-vcodec', self.codec]\n        # For h264, the default format is yuv444p, which is not compatible\n        # with quicktime (and others). Specifying yuv420p fixes playback on\n        # iOS,as well as HTML5 video in firefox and safari (on both Win and\n        # OSX). Also fixes internet explorer. This is as of 2015/10/29.\n        if self.codec == 'h264' and '-pix_fmt' not in self.extra_args:\n            args.extend(['-pix_fmt', 'yuv420p'])\n        if self.bitrate > 0:\n            args.extend(['-b', '%dk' % self.bitrate])\n        if self.extra_args:\n            args.extend(self.extra_args)\n        for k, v in six.iteritems(self.metadata):\n            args.extend(['-metadata', '%s=%s' % (k, v)])\n\n        return args + ['-y', self.outfile]"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "output_args",
          "class_name": "ImageMagickBase",
          "code": "def output_args(self):\n        return [self.outfile]"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "to_html5_video",
          "class_name": "Animation",
          "code": "def to_html5_video(self):\n        r'''Returns animation as an HTML5 video tag.\n\n        This saves the animation as an h264 video, encoded in base64\n        directly into the HTML5 video tag. This respects the rc parameters\n        for the writer as well as the bitrate. This also makes use of the\n        ``interval`` to control the speed, and uses the ``repeat``\n        paramter to decide whether to loop.\n        '''\n        VIDEO_TAG = r'''<video {size} {options}>\n  <source type=\"video/mp4\" src=\"data:video/mp4;base64,{video}\">\n  Your browser does not support the video tag.\n</video>'''\n        # Cache the the rendering of the video as HTML\n        if not hasattr(self, '_base64_video'):\n            # First write the video to a tempfile. Set delete to False\n            # so we can re-open to read binary data.\n            with tempfile.NamedTemporaryFile(suffix='.m4v',\n                                             delete=False) as f:\n                # We create a writer manually so that we can get the\n                # appropriate size for the tag\n                Writer = writers[rcParams['animation.writer']]\n                writer = Writer(codec='h264',\n                                bitrate=rcParams['animation.bitrate'],\n                                fps=1000. / self._interval)\n                self.save(f.name, writer=writer)\n\n            # Now open and base64 encode\n            with open(f.name, 'rb') as video:\n                vid64 = encodebytes(video.read())\n                self._base64_video = vid64.decode('ascii')\n                self._video_size = 'width=\"{0}\" height=\"{1}\"'.format(\n                        *writer.frame_size)\n\n            # Now we can remove\n            os.remove(f.name)\n\n        # Default HTML5 options are to autoplay and to display video controls\n        options = ['controls', 'autoplay']\n\n        # If we're set to repeat, make it loop\n        if self.repeat:\n            options.append('loop')\n        return VIDEO_TAG.format(video=self._base64_video,\n                                size=self._video_size,\n                                options=' '.join(options))"
        },
        {
          "file": "lib/matplotlib/axes/_base.py",
          "type": "function",
          "name": "__init__",
          "class_name": "_AxesBase",
          "code": "def __init__(self, fig, rect,\n                 facecolor=None,  # defaults to rc axes.facecolor\n                 frameon=True,\n                 sharex=None,  # use Axes instance's xaxis info\n                 sharey=None,  # use Axes instance's yaxis info\n                 label='',\n                 xscale=None,\n                 yscale=None,\n                 axisbg=None,  # This will be removed eventually\n                 **kwargs\n                 ):\n        \"\"\"\n        Build an :class:`Axes` instance in\n        :class:`~matplotlib.figure.Figure` *fig* with\n        *rect=[left, bottom, width, height]* in\n        :class:`~matplotlib.figure.Figure` coordinates\n\n        Optional keyword arguments:\n\n          ================   =========================================\n          Keyword            Description\n          ================   =========================================\n          *adjustable*       [ 'box' | 'datalim' | 'box-forced']\n          *alpha*            float: the alpha transparency (can be None)\n          *anchor*           [ 'C', 'SW', 'S', 'SE', 'E', 'NE', 'N',\n                               'NW', 'W' ]\n          *aspect*           [ 'auto' | 'equal' | aspect_ratio ]\n          *autoscale_on*     [ *True* | *False* ] whether or not to\n                             autoscale the *viewlim*\n          *axisbelow*        [ *True* | *False* | 'line'] draw the grids\n                             and ticks below or above most other artists,\n                             or below lines but above patches\n          *cursor_props*     a (*float*, *color*) tuple\n          *figure*           a :class:`~matplotlib.figure.Figure`\n                             instance\n          *frame_on*         a boolean - draw the axes frame\n          *label*            the axes label\n          *navigate*         [ *True* | *False* ]\n          *navigate_mode*    [ 'PAN' | 'ZOOM' | None ] the navigation\n                             toolbar button status\n          *position*         [left, bottom, width, height] in\n                             class:`~matplotlib.figure.Figure` coords\n          *sharex*           an class:`~matplotlib.axes.Axes` instance\n                             to share the x-axis with\n          *sharey*           an class:`~matplotlib.axes.Axes` instance\n                             to share the y-axis with\n          *title*            the title string\n          *visible*          [ *True* | *False* ] whether the axes is\n                             visible\n          *xlabel*           the xlabel\n          *xlim*             (*xmin*, *xmax*) view limits\n          *xscale*           [%(scale)s]\n          *xticklabels*      sequence of strings\n          *xticks*           sequence of floats\n          *ylabel*           the ylabel strings\n          *ylim*             (*ymin*, *ymax*) view limits\n          *yscale*           [%(scale)s]\n          *yticklabels*      sequence of strings\n          *yticks*           sequence of floats\n          ================   =========================================\n        \"\"\" % {'scale': ' | '.join(\n            [repr(x) for x in mscale.get_scale_names()])}\n        martist.Artist.__init__(self)\n        if isinstance(rect, mtransforms.Bbox):\n            self._position = rect\n        else:\n            self._position = mtransforms.Bbox.from_bounds(*rect)\n        self._originalPosition = self._position.frozen()\n        # self.set_axes(self)\n        self.axes = self\n        self.set_aspect('auto')\n        self._adjustable = 'box'\n        self.set_anchor('C')\n        self._sharex = sharex\n        self._sharey = sharey\n        if sharex is not None:\n            self._shared_x_axes.join(self, sharex)\n            if sharex._adjustable == 'box':\n                sharex._adjustable = 'datalim'\n                # warnings.warn(\n                #    'shared axes: \"adjustable\" is being changed to \"datalim\"')\n            self._adjustable = 'datalim'\n        if sharey is not None:\n            self._shared_y_axes.join(self, sharey)\n            if sharey._adjustable == 'box':\n                sharey._adjustable = 'datalim'\n                # warnings.warn(\n                #    'shared axes: \"adjustable\" is being changed to \"datalim\"')\n            self._adjustable = 'datalim'\n        self.set_label(label)\n        self.set_figure(fig)\n\n        self.set_axes_locator(kwargs.get(\"axes_locator\", None))\n\n        self.spines = self._gen_axes_spines()\n\n        # this call may differ for non-sep axes, e.g., polar\n        self._init_axis()\n        if axisbg is not None and facecolor is not None:\n            raise TypeError('Both axisbg and facecolor are not None. '\n                            'These keywords are aliases, only one may be '\n                            'provided.')\n        if axisbg is not None:\n            cbook.warn_deprecated(\n                '2.0', name='axisbg', alternative='facecolor')\n            facecolor = axisbg\n        if facecolor is None:\n            facecolor = rcParams['axes.facecolor']\n        self._facecolor = facecolor\n        self._frameon = frameon\n        self._axisbelow = rcParams['axes.axisbelow']\n\n        self._rasterization_zorder = None\n\n        self._hold = rcParams['axes.hold']\n        self._connected = {}  # a dict from events to (id, func)\n        self.cla()\n        # funcs used to format x and y - fall back on major formatters\n        self.fmt_xdata = None\n        self.fmt_ydata = None\n\n        self.set_cursor_props((1, 'k'))  # set the cursor properties for axes\n\n        self._cachedRenderer = None\n        self.set_navigate(True)\n        self.set_navigate_mode(None)\n\n        if xscale:\n            self.set_xscale(xscale)\n        if yscale:\n            self.set_yscale(yscale)\n\n        if len(kwargs):\n            self.update(kwargs)\n\n        if self.xaxis is not None:\n            self._xcid = self.xaxis.callbacks.connect('units finalize',\n                                                      self.relim)\n\n        if self.yaxis is not None:\n            self._ycid = self.yaxis.callbacks.connect('units finalize',\n                                                      self.relim)\n        self.tick_params(top=rcParams['xtick.top'],\n                         bottom=rcParams['xtick.bottom'],\n                         left=rcParams['ytick.left'],\n                         right=rcParams['ytick.right'])"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "__init__",
          "class_name": "SubplotToolQt",
          "code": "def __init__(self, targetfig, parent):\n        UiSubplotTool.__init__(self, None)\n\n        self.targetfig = targetfig\n        self.parent = parent\n        self.donebutton.clicked.connect(self.close)\n        self.resetbutton.clicked.connect(self.reset)\n        self.tightlayout.clicked.connect(self.functight)\n\n        # constraints\n        self.sliderleft.valueChanged.connect(self.sliderright.setMinimum)\n        self.sliderright.valueChanged.connect(self.sliderleft.setMaximum)\n        self.sliderbottom.valueChanged.connect(self.slidertop.setMinimum)\n        self.slidertop.valueChanged.connect(self.sliderbottom.setMaximum)\n\n        self.defaults = {}\n        for attr in ('left', 'bottom', 'right', 'top', 'wspace', 'hspace', ):\n            self.defaults[attr] = getattr(self.targetfig.subplotpars, attr)\n            slider = getattr(self, 'slider' + attr)\n            slider.setMinimum(0)\n            slider.setMaximum(1000)\n            slider.setSingleStep(5)\n            slider.valueChanged.connect(getattr(self, 'func' + attr))\n\n        self._setSliderPositions()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "funcleft",
          "class_name": "SubplotToolQt",
          "code": "def funcleft(self, val):\n        if val == self.sliderright.value():\n            val -= 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(left=val)\n        self.leftvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "funcright",
          "class_name": "SubplotToolQt",
          "code": "def funcright(self, val):\n        if val == self.sliderleft.value():\n            val += 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(right=val)\n        self.rightvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "funcbottom",
          "class_name": "SubplotToolQt",
          "code": "def funcbottom(self, val):\n        if val == self.slidertop.value():\n            val -= 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(bottom=val)\n        self.bottomvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "functop",
          "class_name": "SubplotToolQt",
          "code": "def functop(self, val):\n        if val == self.sliderbottom.value():\n            val += 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(top=val)\n        self.topvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/cbook.py",
          "type": "function",
          "name": "issubclass_safe",
          "class_name": null,
          "code": "def issubclass_safe(x, klass):\n    'return issubclass(x, klass) and return False on a TypeError'\n\n    try:\n        return issubclass(x, klass)\n    except TypeError:\n        return False"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "changed",
          "class_name": "ContourSet",
          "code": "def changed(self):\n        tcolors = [(tuple(rgba),)\n                   for rgba in self.to_rgba(self.cvalues, alpha=self.alpha)]\n        self.tcolors = tcolors\n        hatches = self.hatches * len(tcolors)\n        for color, hatch, collection in zip(tcolors, hatches,\n                                            self.collections):\n            if self.filled:\n                collection.set_facecolor(color)\n                # update the collection's hatch (may be None)\n                collection.set_hatch(hatch)\n            else:\n                collection.set_color(color)\n        for label, cv in zip(self.labelTexts, self.labelCValues):\n            label.set_alpha(self.alpha)\n            label.set_color(self.labelMappable.to_rgba(cv))\n        # add label colors\n        cm.ScalarMappable.changed(self)"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_contour_level_args",
          "class_name": "ContourSet",
          "code": "def _contour_level_args(self, z, args):\n        \"\"\"\n        Determine the contour levels and store in self.levels.\n        \"\"\"\n        if self.filled:\n            fn = 'contourf'\n        else:\n            fn = 'contour'\n        self._auto = False\n        if self.levels is None:\n            if len(args) == 0:\n                lev = self._autolev(z, 7)\n            else:\n                level_arg = args[0]\n                try:\n                    if type(level_arg) == int:\n                        lev = self._autolev(z, level_arg)\n                    else:\n                        lev = np.asarray(level_arg).astype(np.float64)\n                except:\n                    raise TypeError(\n                        \"Last %s arg must give levels; see help(%s)\" %\n                        (fn, fn))\n            self.levels = lev\n        if self.filled and len(self.levels) < 2:\n            raise ValueError(\"Filled contours require at least 2 levels.\")\n\n        if len(self.levels) > 1 and np.amin(np.diff(self.levels)) <= 0.0:\n            if hasattr(self, '_corner_mask') and self._corner_mask == 'legacy':\n                warnings.warn(\"Contour levels are not increasing\")\n            else:\n                raise ValueError(\"Contour levels must be increasing\")"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_contour_args",
          "class_name": "QuadContourSet",
          "code": "def _contour_args(self, args, kwargs):\n        if self.filled:\n            fn = 'contourf'\n        else:\n            fn = 'contour'\n        Nargs = len(args)\n        if Nargs <= 2:\n            z = ma.asarray(args[0], dtype=np.float64)\n            x, y = self._initialize_x_y(z)\n            args = args[1:]\n        elif Nargs <= 4:\n            x, y, z = self._check_xyz(args[:3], kwargs)\n            args = args[3:]\n        else:\n            raise TypeError(\"Too many arguments to %s; see help(%s)\" %\n                            (fn, fn))\n        z = ma.masked_invalid(z, copy=False)\n        self.zmax = ma.maximum(z)\n        self.zmin = ma.minimum(z)\n        if self.logscale and self.zmin <= 0:\n            z = ma.masked_where(z <= 0, z)\n            warnings.warn('Log scale: values of z <= 0 have been masked')\n            self.zmin = z.min()\n        self._contour_level_args(z, args)\n        return (x, y, z)"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "class",
          "name": "AutoDateFormatter",
          "code": "class AutoDateFormatter(ticker.Formatter):\n    \"\"\"\n    This class attempts to figure out the best format to use.  This is\n    most useful when used with the :class:`AutoDateLocator`.\n\n\n    The AutoDateFormatter has a scale dictionary that maps the scale\n    of the tick (the distance in days between one major tick) and a\n    format string.  The default looks like this::\n\n        self.scaled = {\n            DAYS_PER_YEAR: rcParams['date.autoformat.year'],\n            DAYS_PER_MONTH: rcParams['date.autoformat.month'],\n            1.0: rcParams['date.autoformat.day'],\n            1. / HOURS_PER_DAY: rcParams['date.autoformat.hour'],\n            1. / (MINUTES_PER_DAY): rcParams['date.autoformat.minute'],\n            1. / (SEC_PER_DAY): rcParams['date.autoformat.second']}\n\n\n    The algorithm picks the key in the dictionary that is >= the\n    current scale and uses that format string.  You can customize this\n    dictionary by doing::\n\n\n    >>> locator = AutoDateLocator()\n    >>> formatter = AutoDateFormatter(locator)\n    >>> formatter.scaled[1/(24.*60.)] = '%M:%S' # only show min and sec\n\n    A custom :class:`~matplotlib.ticker.FuncFormatter` can also be used.\n    The following example shows how to use a custom format function to strip\n    trailing zeros from decimal seconds and adds the date to the first\n    ticklabel::\n\n        >>> def my_format_function(x, pos=None):\n        ...     x = matplotlib.dates.num2date(x)\n        ...     if pos == 0:\n        ...         fmt = '%D %H:%M:%S.%f'\n        ...     else:\n        ...         fmt = '%H:%M:%S.%f'\n        ...     label = x.strftime(fmt)\n        ...     label = label.rstrip(\"0\")\n        ...     label = label.rstrip(\".\")\n        ...     return label\n        >>> from matplotlib.ticker import FuncFormatter\n        >>> formatter.scaled[1/(24.*60.)] = FuncFormatter(my_format_function)\n    \"\"\"\n\n    # This can be improved by providing some user-level direction on\n    # how to choose the best format (precedence, etc...)\n\n    # Perhaps a 'struct' that has a field for each time-type where a\n    # zero would indicate \"don't show\" and a number would indicate\n    # \"show\" with some sort of priority.  Same priorities could mean\n    # show all with the same priority.\n\n    # Or more simply, perhaps just a format string for each\n    # possibility...\n\n    def __init__(self, locator, tz=None, defaultfmt='%Y-%m-%d'):\n        \"\"\"\n        Autoformat the date labels.  The default format is the one to use\n        if none of the values in ``self.scaled`` are greater than the unit\n        returned by ``locator._get_unit()``.\n        \"\"\"\n        self._locator = locator\n        self._tz = tz\n        self.defaultfmt = defaultfmt\n        self._formatter = DateFormatter(self.defaultfmt, tz)\n        self.scaled = {DAYS_PER_YEAR: rcParams['date.autoformatter.year'],\n                       DAYS_PER_MONTH: rcParams['date.autoformatter.month'],\n                       1.0: rcParams['date.autoformatter.day'],\n                       1. / HOURS_PER_DAY: rcParams['date.autoformatter.hour'],\n                       1. / (MINUTES_PER_DAY):\n                           rcParams['date.autoformatter.minute'],\n                       1. / (SEC_PER_DAY):\n                           rcParams['date.autoformatter.second']}\n\n    def __call__(self, x, pos=None):\n        locator_unit_scale = float(self._locator._get_unit())\n        fmt = self.defaultfmt\n\n        # Pick the first scale which is greater than the locator unit.\n        for possible_scale in sorted(self.scaled):\n            if possible_scale >= locator_unit_scale:\n                fmt = self.scaled[possible_scale]\n                break\n\n        if isinstance(fmt, six.string_types):\n            self._formatter = DateFormatter(fmt, self._tz)\n            result = self._formatter(x, pos)\n        elif six.callable(fmt):\n            result = fmt(x, pos)\n        else:\n            raise TypeError('Unexpected type passed to {0!r}.'.format(self))\n\n        return result"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "__init__",
          "class_name": "MicrosecondLocator",
          "code": "def __init__(self, interval=1, tz=None):\n        \"\"\"\n        *interval* is the interval between each iteration.  For\n        example, if ``interval=2``, mark every second microsecond.\n\n        \"\"\"\n        self._interval = interval\n        self._wrapped_locator = ticker.MultipleLocator(interval)\n        self.tz = tz"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "set_data",
          "class_name": "PcolorImage",
          "code": "def set_data(self, x, y, A):\n        A = cbook.safe_masked_invalid(A)\n        if x is None:\n            x = np.arange(0, A.shape[1]+1, dtype=np.float64)\n        else:\n            x = np.asarray(x, np.float64).ravel()\n        if y is None:\n            y = np.arange(0, A.shape[0]+1, dtype=np.float64)\n        else:\n            y = np.asarray(y, np.float64).ravel()\n\n        if A.shape[:2] != (y.size-1, x.size-1):\n            raise ValueError(\n                \"Axes don't match array shape. Got %s, expected %s.\" %\n                (A.shape[:2], (y.size - 1, x.size - 1)))\n        if A.ndim not in [2, 3]:\n            raise ValueError(\"A must be 2D or 3D\")\n        if A.ndim == 3 and A.shape[2] == 1:\n            A.shape = A.shape[:2]\n        self.is_grayscale = False\n        if A.ndim == 3:\n            if A.shape[2] in [3, 4]:\n                if ((A[:, :, 0] == A[:, :, 1]).all() and\n                        (A[:, :, 0] == A[:, :, 2]).all()):\n                    self.is_grayscale = True\n            else:\n                raise ValueError(\"3D arrays must have RGB or RGBA as last dim\")\n        self._A = A\n        self._Ax = x\n        self._Ay = y\n        self._rgbacache = None\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "set_data",
          "class_name": "PcolorImage",
          "code": "def set_data(self, x, y, A):\n        A = cbook.safe_masked_invalid(A)\n        if x is None:\n            x = np.arange(0, A.shape[1]+1, dtype=np.float64)\n        else:\n            x = np.asarray(x, np.float64).ravel()\n        if y is None:\n            y = np.arange(0, A.shape[0]+1, dtype=np.float64)\n        else:\n            y = np.asarray(y, np.float64).ravel()\n\n        if A.shape[:2] != (y.size-1, x.size-1):\n            raise ValueError(\n                \"Axes don't match array shape. Got %s, expected %s.\" %\n                (A.shape[:2], (y.size - 1, x.size - 1)))\n        if A.ndim not in [2, 3]:\n            raise ValueError(\"A must be 2D or 3D\")\n        if A.ndim == 3 and A.shape[2] == 1:\n            A.shape = A.shape[:2]\n        self.is_grayscale = False\n        if A.ndim == 3:\n            if A.shape[2] in [3, 4]:\n                if ((A[:, :, 0] == A[:, :, 1]).all() and\n                        (A[:, :, 0] == A[:, :, 2]).all()):\n                    self.is_grayscale = True\n            else:\n                raise ValueError(\"3D arrays must have RGB or RGBA as last dim\")\n        self._A = A\n        self._Ax = x\n        self._Ay = y\n        self._rgbacache = None\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "__init__",
          "class_name": "BboxImage",
          "code": "def __init__(self, bbox,\n                 cmap=None,\n                 norm=None,\n                 interpolation=None,\n                 origin=None,\n                 filternorm=1,\n                 filterrad=4.0,\n                 resample=False,\n                 interp_at_native=True,\n                 **kwargs\n                 ):\n\n        \"\"\"\n        cmap is a colors.Colormap instance\n        norm is a colors.Normalize instance to map luminance to 0-1\n\n        interp_at_native is a flag that determines whether or not\n        interpolation should still be applied when the image is\n        displayed at its native resolution.  A common use case for this\n        is when displaying an image for annotational purposes; it is\n        treated similarly to Photoshop (interpolation is only used when\n        displaying the image at non-native resolutions).\n\n\n        kwargs are an optional list of Artist keyword args\n\n        \"\"\"\n        super(BboxImage, self).__init__(\n            None,\n            cmap=cmap,\n            norm=norm,\n            interpolation=interpolation,\n            origin=origin,\n            filternorm=filternorm,\n            filterrad=filterrad,\n            resample=resample,\n            **kwargs\n        )\n\n        self.bbox = bbox\n        self.interp_at_native = interp_at_native\n        self._transform = IdentityTransform()"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "_check_unsampled_image",
          "class_name": "PcolorImage",
          "code": "def _check_unsampled_image(self, renderer):\n        return False"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "make_image",
          "class_name": "BboxImage",
          "code": "def make_image(self, renderer, magnification=1.0, unsampled=False):\n        width, height = renderer.get_canvas_width_height()\n\n        bbox_in = self.get_window_extent(renderer).frozen()\n        bbox_in._points /= [width, height]\n        bbox_out = self.get_window_extent(renderer)\n        clip = Bbox([[0, 0], [width, height]])\n        self._transform = BboxTransform(Bbox([[0, 0], [1, 1]]), clip)\n\n        return self._make_image(\n            self._A,\n            bbox_in, bbox_out, clip, magnification, unsampled=unsampled)"
        },
        {
          "file": "lib/matplotlib/rcsetup.py",
          "type": "function",
          "name": "validate_animation_writer_path",
          "class_name": null,
          "code": "def validate_animation_writer_path(p):\n    # Make sure it's a string and then figure out if the animations\n    # are already loaded and reset the writers (which will validate\n    # the path on next call)\n    if not isinstance(p, six.text_type):\n        raise ValueError(\"path must be a (unicode) string\")\n    from sys import modules\n    # set dirty, so that the next call to the registry will re-evaluate\n    # the state.\n    # only set dirty if already loaded. If not loaded, the load will\n    # trigger the checks.\n    if \"matplotlib.animation\" in modules:\n        modules[\"matplotlib.animation\"].writers.set_dirty()\n    return p"
        },
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "__init__",
          "class_name": "OldAutoLocator",
          "code": "def __init__(self):\n        self._locator = LinearLocator()"
        },
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "tick_values",
          "class_name": "OldAutoLocator",
          "code": "def tick_values(self, vmin, vmax):\n        raise NotImplementedError('Cannot get tick locations for a '\n                                  '%s type.' % type(self))"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "check",
          "class_name": "PdfToPs",
          "code": "def check(self):\n        try:\n            output = check_output('pdftops -v', shell=True,\n                                  stderr=subprocess.STDOUT)\n            for line in output.splitlines():\n                line = line.decode()\n                if 'version' in line:\n                    return \"version %s\" % line.split()[2]\n        except (IndexError, ValueError, subprocess.CalledProcessError):\n            pass\n\n        raise CheckFailed()"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "check",
          "class_name": "PdfToPs",
          "code": "def check(self):\n        try:\n            output = check_output('pdftops -v', shell=True,\n                                  stderr=subprocess.STDOUT)\n            for line in output.splitlines():\n                line = line.decode()\n                if 'version' in line:\n                    return \"version %s\" % line.split()[2]\n        except (IndexError, ValueError, subprocess.CalledProcessError):\n            pass\n\n        raise CheckFailed()"
        }
      ]
    },
    {
      "pr_number": 20009,
      "pr_title": "Fix removal of shared polar axes.",
      "pr_body": "There's really two separate fixes here:\r\n\r\n- Move isDefault_{maj,min}{loc,fmt} tracking to the Ticker instances,\r\n  where they logically belong (note the previous need to additionally\r\n  track them manually on axes removal, when that info was tracked on\r\n  the Axis).  This has the side effect of fixing removal of sharex'd\r\n  polar axes, as ThetaLocators rely on _AxisWrappers which don't have\r\n  that isDefault attribute.  (Note that the patch would have resulted\r\n  in a net decrease of lines of code if it didn't need to maintain\r\n  backcompat on isDefault_foos).  Closes #19988.  Split out of #13482.\r\n\r\n- Ensure that RadialLocator correctly propagates Axis information to\r\n  the linear locator it wraps (consistently with ThetaLocator), so that\r\n  when an axes is removed the wrapped linear locator doesn't stay\r\n  pointing at an obsolete axes.  This, together with the first patch,\r\n  fixes removal of sharey'd polar axes.  Closes #19989.\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n\r\n- [ ] Has pytest style unit tests (and `pytest` passes).\r\n- [ ] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [ ] New features are documented, with examples if plot related.\r\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [ ] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [ ] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [ ] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
      "issue_id": 19989,
      "issue_title": "Removal of y-shared polar axes causes crash at draw time",
      "issue_body": "<!--To help us understand and resolve your issue, please fill out the form to the best of your ability.-->\r\n<!--You can feel free to delete the sections that do not apply.-->\r\n\r\n### Bug report\r\n\r\n**Bug summary**\r\n\r\nAll's in the title.\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug.\r\nPlease make sure to minimize the number of dependencies required, and provide\r\nany necessary plotted data.\r\nAvoid using threads, as Matplotlib is (explicitly) not thread-safe.-->\r\n\r\n```python\r\nfrom pylab import *\r\nax1, ax2 = gcf().subplots(2, sharey=True, subplot_kw={\"projection\": \"polar\"}); ax2.remove(); show()\r\n```\r\n\r\n**Actual outcome**\r\n\r\n<!--The output produced by the above code, which may be a screenshot, console output, etc.-->\r\n\r\n```\r\nTraceback (most recent call last):\r\nTraceback (most recent call last):\r\n  File \".../path/to/matplotlib/backends/backend_qt5.py\", line 440, in _draw_idle\r\n    self.draw()\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 269, in draw\r\n    self.get_renderer(_ensure_cleared=True, _ensure_drawn=True)\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 261, in get_renderer\r\n    return self._get_cached_or_new_renderer(\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 256, in _get_cached_or_new_renderer\r\n    self.figure.draw(renderer)\r\n  File \".../path/to/matplotlib/artist.py\", line 74, in draw_wrapper\r\n    result = draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/figure.py\", line 2730, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \".../path/to/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 447, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 447, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 994, in draw\r\n    super().draw(renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 421, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/axes/_base.py\", line 3102, in draw\r\n    mimage._draw_list_compositing_images(renderer, self, artists)\r\n  File \".../path/to/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/axis.py\", line 1124, in draw\r\n    ticks_to_draw = self._update_ticks()\r\n  File \".../path/to/matplotlib/axis.py\", line 1011, in _update_ticks\r\n    major_locs = self.get_majorticklocs()\r\n  File \".../path/to/matplotlib/axis.py\", line 1243, in get_majorticklocs\r\n    return self.major.locator()\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 432, in __call__\r\n    return [tick for tick in self.base() if tick > rorigin]\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 432, in __call__\r\n    return [tick for tick in self.base() if tick > rorigin]\r\n  File \".../path/to/matplotlib/ticker.py\", line 2265, in __call__\r\n    return self.tick_values(vmin, vmax)\r\n  File \".../path/to/matplotlib/ticker.py\", line 2273, in tick_values\r\n    locs = self._raw_ticks(vmin, vmax)\r\n  File \".../path/to/matplotlib/ticker.py\", line 2212, in _raw_ticks\r\n    nbins = np.clip(self.axis.get_tick_space(),\r\n  File \".../path/to/matplotlib/axis.py\", line 2513, in get_tick_space\r\n    length = ((ends[1][1] - ends[0][1]) / self.axes.figure.dpi) * 72\r\nAttributeError: 'NoneType' object has no attribute 'dpi'\r\n```\r\n\r\n**Expected outcome**\r\n\r\nNormal axes removal.\r\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Operating system: linux\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): head\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): mplcairo\r\n  * Python version: 39\r\n  * Jupyter version (if applicable): \r\n  * Other libraries: \r\n\r\n(Note that this is a separate issue from https://github.com/matplotlib/matplotlib/issues/19988 as the root cause seems very different.)\r\n\r\nEdit: I have a fix, but the test is a bit simpler if https://github.com/matplotlib/matplotlib/pull/19994 goes in first.",
      "issue_closed_at": "2021-04-29T20:19:03Z",
      "base_commit": "a94acb3d2c6829d7ce3a4ef2cc33058c677e0526",
      "changes": [
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "__init__",
          "class_name": "YAxis",
          "code": "def __init__(self, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        # x in display coords, y in axes coords (to be updated at draw time by\n        # _update_label_positions and _update_offset_text_position).\n        self.label.set(\n            x=0, y=0.5,\n            verticalalignment='bottom', horizontalalignment='center',\n            rotation='vertical', rotation_mode='anchor',\n            transform=mtransforms.blended_transform_factory(\n                mtransforms.IdentityTransform(), self.axes.transAxes),\n        )\n        self.label_position = 'left'\n        # x in axes coords, y in display coords(!).\n        self.offsetText.set(\n            x=0, y=0.5,\n            verticalalignment='baseline', horizontalalignment='left',\n            transform=mtransforms.blended_transform_factory(\n                self.axes.transAxes, mtransforms.IdentityTransform()),\n            fontsize=mpl.rcParams['ytick.labelsize'],\n            color=mpl.rcParams['ytick.color'],\n        )\n        self.offset_text_position = 'left'"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_reset_locators_and_formatters",
          "class_name": "FigureBase",
          "code": "def _reset_locators_and_formatters(axis):\n            # Set the formatters and locators to be associated with axis\n            # (where previously they may have been associated with another\n            # Axis instance)\n            #\n            # Because set_major_formatter() etc. force isDefault_* to be False,\n            # we have to manually check if the original formatter was a\n            # default and manually set isDefault_* if that was the case.\n            majfmt = axis.get_major_formatter()\n            isDefault = majfmt.axis.isDefault_majfmt\n            axis.set_major_formatter(majfmt)\n            if isDefault:\n                majfmt.axis.isDefault_majfmt = True\n\n            majloc = axis.get_major_locator()\n            isDefault = majloc.axis.isDefault_majloc\n            axis.set_major_locator(majloc)\n            if isDefault:\n                majloc.axis.isDefault_majloc = True\n\n            minfmt = axis.get_minor_formatter()\n            isDefault = majloc.axis.isDefault_minfmt\n            axis.set_minor_formatter(minfmt)\n            if isDefault:\n                minfmt.axis.isDefault_minfmt = True\n\n            minloc = axis.get_minor_locator()\n            isDefault = majloc.axis.isDefault_minloc\n            axis.set_minor_locator(minloc)\n            if isDefault:\n                minloc.axis.isDefault_minloc = True"
        },
        {
          "file": "lib/matplotlib/projections/polar.py",
          "type": "function",
          "name": "__init__",
          "class_name": "PolarAxes",
          "code": "def __init__(self, *args,\n                 theta_offset=0, theta_direction=1, rlabel_position=22.5,\n                 **kwargs):\n        # docstring inherited\n        self._default_theta_offset = theta_offset\n        self._default_theta_direction = theta_direction\n        self._default_rlabel_position = np.deg2rad(rlabel_position)\n        super().__init__(*args, **kwargs)\n        self.use_sticky_edges = True\n        self.set_aspect('equal', adjustable='box', anchor='C')\n        self.cla()"
        }
      ]
    }
  ]
}