{
  "instance_id": "matplotlib__matplotlib-23314",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-06-21T02:41:34Z",
  "problem_statement": "[Bug]: set_visible() not working for 3d projection \n### Bug summary\r\n\r\nin the subplot projection=\"3d\" the set_visible function doesn't work even if the value is set to False\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nfrom matplotlib.gridspec import GridSpec\r\n\r\nfig, (ax1, ax2) = plt.subplots(1, 2, subplot_kw={'projection': '3d'})\r\nax1.scatter(1,1,1)\r\nax2.scatter(1,1,1, c='r')\r\nax1.set_visible(False)\r\n\r\nplt.show()\r\n# Thanks Tim for your help! \r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\nthe subplot remains visible which should not happen if the value is set to False\r\n\r\n### Expected outcome\r\n\r\nthe subplot is not visible if the value is set to False\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\n_No response_\r\n\r\n### Matplotlib Version\r\n\r\n3.4.2\r\n\r\n### Matplotlib Backend\r\n\r\nQt5Agg\r\n\r\n### Python version\r\n\r\n3.8.10\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\n_No response_\n",
  "patch": "diff --git a/lib/mpl_toolkits/mplot3d/axes3d.py b/lib/mpl_toolkits/mplot3d/axes3d.py\n--- a/lib/mpl_toolkits/mplot3d/axes3d.py\n+++ b/lib/mpl_toolkits/mplot3d/axes3d.py\n@@ -387,6 +387,8 @@ def apply_aspect(self, position=None):\n \n     @martist.allow_rasterization\n     def draw(self, renderer):\n+        if not self.get_visible():\n+            return\n         self._unstale_viewLim()\n \n         # draw the background patch\n",
  "similar_bug_items": [
    {
      "pr_number": 19949,
      "pr_title": "FIX: subfigure indexing error",
      "pr_body": "## PR Summary\r\n\r\nCloses #19947.  \r\n\r\nThe indexing to locate subfigures was wildly off.  Worked fine for 2x2 but was incorrect for anything larger.  \r\n\r\nThis PR fixes that and adds a test.\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": 19947,
      "issue_title": "Figure.subfigures dont show/update correctly",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\nWhen using Figure.subfigures(n, m) for any n or m larger than 2 the resulting figure does not draw properly when shown. \r\nInstead only the subfigures at [0:2, 0] and [0:2, -1] are drawn properly.\r\n\r\n**Code for reproduction**\r\n```python\r\nimport matplotlib.pyplot as plt\r\n\r\nfig = plt.figure()\r\nsub_figs = fig.subfigures(3, 3)\r\n\r\ncolors = [\"red\", \"blue\", \"magenta\", \"gold\", \"green\", \"grey\", \"orange\", \"pink\", \"navy\", \"lightblue\", \"black\"]\r\n\r\nfor i, (color, subfig) in enumerate(zip(colors,sub_figs.flatten())):\r\n    subfig.set_facecolor(color)\r\n    subfig.suptitle(f\"subfig: {i}\")\r\n\r\nplt.show()\r\n\r\n```\r\n\r\n######### for (n, m) figures\r\n```python \r\nimport matplotlib.pyplot as plt\r\n\r\nfig = plt.figure()\r\nsub_figs = fig.subfigures(9, 10)\r\n\r\nfor i, subfig in enumerate(sub_figs.flatten()):\r\n    subfig.set_facecolor(\"grey\")\r\n    subfig.suptitle(f\"subfig: {i}\")\r\n\r\nplt.show()\r\n\r\n```\r\n**Actual outcome**\r\n\r\n![image](https://user-images.githubusercontent.com/29613344/114410466-ae3edb80-9bab-11eb-92a5-aa07e68f4265.png)\r\n\r\nOnly the subfigures at [0:2, 0] and [0:2, -1] are drawn properly for any (n, m) figure.\r\n\r\n**Expected outcome**\r\n3x3 grid with the proper face colors. Similar results for 2x3 or 3x2 or larger grids. \r\n\r\n**Matplotlib version**\r\n  * Operating system: Fedora (UNIX)\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): 3.4.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): module://backend_interagg\r\n  * Python version: 3.7\r\n  * Jupyter version (if applicable): NA\r\n  * Other libraries: NA\r\n\r\nBoth python and Matplotlib from conda with manual update to Matplotlib version 3.4.1\r\nconda channel: conda-forge\r\n\r\n",
      "issue_closed_at": "2021-04-20T11:49:43Z",
      "base_commit": "92ce41defddde23e2fd452c39b6e18d61a5ae5e5",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_redo_transform_rel_fig",
          "class_name": "SubFigure",
          "code": "def _redo_transform_rel_fig(self, bbox=None):\n        \"\"\"\n        Make the transSubfigure bbox relative to Figure transform.\n\n        Parameters\n        ----------\n        bbox : bbox or None\n            If not None, then the bbox is used for relative bounding box.\n            Otherwise it is calculated from the subplotspec.\n        \"\"\"\n\n        if bbox is not None:\n            self.bbox_relative.p0 = bbox.p0\n            self.bbox_relative.p1 = bbox.p1\n            return\n\n        gs = self._subplotspec.get_gridspec()\n        # need to figure out *where* this subplotspec is.\n        wr = gs.get_width_ratios()\n        hr = gs.get_height_ratios()\n        nrows, ncols = gs.get_geometry()\n        if wr is None:\n            wr = np.ones(ncols)\n        else:\n            wr = np.array(wr)\n        if hr is None:\n            hr = np.ones(nrows)\n        else:\n            hr = np.array(hr)\n        widthf = np.sum(wr[self._subplotspec.colspan]) / np.sum(wr)\n        heightf = np.sum(hr[self._subplotspec.rowspan]) / np.sum(hr)\n\n        x0 = 0\n        if not self._subplotspec.is_first_col():\n            x0 += np.sum(wr[self._subplotspec.colspan.start - 1]) / np.sum(wr)\n\n        y0 = 0\n        if not self._subplotspec.is_last_row():\n            y0 += 1 - (np.sum(hr[self._subplotspec.rowspan.stop - 1]) /\n                       np.sum(hr))\n\n        if self.bbox_relative is None:\n            self.bbox_relative = Bbox.from_bounds(x0, y0, widthf, heightf)\n        else:\n            self.bbox_relative.p0 = (x0, y0)\n            self.bbox_relative.p1 = (x0 + widthf, y0 + heightf)"
        }
      ]
    },
    {
      "pr_number": 14014,
      "pr_title": "Disallow figure argument for pyplot.subplot() and Figure.add_subplot()",
      "pr_body": "## PR Summary\r\n\r\nCloses #14011.\r\n\r\n`Figure.add_subplot()` and `pyplot.subplot()` have technically accepted a `figure` keyword argument by allowing all keywords from the `Axes` constructor.\r\n\r\nIn this context, supplying a figure does not make sense since the axes should be bound to `self` or the current figure respecively.\r\n\r\nI'm daring to go without a deprecation since this anyway did only work so far if the user supplied the same figure that would have been used without the parameter. Please let me know if that's too bold and we should deprecate this first.",
      "issue_id": 14011,
      "issue_title": "TypeError on plt.subplot(figure=plt.figure())",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\nWhen calling `plt.subplot` with a `figure` argument, we get:\r\n```\r\nTypeError: process_projection_requirements() got multiple values for argument 'figure'\r\n```\r\nThe argument 'figure' is specified in the [doc](https://matplotlib.org/api/_as_gen/matplotlib.pyplot.subplot.html)\r\n\r\nThe function `plt.subplot` uses `plt.gcf()` to get the current figure. It should check whether a figure was provided, eg. `fig = kwargs.pop('figure') if kwargs.get('figure') else plt.gcf()`\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nplt.subplot(figure=plt.figure())\r\n# OR EVEN\r\nplt.subplot(figure=None)\r\n```\r\n\r\n**Actual outcome**\r\n```                              \r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n<ipython-input-30-bdfa817205a7> in <module>\r\n----> 1 plt.subplot(figure=None)\r\n\r\n~/bin/anaconda3/lib/python3.7/site-packages/matplotlib/pyplot.py in subplot(*args, **kwargs)\r\n   1082 \r\n   1083     fig = gcf()\r\n-> 1084     a = fig.add_subplot(*args, **kwargs)\r\n   1085     bbox = a.bbox\r\n   1086     byebye = []\r\n\r\n~/bin/anaconda3/lib/python3.7/site-packages/matplotlib/figure.py in add_subplot(self, *args, **kwargs)\r\n   1347         else:\r\n   1348             projection_class, kwargs, key = process_projection_requirements(\r\n-> 1349                 self, *args, **kwargs)\r\n   1350 \r\n   1351             # try to find the axes with this key in the stack\r\n\r\nTypeError: process_projection_requirements() got multiple values for argument 'figure'\r\n```\r\n\r\n**Matplotlib version**\r\n  * Operating system: Linux\r\n  * Matplotlib version: 3.0.2 (with conda and with pip)\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: tested on 3.7.2 and 3.5\r\n\r\n",
      "issue_closed_at": "2019-05-28T22:25:29Z",
      "base_commit": "559925e3ec43a5cbe1697a4496482d38d8489f68",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "add_subplot",
          "class_name": "Figure",
          "code": "def add_subplot(self, *args, **kwargs):\n        \"\"\"\n        Add an `~.axes.Axes` to the figure as part of a subplot arrangement.\n\n        Call signatures::\n\n           add_subplot(nrows, ncols, index, **kwargs)\n           add_subplot(pos, **kwargs)\n           add_subplot(ax)\n           add_subplot()\n\n        Parameters\n        ----------\n        *args\n            Either a 3-digit integer or three separate integers\n            describing the position of the subplot. If the three\n            integers are *nrows*, *ncols*, and *index* in order, the\n            subplot will take the *index* position on a grid with *nrows*\n            rows and *ncols* columns. *index* starts at 1 in the upper left\n            corner and increases to the right.\n\n            *pos* is a three digit integer, where the first digit is the\n            number of rows, the second the number of columns, and the third\n            the index of the subplot. i.e. fig.add_subplot(235) is the same as\n            fig.add_subplot(2, 3, 5). Note that all integers must be less than\n            10 for this form to work.\n\n            If no positional arguments are passed, defaults to (1, 1, 1).\n\n        projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n            The projection type of the subplot (`~.axes.Axes`). *str* is the\n            name of a custom projection, see `~matplotlib.projections`. The\n            default None results in a 'rectilinear' projection.\n\n        polar : boolean, optional\n            If True, equivalent to projection='polar'.\n\n        sharex, sharey : `~.axes.Axes`, optional\n            Share the x or y `~matplotlib.axis` with sharex and/or sharey.\n            The axis will have the same limits, ticks, and scale as the axis\n            of the shared axes.\n\n        label : str\n            A label for the returned axes.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            This method also takes the keyword arguments for\n            the returned axes base class. The keyword arguments for the\n            rectilinear base class `~.axes.Axes` can be found in\n            the following table but there might also be other keyword\n            arguments if another projection is used.\n            %(Axes)s\n\n        Returns\n        -------\n        axes : an `.axes.SubplotBase` subclass of `~.axes.Axes` (or a \\\n               subclass of `~.axes.Axes`)\n\n            The axes of the subplot. The returned axes base class depends on\n            the projection used. It is `~.axes.Axes` if rectilinear projection\n            are used and `.projections.polar.PolarAxes` if polar projection\n            are used. The returned axes is then a subplot subclass of the\n            base class.\n\n        Notes\n        -----\n        If the figure already has a subplot with key (*args*,\n        *kwargs*) then it will simply make that subplot current and\n        return it.  This behavior is deprecated. Meanwhile, if you do\n        not want this behavior (i.e., you want to force the creation of a\n        new subplot), you must use a unique set of args and kwargs.  The axes\n        *label* attribute has been exposed for this purpose: if you want\n        two subplots that are otherwise identical to be added to the figure,\n        make sure you give them unique labels.\n\n        In rare circumstances, `.add_subplot` may be called with a single\n        argument, a subplot axes instance already created in the\n        present figure but not in the figure's list of axes.\n\n        See Also\n        --------\n        .Figure.add_axes\n        .pyplot.subplot\n        .pyplot.axes\n        .Figure.subplots\n        .pyplot.subplots\n\n        Examples\n        --------\n        ::\n\n            fig = plt.figure()\n            fig.add_subplot(221)\n\n            # equivalent but more general\n            ax1 = fig.add_subplot(2, 2, 1)\n\n            # add a subplot with no frame\n            ax2 = fig.add_subplot(222, frameon=False)\n\n            # add a polar subplot\n            fig.add_subplot(223, projection='polar')\n\n            # add a red subplot that share the x-axis with ax1\n            fig.add_subplot(224, sharex=ax1, facecolor='red')\n\n            #delete x2 from the figure\n            fig.delaxes(ax2)\n\n            #add x2 to the figure again\n            fig.add_subplot(ax2)\n        \"\"\"\n        if not len(args):\n            args = (1, 1, 1)\n\n        if len(args) == 1 and isinstance(args[0], Integral):\n            if not 100 <= args[0] <= 999:\n                raise ValueError(\"Integer subplot specification must be a \"\n                                 \"three-digit number, not {}\".format(args[0]))\n            args = tuple(map(int, str(args[0])))\n\n        if isinstance(args[0], SubplotBase):\n\n            a = args[0]\n            if a.get_figure() is not self:\n                raise ValueError(\n                    \"The Subplot must have been created in the present figure\")\n            # make a key for the subplot (which includes the axes object id\n            # in the hash)\n            key = self._make_key(*args, **kwargs)\n        else:\n            projection_class, kwargs, key = \\\n                self._process_projection_requirements(*args, **kwargs)\n\n            # try to find the axes with this key in the stack\n            ax = self._axstack.get(key)\n\n            if ax is not None:\n                if isinstance(ax, projection_class):\n                    # the axes already existed, so set it as active & return\n                    self.sca(ax)\n                    return ax\n                else:\n                    # Undocumented convenience behavior:\n                    # subplot(111); subplot(111, projection='polar')\n                    # will replace the first with the second.\n                    # Without this, add_subplot would be simpler and\n                    # more similar to add_axes.\n                    self._axstack.remove(ax)\n\n            a = subplot_class_factory(projection_class)(self, *args, **kwargs)\n\n        return self._add_axes_internal(key, a)"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "add_subplot",
          "class_name": "Figure",
          "code": "def add_subplot(self, *args, **kwargs):\n        \"\"\"\n        Add an `~.axes.Axes` to the figure as part of a subplot arrangement.\n\n        Call signatures::\n\n           add_subplot(nrows, ncols, index, **kwargs)\n           add_subplot(pos, **kwargs)\n           add_subplot(ax)\n           add_subplot()\n\n        Parameters\n        ----------\n        *args\n            Either a 3-digit integer or three separate integers\n            describing the position of the subplot. If the three\n            integers are *nrows*, *ncols*, and *index* in order, the\n            subplot will take the *index* position on a grid with *nrows*\n            rows and *ncols* columns. *index* starts at 1 in the upper left\n            corner and increases to the right.\n\n            *pos* is a three digit integer, where the first digit is the\n            number of rows, the second the number of columns, and the third\n            the index of the subplot. i.e. fig.add_subplot(235) is the same as\n            fig.add_subplot(2, 3, 5). Note that all integers must be less than\n            10 for this form to work.\n\n            If no positional arguments are passed, defaults to (1, 1, 1).\n\n        projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n            The projection type of the subplot (`~.axes.Axes`). *str* is the\n            name of a custom projection, see `~matplotlib.projections`. The\n            default None results in a 'rectilinear' projection.\n\n        polar : boolean, optional\n            If True, equivalent to projection='polar'.\n\n        sharex, sharey : `~.axes.Axes`, optional\n            Share the x or y `~matplotlib.axis` with sharex and/or sharey.\n            The axis will have the same limits, ticks, and scale as the axis\n            of the shared axes.\n\n        label : str\n            A label for the returned axes.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            This method also takes the keyword arguments for\n            the returned axes base class. The keyword arguments for the\n            rectilinear base class `~.axes.Axes` can be found in\n            the following table but there might also be other keyword\n            arguments if another projection is used.\n            %(Axes)s\n\n        Returns\n        -------\n        axes : an `.axes.SubplotBase` subclass of `~.axes.Axes` (or a \\\n               subclass of `~.axes.Axes`)\n\n            The axes of the subplot. The returned axes base class depends on\n            the projection used. It is `~.axes.Axes` if rectilinear projection\n            are used and `.projections.polar.PolarAxes` if polar projection\n            are used. The returned axes is then a subplot subclass of the\n            base class.\n\n        Notes\n        -----\n        If the figure already has a subplot with key (*args*,\n        *kwargs*) then it will simply make that subplot current and\n        return it.  This behavior is deprecated. Meanwhile, if you do\n        not want this behavior (i.e., you want to force the creation of a\n        new subplot), you must use a unique set of args and kwargs.  The axes\n        *label* attribute has been exposed for this purpose: if you want\n        two subplots that are otherwise identical to be added to the figure,\n        make sure you give them unique labels.\n\n        In rare circumstances, `.add_subplot` may be called with a single\n        argument, a subplot axes instance already created in the\n        present figure but not in the figure's list of axes.\n\n        See Also\n        --------\n        .Figure.add_axes\n        .pyplot.subplot\n        .pyplot.axes\n        .Figure.subplots\n        .pyplot.subplots\n\n        Examples\n        --------\n        ::\n\n            fig = plt.figure()\n            fig.add_subplot(221)\n\n            # equivalent but more general\n            ax1 = fig.add_subplot(2, 2, 1)\n\n            # add a subplot with no frame\n            ax2 = fig.add_subplot(222, frameon=False)\n\n            # add a polar subplot\n            fig.add_subplot(223, projection='polar')\n\n            # add a red subplot that share the x-axis with ax1\n            fig.add_subplot(224, sharex=ax1, facecolor='red')\n\n            #delete x2 from the figure\n            fig.delaxes(ax2)\n\n            #add x2 to the figure again\n            fig.add_subplot(ax2)\n        \"\"\"\n        if not len(args):\n            args = (1, 1, 1)\n\n        if len(args) == 1 and isinstance(args[0], Integral):\n            if not 100 <= args[0] <= 999:\n                raise ValueError(\"Integer subplot specification must be a \"\n                                 \"three-digit number, not {}\".format(args[0]))\n            args = tuple(map(int, str(args[0])))\n\n        if isinstance(args[0], SubplotBase):\n\n            a = args[0]\n            if a.get_figure() is not self:\n                raise ValueError(\n                    \"The Subplot must have been created in the present figure\")\n            # make a key for the subplot (which includes the axes object id\n            # in the hash)\n            key = self._make_key(*args, **kwargs)\n        else:\n            projection_class, kwargs, key = \\\n                self._process_projection_requirements(*args, **kwargs)\n\n            # try to find the axes with this key in the stack\n            ax = self._axstack.get(key)\n\n            if ax is not None:\n                if isinstance(ax, projection_class):\n                    # the axes already existed, so set it as active & return\n                    self.sca(ax)\n                    return ax\n                else:\n                    # Undocumented convenience behavior:\n                    # subplot(111); subplot(111, projection='polar')\n                    # will replace the first with the second.\n                    # Without this, add_subplot would be simpler and\n                    # more similar to add_axes.\n                    self._axstack.remove(ax)\n\n            a = subplot_class_factory(projection_class)(self, *args, **kwargs)\n\n        return self._add_axes_internal(key, a)"
        },
        {
          "file": "lib/matplotlib/pyplot.py",
          "type": "function",
          "name": "subplot",
          "class_name": null,
          "code": "def subplot(*args, **kwargs):\n    \"\"\"\n    Add a subplot to the current figure.\n\n    Wrapper of `.Figure.add_subplot` with a difference in behavior\n    explained in the notes section.\n\n    Call signatures::\n\n       subplot(nrows, ncols, index, **kwargs)\n       subplot(pos, **kwargs)\n       subplot(ax)\n\n    Parameters\n    ----------\n    *args\n        Either a 3-digit integer or three separate integers\n        describing the position of the subplot. If the three\n        integers are *nrows*, *ncols*, and *index* in order, the\n        subplot will take the *index* position on a grid with *nrows*\n        rows and *ncols* columns. *index* starts at 1 in the upper left\n        corner and increases to the right.\n\n        *pos* is a three digit integer, where the first digit is the\n        number of rows, the second the number of columns, and the third\n        the index of the subplot. i.e. fig.add_subplot(235) is the same as\n        fig.add_subplot(2, 3, 5). Note that all integers must be less than\n        10 for this form to work.\n\n    projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n        The projection type of the subplot (`~.axes.Axes`). *str* is the name\n        of a costum projection, see `~matplotlib.projections`. The default\n        None results in a 'rectilinear' projection.\n\n    polar : boolean, optional\n        If True, equivalent to projection='polar'.\n\n    sharex, sharey : `~.axes.Axes`, optional\n        Share the x or y `~matplotlib.axis` with sharex and/or sharey. The\n        axis will have the same limits, ticks, and scale as the axis of the\n        shared axes.\n\n    label : str\n        A label for the returned axes.\n\n    Other Parameters\n    ----------------\n    **kwargs\n        This method also takes the keyword arguments for\n        the returned axes base class. The keyword arguments for the\n        rectilinear base class `~.axes.Axes` can be found in\n        the following table but there might also be other keyword\n        arguments if another projection is used.\n        %(Axes)s\n\n    Returns\n    -------\n    axes : an `.axes.SubplotBase` subclass of `~.axes.Axes` (or a subclass \\\n    of `~.axes.Axes`)\n\n        The axes of the subplot. The returned axes base class depends on\n        the projection used. It is `~.axes.Axes` if rectilinear projection\n        are used and `.projections.polar.PolarAxes` if polar projection\n        are used. The returned axes is then a subplot subclass of the\n        base class.\n\n    Notes\n    -----\n    Creating a subplot will delete any pre-existing subplot that overlaps\n    with it beyond sharing a boundary::\n\n        import matplotlib.pyplot as plt\n        # plot a line, implicitly creating a subplot(111)\n        plt.plot([1,2,3])\n        # now create a subplot which represents the top plot of a grid\n        # with 2 rows and 1 column. Since this subplot will overlap the\n        # first, the plot (and its axes) previously created, will be removed\n        plt.subplot(211)\n\n    If you do not want this behavior, use the `.Figure.add_subplot` method\n    or the `.pyplot.axes` function instead.\n\n    If the figure already has a subplot with key (*args*,\n    *kwargs*) then it will simply make that subplot current and\n    return it.  This behavior is deprecated. Meanwhile, if you do\n    not want this behavior (i.e., you want to force the creation of a\n    new subplot), you must use a unique set of args and kwargs.  The axes\n    *label* attribute has been exposed for this purpose: if you want\n    two subplots that are otherwise identical to be added to the figure,\n    make sure you give them unique labels.\n\n    In rare circumstances, `.add_subplot` may be called with a single\n    argument, a subplot axes instance already created in the\n    present figure but not in the figure's list of axes.\n\n    See Also\n    --------\n    .Figure.add_subplot\n    .pyplot.subplots\n    .pyplot.axes\n    .Figure.subplots\n\n    Examples\n    --------\n    ::\n\n        plt.subplot(221)\n\n        # equivalent but more general\n        ax1=plt.subplot(2, 2, 1)\n\n        # add a subplot with no frame\n        ax2=plt.subplot(222, frameon=False)\n\n        # add a polar subplot\n        plt.subplot(223, projection='polar')\n\n        # add a red subplot that shares the x-axis with ax1\n        plt.subplot(224, sharex=ax1, facecolor='red')\n\n        # delete ax2 from the figure\n        plt.delaxes(ax2)\n\n        # add ax2 to the figure again\n        plt.subplot(ax2)\n        \"\"\"\n\n    # if subplot called without arguments, create subplot(1,1,1)\n    if len(args) == 0:\n        args = (1, 1, 1)\n\n    # This check was added because it is very easy to type\n    # subplot(1, 2, False) when subplots(1, 2, False) was intended\n    # (sharex=False, that is). In most cases, no error will\n    # ever occur, but mysterious behavior can result because what was\n    # intended to be the sharex argument is instead treated as a\n    # subplot index for subplot()\n    if len(args) >= 3 and isinstance(args[2], bool):\n        cbook._warn_external(\"The subplot index argument to subplot() appears \"\n                             \"to be a boolean. Did you intend to use \"\n                             \"subplots()?\")\n\n    fig = gcf()\n    a = fig.add_subplot(*args, **kwargs)\n    bbox = a.bbox\n    byebye = []\n    for other in fig.axes:\n        if other == a:\n            continue\n        if bbox.fully_overlaps(other.bbox):\n            byebye.append(other)\n    for ax in byebye:\n        delaxes(ax)\n\n    return a"
        }
      ]
    },
    {
      "pr_number": 11601,
      "pr_title": "FIX: subplots don't mutate kwargs passed by user.",
      "pr_body": "## PR Summary\r\n\r\nCloses #11515\r\n\r\nDon't mess with the user-supplied dictionaries in `subplots`...\r\n\r\nDunno if this really merits a test..\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\n\r\nsubplot_kw = {'sharex': 'all'}\r\nprint(subplot_kw)\r\nfig, ax = plt.subplots(2, 1, subplot_kw=subplot_kw)\r\nprint(subplot_kw)\r\n```\r\n\r\n### New:\r\n\r\n```\r\n{'sharex': 'all'}\r\n{'sharex': 'all'}\r\n```\r\n\r\n### Old:\r\n\r\n```\r\n{'sharex': 'all'}\r\n{'sharex': None, 'sharey': None}\r\n```\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is PEP 8 compliant\r\n- [ ] New features are documented, with examples if plot related\r\n- [ ] Documentation is sphinx and numpydoc compliant\r\n- [ ] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [ ] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way\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-->",
      "issue_id": 11515,
      "issue_title": "using 'sharex' once in 'subplots' function can affect subsequent calles to 'subplots'",
      "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\nUsing 'sharex' keyword argument once in the subplot function causes all subsequent axes returned by subplots to have a shared x-axis when the subplot_kw argument is also included.\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug, also minimizing the number of dependencies required-->\r\n\r\n```python\r\nimport matplotlib\r\nfrom pylab import *\r\n\r\n\r\ndefaults_subplots = dict()\r\n\r\nx1, y1 = range(30), randn(30)\r\nx2, y2 = range(30, 60), randn(30)\r\n\r\nclose('all')\r\n\r\n## Case 1 : single subplot (can be multiple as well)\r\n##  this behaves correctly\r\nf2, ax2 =  subplots(1,1, sharex=True, subplot_kw=defaults_subplots)\r\ntitle('figure 1')\r\nax2.plot(x2, -y2)\r\n\r\n## Case 2 : multiple subplots (single subplot causes bug to not appear)\r\nf1, (ax1, _) = subplots(2,1, sharex=True, subplot_kw=defaults_subplots)\r\ntitle('figure 2')\r\nax1.plot(x1, y1)\r\n\r\n## Case 3 : same code as case 1, however axis is shared with subplots in case 2\r\nf2, ax3 =  subplots(1,1, subplot_kw=defaults_subplots)\r\ntitle('figure 3')\r\nax3.plot(x2, -y2)\r\n\r\nshow()\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\nFigure 1 has an independent x axis, while subplots in figures 2 and 3 share an x axis. \r\n\r\n**Expected outcome**\r\nThe x axis should be independent for each figure.  The subplots in figure 2 should have a shared axis.\r\n \r\n**Fix**\r\nThe bug is the result of modifying the `subplot_kw` dictionary, following patch resolves the issue:\r\n\r\n```\r\n@@ -1171,6 +1171,9 @@\r\n                          (sharey, share_values))\r\n     if subplot_kw is None:\r\n         subplot_kw = {}\r\n+    else:\r\n+        subplot_kw = subplot_kw.copy()\r\n+\r\n     if gridspec_kw is None:\r\n         gridspec_kw = {}\r\n ```\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: Ubuntu 16.04\r\n  * Matplotlib version: 1.5.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: 3.5.2\r\nMatplotlib installed through ubuntu package manager.\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
      "issue_closed_at": "2018-07-09T19:17:19Z",
      "base_commit": "3d19f94284e15d4d1c4c1d736cf646d18856960a",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "subplots",
          "class_name": "Figure",
          "code": "def subplots(self, nrows=1, ncols=1, sharex=False, sharey=False,\n                 squeeze=True, subplot_kw=None, gridspec_kw=None):\n        \"\"\"\n        Add a set of subplots to this figure.\n\n        This utility wrapper makes it convenient to create common layouts of\n        subplots in a single call.\n\n        Parameters\n        ----------\n        nrows, ncols : int, optional, default: 1\n            Number of rows/columns of the subplot grid.\n\n        sharex, sharey : bool or {'none', 'all', 'row', 'col'}, default: False\n            Controls sharing of properties among x (`sharex`) or y (`sharey`)\n            axes:\n\n                - True or 'all': x- or y-axis will be shared among all\n                  subplots.\n                - False or 'none': each subplot x- or y-axis will be\n                  independent.\n                - 'row': each subplot row will share an x- or y-axis.\n                - 'col': each subplot column will share an x- or y-axis.\n\n            When subplots have a shared x-axis along a column, only the x tick\n            labels of the bottom subplot are created. Similarly, when subplots\n            have a shared y-axis along a row, only the y tick labels of the\n            first column subplot are created. To later turn other subplots'\n            ticklabels on, use `~matplotlib.axes.Axes.tick_params`.\n\n        squeeze : bool, optional, default: True\n            - If True, extra dimensions are squeezed out from the returned\n              array of Axes:\n\n                - if only one subplot is constructed (nrows=ncols=1), the\n                  resulting single Axes object is returned as a scalar.\n                - for Nx1 or 1xM subplots, the returned object is a 1D numpy\n                  object array of Axes objects.\n                - for NxM, subplots with N>1 and M>1 are returned\n                  as a 2D array.\n\n            - If False, no squeezing at all is done: the returned Axes object\n              is always a 2D array containing Axes instances, even if it ends\n              up being 1x1.\n\n        subplot_kw : dict, optional\n            Dict with keywords passed to the\n            :meth:`~matplotlib.figure.Figure.add_subplot` call used to create\n            each subplot.\n\n        gridspec_kw : dict, optional\n            Dict with keywords passed to the\n            `~matplotlib.gridspec.GridSpec` constructor used to create\n            the grid the subplots are placed on.\n\n        Returns\n        -------\n        ax : `~.axes.Axes` object or array of Axes objects.\n            *ax* can be either a single `~matplotlib.axes.Axes` object or\n            an array of Axes objects if more than one subplot was created. The\n            dimensions of the resulting array can be controlled with the\n            squeeze keyword, see above.\n\n        Examples\n        --------\n        ::\n\n            # First create some toy data:\n            x = np.linspace(0, 2*np.pi, 400)\n            y = np.sin(x**2)\n\n            # Create a figure\n            plt.figure(1, clear=True)\n\n            # Creates a subplot\n            ax = fig.subplots()\n            ax.plot(x, y)\n            ax.set_title('Simple plot')\n\n            # Creates two subplots and unpacks the output array immediately\n            ax1, ax2 = fig.subplots(1, 2, sharey=True)\n            ax1.plot(x, y)\n            ax1.set_title('Sharing Y axis')\n            ax2.scatter(x, y)\n\n            # Creates four polar axes, and accesses them through the\n            # returned array\n            axes = fig.subplots(2, 2, subplot_kw=dict(polar=True))\n            axes[0, 0].plot(x, y)\n            axes[1, 1].scatter(x, y)\n\n            # Share a X axis with each column of subplots\n            fig.subplots(2, 2, sharex='col')\n\n            # Share a Y axis with each row of subplots\n            fig.subplots(2, 2, sharey='row')\n\n            # Share both X and Y axes with all subplots\n            fig.subplots(2, 2, sharex='all', sharey='all')\n\n            # Note that this is the same as\n            fig.subplots(2, 2, sharex=True, sharey=True)\n\n            See Also\n            --------\n            .pyplot.subplots\n            .Figure.add_subplot\n            .pyplot.subplot\n            \"\"\"\n\n        if isinstance(sharex, bool):\n            sharex = \"all\" if sharex else \"none\"\n        if isinstance(sharey, bool):\n            sharey = \"all\" if sharey else \"none\"\n        share_values = [\"all\", \"row\", \"col\", \"none\"]\n        if sharex not in share_values:\n            # This check was added because it is very easy to type\n            # `subplots(1, 2, 1)` when `subplot(1, 2, 1)` was intended.\n            # In most cases, no error will ever occur, but mysterious behavior\n            # will result because what was intended to be the subplot index is\n            # instead treated as a bool for sharex.\n            if isinstance(sharex, Integral):\n                warnings.warn(\n                    \"sharex argument to subplots() was an integer. \"\n                    \"Did you intend to use subplot() (without 's')?\")\n\n            raise ValueError(\"sharex [%s] must be one of %s\" %\n                             (sharex, share_values))\n        if sharey not in share_values:\n            raise ValueError(\"sharey [%s] must be one of %s\" %\n                             (sharey, share_values))\n        if subplot_kw is None:\n            subplot_kw = {}\n        if gridspec_kw is None:\n            gridspec_kw = {}\n\n        if self.get_constrained_layout():\n            gs = GridSpec(nrows, ncols, figure=self, **gridspec_kw)\n        else:\n            # this should turn constrained_layout off if we don't want it\n            gs = GridSpec(nrows, ncols, figure=None, **gridspec_kw)\n\n        # Create array to hold all axes.\n        axarr = np.empty((nrows, ncols), dtype=object)\n        for row in range(nrows):\n            for col in range(ncols):\n                shared_with = {\"none\": None, \"all\": axarr[0, 0],\n                               \"row\": axarr[row, 0], \"col\": axarr[0, col]}\n                subplot_kw[\"sharex\"] = shared_with[sharex]\n                subplot_kw[\"sharey\"] = shared_with[sharey]\n                axarr[row, col] = self.add_subplot(gs[row, col], **subplot_kw)\n\n        # turn off redundant tick labeling\n        if sharex in [\"col\", \"all\"]:\n            # turn off all but the bottom row\n            for ax in axarr[:-1, :].flat:\n                ax.xaxis.set_tick_params(which='both',\n                                         labelbottom=False, labeltop=False)\n                ax.xaxis.offsetText.set_visible(False)\n        if sharey in [\"row\", \"all\"]:\n            # turn off all but the first column\n            for ax in axarr[:, 1:].flat:\n                ax.yaxis.set_tick_params(which='both',\n                                         labelleft=False, labelright=False)\n                ax.yaxis.offsetText.set_visible(False)\n\n        if squeeze:\n            # Discarding unneeded dimensions that equal 1.  If we only have one\n            # subplot, just return it instead of a 1-element array.\n            return axarr.item() if axarr.size == 1 else axarr.squeeze()\n        else:\n            # Returned axis array will be always 2-d, even if nrows=ncols=1.\n            return axarr"
        }
      ]
    },
    {
      "pr_number": 14997,
      "pr_title": "Correctly set formatters and locators on removed shared axis",
      "pr_body": "Fixes #14911 \r\n\r\nThe issue was that a shared axis might never have it's `Formatter/Locator` set, which is fine when it is shared, but when the axis storing the `Formatter/Locator` it relies on is removed, the previous shared axis needs to have the `Formatter/Locator`s set.\r\n\r\nAlso fixed typo `_reset_axis_form` > `_reset_axis_from`.\r\n\r\nI've tried a few things, but cannot come up with a more elegant solution than this; if anyone has any ideas suggestions welcome.",
      "issue_id": 14911,
      "issue_title": "Removing a shared axes via `ax.remove()` leads to an error.",
      "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\nremoving a shared axes via `ax.remove()` leads to an error.\r\n\r\nThis bisects to \"Fix locator/formatter setting when removing shared Axes\" #13983 (in current master, targeted for 3.2) which initially headed out to make the situation of removing a shared axes better, fixing https://github.com/matplotlib/matplotlib/issues/12853, but apparently the test doesn't capture the below simple case.\r\n\r\nHence marking as release critical for 3.2, in order not to release this bug.\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\nimport matplotlib.pyplot as plt\r\n\r\nfig, axs = plt.subplots(2, sharex=True)\r\n\r\naxs[0].remove()\r\n\r\nplt.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\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\backends\\backend_qt5.py\", line 488, in _draw_idle\r\n    self.draw()\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\backends\\backend_agg.py\", line 396, in draw\r\n    self.figure.draw(self.renderer)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\artist.py\", line 38, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\figure.py\", line 1722, in draw\r\n    renderer, self, artists, self.suppressComposite)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\image.py\", line 136, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\artist.py\", line 38, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\axes\\_base.py\", line 2622, in draw\r\n    mimage._draw_list_compositing_images(renderer, self, artists)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\image.py\", line 136, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\artist.py\", line 38, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\axis.py\", line 1227, in draw\r\n    ticks_to_draw = self._update_ticks()\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\axis.py\", line 1103, in _update_ticks\r\n    major_locs = self.get_majorticklocs()\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\axis.py\", line 1348, in get_majorticklocs\r\n    return self.major.locator()\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\ticker.py\", line 2032, in __call__\r\n    return self.tick_values(vmin, vmax)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\ticker.py\", line 2040, in tick_values\r\n    locs = self._raw_ticks(vmin, vmax)\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\ticker.py\", line 1979, in _raw_ticks\r\n    nbins = np.clip(self.axis.get_tick_space(),\r\n  File \"d:\\***\\matplotlib\\lib\\matplotlib\\axis.py\", line 2193, in get_tick_space\r\n    length = ((ends[1][0] - ends[0][0]) / self.axes.figure.dpi) * 72\r\nAttributeError: 'NoneType' object has no attribute 'dpi'\r\n```\r\n\r\n**Expected outcome**\r\n\r\nA plot with the first axes removed, as would be achieved with the same code running in 3.1\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: Win 8.1\r\n  * Matplotlib version: current master\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.6\r\n\r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
      "issue_closed_at": "2019-08-19T13:22:50Z",
      "base_commit": "7de91c8ba92f86f46a3090a2922096648c3df54b",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "subplots",
          "class_name": "Figure",
          "code": "def subplots(self, nrows=1, ncols=1, sharex=False, sharey=False,\n                 squeeze=True, subplot_kw=None, gridspec_kw=None):\n        \"\"\"\n        Add a set of subplots to this figure.\n\n        This utility wrapper makes it convenient to create common layouts of\n        subplots in a single call.\n\n        Parameters\n        ----------\n        nrows, ncols : int, optional, default: 1\n            Number of rows/columns of the subplot grid.\n\n        sharex, sharey : bool or {'none', 'all', 'row', 'col'}, default: False\n            Controls sharing of properties among x (`sharex`) or y (`sharey`)\n            axes:\n\n            - True or 'all': x- or y-axis will be shared among all subplots.\n            - False or 'none': each subplot x- or y-axis will be independent.\n            - 'row': each subplot row will share an x- or y-axis.\n            - 'col': each subplot column will share an x- or y-axis.\n\n            When subplots have a shared x-axis along a column, only the x tick\n            labels of the bottom subplot are created. Similarly, when subplots\n            have a shared y-axis along a row, only the y tick labels of the\n            first column subplot are created. To later turn other subplots'\n            ticklabels on, use `~matplotlib.axes.Axes.tick_params`.\n\n        squeeze : bool, optional, default: True\n            - If True, extra dimensions are squeezed out from the returned\n              array of Axes:\n\n              - if only one subplot is constructed (nrows=ncols=1), the\n                resulting single Axes object is returned as a scalar.\n              - for Nx1 or 1xM subplots, the returned object is a 1D numpy\n                object array of Axes objects.\n              - for NxM, subplots with N>1 and M>1 are returned as a 2D array.\n\n            - If False, no squeezing at all is done: the returned Axes object\n              is always a 2D array containing Axes instances, even if it ends\n              up being 1x1.\n\n        subplot_kw : dict, optional\n            Dict with keywords passed to the\n            :meth:`~matplotlib.figure.Figure.add_subplot` call used to create\n            each subplot.\n\n        gridspec_kw : dict, optional\n            Dict with keywords passed to the\n            `~matplotlib.gridspec.GridSpec` constructor used to create\n            the grid the subplots are placed on.\n\n        Returns\n        -------\n        ax : `~.axes.Axes` object or array of Axes objects.\n            *ax* can be either a single `~matplotlib.axes.Axes` object or\n            an array of Axes objects if more than one subplot was created. The\n            dimensions of the resulting array can be controlled with the\n            squeeze keyword, see above.\n\n        Examples\n        --------\n        ::\n\n            # First create some toy data:\n            x = np.linspace(0, 2*np.pi, 400)\n            y = np.sin(x**2)\n\n            # Create a figure\n            plt.figure()\n\n            # Create a subplot\n            ax = fig.subplots()\n            ax.plot(x, y)\n            ax.set_title('Simple plot')\n\n            # Create two subplots and unpack the output array immediately\n            ax1, ax2 = fig.subplots(1, 2, sharey=True)\n            ax1.plot(x, y)\n            ax1.set_title('Sharing Y axis')\n            ax2.scatter(x, y)\n\n            # Create four polar axes and access them through the returned array\n            axes = fig.subplots(2, 2, subplot_kw=dict(polar=True))\n            axes[0, 0].plot(x, y)\n            axes[1, 1].scatter(x, y)\n\n            # Share a X axis with each column of subplots\n            fig.subplots(2, 2, sharex='col')\n\n            # Share a Y axis with each row of subplots\n            fig.subplots(2, 2, sharey='row')\n\n            # Share both X and Y axes with all subplots\n            fig.subplots(2, 2, sharex='all', sharey='all')\n\n            # Note that this is the same as\n            fig.subplots(2, 2, sharex=True, sharey=True)\n\n        See Also\n        --------\n        .pyplot.subplots\n        .Figure.add_subplot\n        .pyplot.subplot\n        \"\"\"\n\n        if isinstance(sharex, bool):\n            sharex = \"all\" if sharex else \"none\"\n        if isinstance(sharey, bool):\n            sharey = \"all\" if sharey else \"none\"\n        # This check was added because it is very easy to type\n        # `subplots(1, 2, 1)` when `subplot(1, 2, 1)` was intended.\n        # In most cases, no error will ever occur, but mysterious behavior\n        # will result because what was intended to be the subplot index is\n        # instead treated as a bool for sharex.\n        if isinstance(sharex, Integral):\n            cbook._warn_external(\n                \"sharex argument to subplots() was an integer.  Did you \"\n                \"intend to use subplot() (without 's')?\")\n        cbook._check_in_list([\"all\", \"row\", \"col\", \"none\"],\n                             sharex=sharex, sharey=sharey)\n        if subplot_kw is None:\n            subplot_kw = {}\n        if gridspec_kw is None:\n            gridspec_kw = {}\n        # don't mutate kwargs passed by user...\n        subplot_kw = subplot_kw.copy()\n        gridspec_kw = gridspec_kw.copy()\n\n        if self.get_constrained_layout():\n            gs = GridSpec(nrows, ncols, figure=self, **gridspec_kw)\n        else:\n            # this should turn constrained_layout off if we don't want it\n            gs = GridSpec(nrows, ncols, figure=None, **gridspec_kw)\n        self._gridspecs.append(gs)\n\n        # Create array to hold all axes.\n        axarr = np.empty((nrows, ncols), dtype=object)\n        for row in range(nrows):\n            for col in range(ncols):\n                shared_with = {\"none\": None, \"all\": axarr[0, 0],\n                               \"row\": axarr[row, 0], \"col\": axarr[0, col]}\n                subplot_kw[\"sharex\"] = shared_with[sharex]\n                subplot_kw[\"sharey\"] = shared_with[sharey]\n                axarr[row, col] = self.add_subplot(gs[row, col], **subplot_kw)\n\n        # turn off redundant tick labeling\n        if sharex in [\"col\", \"all\"]:\n            # turn off all but the bottom row\n            for ax in axarr[:-1, :].flat:\n                ax.xaxis.set_tick_params(which='both',\n                                         labelbottom=False, labeltop=False)\n                ax.xaxis.offsetText.set_visible(False)\n        if sharey in [\"row\", \"all\"]:\n            # turn off all but the first column\n            for ax in axarr[:, 1:].flat:\n                ax.yaxis.set_tick_params(which='both',\n                                         labelleft=False, labelright=False)\n                ax.yaxis.offsetText.set_visible(False)\n\n        if squeeze:\n            # Discarding unneeded dimensions that equal 1.  If we only have one\n            # subplot, just return it instead of a 1-element array.\n            return axarr.item() if axarr.size == 1 else axarr.squeeze()\n        else:\n            # Returned axis array will be always 2-d, even if nrows=ncols=1.\n            return axarr"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_break_share_link",
          "class_name": "Figure",
          "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": 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()"
        }
      ]
    }
  ]
}