{
  "instance_id": "matplotlib__matplotlib-23562",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-08-05T13:44:06Z",
  "problem_statement": "'Poly3DCollection' object has no attribute '_facecolors2d'\nThe following minimal example demonstrates the issue:\n\n```\nimport numpy as np\nimport matplotlib.tri as mtri\nimport matplotlib.pyplot as plt\nfrom mpl_toolkits.mplot3d import Axes3D\n\ny,x = np.ogrid[1:10:100j, 1:10:100j]\nz2 = np.cos(x)**3 - np.sin(y)**2\nfig = plt.figure()\nax = fig.add_subplot(111, projection='3d')\nr = ax.plot_surface(x,y,z2, cmap='hot')\nr.get_facecolors()\n```\n\nIt fails on the last line with the following traceback:\n\n```\nAttributeError                            Traceback (most recent call last)\n<ipython-input-13-de0f41d662cd> in <module>()\n----> 1 r.get_facecolors()\n\n/home/oliver/.virtualenvs/mpl/local/lib/python2.7/site-packages/mpl_toolkits/mplot3d/art3d.pyc in get_facecolors(self)\n    634\n    635     def get_facecolors(self):\n--> 636         return self._facecolors2d\n    637     get_facecolor = get_facecolors\n    638\n\nAttributeError: 'Poly3DCollection' object has no attribute '_facecolors2d'\n```\n\nTested with mpl versions 1.3.1 and 1.4.2.\n\nSent here by Benjamin, from the mpl users mailing list (mail with the same title). Sorry for dumping this without more assistance, I'm not yet at a python level where I can help in debugging, I think (well, it seems daunting).\n\n",
  "patch": "diff --git a/lib/mpl_toolkits/mplot3d/art3d.py b/lib/mpl_toolkits/mplot3d/art3d.py\n--- a/lib/mpl_toolkits/mplot3d/art3d.py\n+++ b/lib/mpl_toolkits/mplot3d/art3d.py\n@@ -867,9 +867,19 @@ def set_alpha(self, alpha):\n         self.stale = True\n \n     def get_facecolor(self):\n+        # docstring inherited\n+        # self._facecolors2d is not initialized until do_3d_projection\n+        if not hasattr(self, '_facecolors2d'):\n+            self.axes.M = self.axes.get_proj()\n+            self.do_3d_projection()\n         return self._facecolors2d\n \n     def get_edgecolor(self):\n+        # docstring inherited\n+        # self._edgecolors2d is not initialized until do_3d_projection\n+        if not hasattr(self, '_edgecolors2d'):\n+            self.axes.M = self.axes.get_proj()\n+            self.do_3d_projection()\n         return self._edgecolors2d\n \n \n",
  "similar_bug_items": [
    {
      "pr_number": 10772,
      "pr_title": "API: check locator and formatter args when passed",
      "pr_body": "## PR Summary\r\n\r\nRight now, `ax.set_major_formatter` and `ax.set_major_locator` throw pretty deep error messages if the wrong type of data is passed to them.  This catches at entry. \r\n\r\nCloses #10769\r\n\r\n```python\r\nimport matplotlib\r\nimport matplotlib.pyplot as plt\r\n\r\nf, ax = plt.subplots(1, 1, figsize=(15, 10))\r\nplt.plot([10**4, 11*10**4], [100, 150])\r\nax.xaxis.set_major_formatter(matplotlib.ticker.LogLocator())\r\nplt.show()\r\n```\r\nNow returns:\r\n```\r\nTraceback (most recent call last):\r\n  File \"testEngTicker.py\", line 6, in <module>\r\n    ax.xaxis.set_major_formatter(matplotlib.ticker.LogLocator())\r\n  File \"/Users/jklymak/matplotlib/lib/matplotlib/axis.py\", line 1573, in set_major_formatter\r\n    \"formatter argument must be matplotlib.ticker.Formatter\")\r\nValueError: formatter argument must be matplotlib.ticker.Formatter\r\n```\r\ninstead of something down in the bowels of drawing the ticker...\r\n\r\n## PR Checklist\r\n\r\n- [x] Code is PEP 8 compliant\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": 10769,
      "issue_title": "DOC: set_major_locator could check that its getting a Locator (was EngFormatter broken?)",
      "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`matplotlib.ticker.EngFormatter` seems to be broken:\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n```\r\nf, ax = plt.subplots(1, 1, figsize=(15, 10))\r\nplt.plot([10**4, 11*10**4], [100, 150])\r\nax.xaxis.set_major_locator(matplotlib.ticker.EngFormatter())\r\n```\r\n```\r\nError in callback <function install_repl_displayhook.<locals>.post_execute at 0x7f60a9b120d0> (for post_execute):\r\n\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/pyplot.py in post_execute()\r\n    147             def post_execute():\r\n    148                 if matplotlib.is_interactive():\r\n--> 149                     draw_all()\r\n    150 \r\n    151             # IPython >= 2\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/_pylab_helpers.py in draw_all(cls, force)\r\n    134         for f_mgr in cls.get_all_fig_managers():\r\n    135             if force or f_mgr.canvas.figure.stale:\r\n--> 136                 f_mgr.canvas.draw_idle()\r\n    137 \r\n    138 atexit.register(Gcf.destroy_all)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backend_bases.py in draw_idle(self, *args, **kwargs)\r\n   2053         if not self._is_idle_drawing:\r\n   2054             with self._idle_draw_cntx():\r\n-> 2055                 self.draw(*args, **kwargs)\r\n   2056 \r\n   2057     def draw_cursor(self, event):\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in draw(self)\r\n    429             # if toolbar:\r\n    430             #     toolbar.set_cursor(cursors.WAIT)\r\n--> 431             self.figure.draw(self.renderer)\r\n    432             # A GUI class may be need to update a window using this draw, so\r\n    433             # don't forget to call the superclass.\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py in draw(self, renderer)\r\n   1473 \r\n   1474             mimage._draw_list_compositing_images(\r\n-> 1475                 renderer, self, artists, self.suppressComposite)\r\n   1476 \r\n   1477             renderer.close_group('figure')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py in draw(self, renderer, inframe)\r\n   2605             renderer.stop_rasterizing()\r\n   2606 \r\n-> 2607         mimage._draw_list_compositing_images(renderer, self, artists)\r\n   2608 \r\n   2609         renderer.close_group('axes')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in draw(self, renderer, *args, **kwargs)\r\n   1185         renderer.open_group(__name__)\r\n   1186 \r\n-> 1187         ticks_to_draw = self._update_ticks(renderer)\r\n   1188         ticklabelBoxes, ticklabelBoxes2 = self._get_tick_bboxes(ticks_to_draw,\r\n   1189                                                                 renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in _update_ticks(self, renderer)\r\n   1023 \r\n   1024         interval = self.get_view_interval()\r\n-> 1025         tick_tups = list(self.iter_ticks())  # iter_ticks calls the locator\r\n   1026         if self._smart_bounds and tick_tups:\r\n   1027             # handle inverted limits\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in iter_ticks(self)\r\n    966         Iterate through all of the major and minor ticks.\r\n    967         \"\"\"\r\n--> 968         majorLocs = self.major.locator()\r\n    969         majorTicks = self.get_major_ticks(len(majorLocs))\r\n    970         self.major.formatter.set_locs(majorLocs)\r\n\r\nTypeError: __call__() missing 1 required positional argument: 'x'\r\n\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/formatters.py in __call__(self, obj)\r\n    339                 pass\r\n    340             else:\r\n--> 341                 return printer(obj)\r\n    342             # Finally look for special method names\r\n    343             method = get_real_method(obj, self.print_method)\r\n\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/pylabtools.py in <lambda>(fig)\r\n    236 \r\n    237     if 'png' in formats:\r\n--> 238         png_formatter.for_type(Figure, lambda fig: print_figure(fig, 'png', **kwargs))\r\n    239     if 'retina' in formats or 'png2x' in formats:\r\n    240         png_formatter.for_type(Figure, lambda fig: retina_figure(fig, **kwargs))\r\n\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/pylabtools.py in print_figure(fig, fmt, bbox_inches, **kwargs)\r\n    120 \r\n    121     bytes_io = BytesIO()\r\n--> 122     fig.canvas.print_figure(bytes_io, **kw)\r\n    123     data = bytes_io.getvalue()\r\n    124     if fmt == 'svg':\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backend_bases.py in print_figure(self, filename, dpi, facecolor, edgecolor, orientation, format, **kwargs)\r\n   2208                     orientation=orientation,\r\n   2209                     dryrun=True,\r\n-> 2210                     **kwargs)\r\n   2211                 renderer = self.figure._cachedRenderer\r\n   2212                 bbox_inches = self.figure.get_tightbbox(renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in print_png(self, filename_or_obj, *args, **kwargs)\r\n    509 \r\n    510     def print_png(self, filename_or_obj, *args, **kwargs):\r\n--> 511         FigureCanvasAgg.draw(self)\r\n    512         renderer = self.get_renderer()\r\n    513         original_dpi = renderer.dpi\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in draw(self)\r\n    429             # if toolbar:\r\n    430             #     toolbar.set_cursor(cursors.WAIT)\r\n--> 431             self.figure.draw(self.renderer)\r\n    432             # A GUI class may be need to update a window using this draw, so\r\n    433             # don't forget to call the superclass.\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py in draw(self, renderer)\r\n   1473 \r\n   1474             mimage._draw_list_compositing_images(\r\n-> 1475                 renderer, self, artists, self.suppressComposite)\r\n   1476 \r\n   1477             renderer.close_group('figure')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py in draw(self, renderer, inframe)\r\n   2605             renderer.stop_rasterizing()\r\n   2606 \r\n-> 2607         mimage._draw_list_compositing_images(renderer, self, artists)\r\n   2608 \r\n   2609         renderer.close_group('axes')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in draw(self, renderer, *args, **kwargs)\r\n   1185         renderer.open_group(__name__)\r\n   1186 \r\n-> 1187         ticks_to_draw = self._update_ticks(renderer)\r\n   1188         ticklabelBoxes, ticklabelBoxes2 = self._get_tick_bboxes(ticks_to_draw,\r\n   1189                                                                 renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in _update_ticks(self, renderer)\r\n   1023 \r\n   1024         interval = self.get_view_interval()\r\n-> 1025         tick_tups = list(self.iter_ticks())  # iter_ticks calls the locator\r\n   1026         if self._smart_bounds and tick_tups:\r\n   1027             # handle inverted limits\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in iter_ticks(self)\r\n    966         Iterate through all of the major and minor ticks.\r\n    967         \"\"\"\r\n--> 968         majorLocs = self.major.locator()\r\n    969         majorTicks = self.get_major_ticks(len(majorLocs))\r\n    970         self.major.formatter.set_locs(majorLocs)\r\n\r\nTypeError: __call__() missing 1 required positional argument: 'x'\r\n\r\n<Figure size 1080x720 with 1 Axes>\r\n```\r\n<!--A minimum code snippet required to reproduce the bug, also minimizing the number of dependencies required-->\r\n\r\n\r\n<!--A description of the expected outcome from the code snippet-->\r\n<!--If this used to work in an earlier version of Matplotlib, please note the version it used to work on-->\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: 2.2.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): module://ipykernel.pylab.backend_inline\r\n  * Python version: 3.5.2 (IPython 6.2.1)\r\n  * Jupyter version (if applicable): Jupyter Lab beta 0.31.8\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": "2018-03-14T06:41:19Z",
      "base_commit": "8af1fac8baf9bbf06160d8a91575d3d6d8fdca04",
      "changes": [
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_major_formatter",
          "class_name": "Axis",
          "code": "def set_major_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the major ticker\n\n        ACCEPTS: A :class:`~matplotlib.ticker.Formatter` instance\n        \"\"\"\n        self.isDefault_majfmt = False\n        self.major.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_minor_formatter",
          "class_name": "Axis",
          "code": "def set_minor_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the minor ticker\n\n        ACCEPTS: A :class:`~matplotlib.ticker.Formatter` instance\n        \"\"\"\n        self.isDefault_minfmt = False\n        self.minor.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_major_locator",
          "class_name": "Axis",
          "code": "def set_major_locator(self, locator):\n        \"\"\"\n        Set the locator of the major ticker\n\n        ACCEPTS: a :class:`~matplotlib.ticker.Locator` instance\n        \"\"\"\n        self.isDefault_majloc = False\n        self.major.locator = locator\n        locator.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_minor_locator",
          "class_name": "Axis",
          "code": "def set_minor_locator(self, locator):\n        \"\"\"\n        Set the locator of the minor ticker\n\n        ACCEPTS: a :class:`~matplotlib.ticker.Locator` instance\n        \"\"\"\n        self.isDefault_minloc = False\n        self.minor.locator = locator\n        locator.set_axis(self)\n        self.stale = True"
        }
      ]
    },
    {
      "pr_number": 14719,
      "pr_title": "Make Qt navtoolbar more robust against removal of either pan or zoom.",
      "pr_body": "... by using a QButtonGroup to ensure that a button gets unchecked when\r\nthe other gets checked.\r\n\r\nCloses #12893.\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) 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-->\r\n",
      "issue_id": 12893,
      "issue_title": "[PyQt] NavigationToolbar2QT : Error when removing tools",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n\r\n**Code**\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\nclass FigureWidget(QtWidgets.QWidget):\r\n    \"\"\"Figure widget used by different views\r\n\r\n    Parameters\r\n    ----------\r\n    parent : Qt parent\r\n       Qt parent object\r\n    canvas : FigureCanvas\r\n       the figure canvas class\r\n    exclude_toolbar_items : tuple\r\n       elements to exclude from the toolbar\r\n    \"\"\"\r\n    def __init__(self, parent=None, canvas=None, exclude_toolbar_items=()):\r\n        super().__init__(parent)\r\n\r\n        self.canvas = canvas\r\n        self.layout = QtWidgets.QVBoxLayout()\r\n        self.layout.addWidget(self.canvas)\r\n\r\n        if isinstance(canvas, FigureCanvas):\r\n          class _SimpleNavigationToolbar(NavigationToolbar):\r\n            # only display the buttons we need\r\n            toolitems = [t for t in NavigationToolbar.toolitems if\r\n                        t[0] not in exclude_toolbar_items]\r\n          self.toolbar = _SimpleNavigationToolbar(self.canvas, self.canvas)\r\n\r\n          self.layout.addWidget(self.toolbar)\r\n      else:\r\n          self.toolbar = None\r\n\r\n      self.setLayout(self.layout)\r\n```\r\n\r\n**Error example**\r\n\r\nWhen **exclude_toolbar_items=('Pan',))**, we get the following result on our window :\r\n\r\n![toolbar_error](https://user-images.githubusercontent.com/39213808/49034667-e5ff8100-f1b2-11e8-80d2-25ad2ca06d02.png)\r\n\r\nBut if we click on the **Zoom** button (_encircled in red_), we get the error message on the log and written just below as if the **Pan** button stills exists.\r\n\r\n```\r\n2018-11-26 19:35:01.566 [ERROR]\t[utils.py:28]\t\tUncaught exception\r\nTraceback (most recent call last):\r\n  File \"C:\\Users\\Renaud Gautier\\Anaconda3\\envs\\some_env\\lib\\site-packages\\matplotlib\\backends\\backend_qt5.py\", line 803, in zoom\r\n    self._update_buttons_checked()\r\n  File \"C:\\Users\\Renaud Gautier\\Anaconda3\\envs\\some_env\\lib\\site-packages\\matplotlib\\backends\\backend_qt5.py\", line 794, in _update_buttons_checked\r\n    self._actions['pan'].setChecked(self._active == 'PAN')\r\nKeyError: 'pan'\r\n```\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating system: Windows 10\r\n  * Matplotlib version: 3.0.2\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.1\r\n  * Other libraries: PyQt 5 (PySide)\r\n\r\n\r\n",
      "issue_closed_at": "2019-07-17T18:39:21Z",
      "base_commit": "9f1c7306d23cd9eac858897b3c873c8314e300cf",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "_icon",
          "class_name": "ToolbarQt",
          "code": "def _icon(self, name):\n        pm = QtGui.QPixmap(name)\n        if hasattr(pm, 'setDevicePixelRatio'):\n            pm.setDevicePixelRatio(self.toolmanager.canvas._dpi_ratio)\n        return QtGui.QIcon(pm)"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "_init_toolbar",
          "class_name": "NavigationToolbar2QT",
          "code": "def _init_toolbar(self):\n        self.basedir = str(cbook._get_data_path('images'))\n\n        for text, tooltip_text, image_file, callback in self.toolitems:\n            if text is None:\n                self.addSeparator()\n            else:\n                a = self.addAction(self._icon(image_file + '.png'),\n                                   text, getattr(self, callback))\n                self._actions[callback] = a\n                if callback in ['zoom', 'pan']:\n                    a.setCheckable(True)\n                if tooltip_text is not None:\n                    a.setToolTip(tooltip_text)\n                if text == 'Subplots':\n                    a = self.addAction(self._icon(\"qt4_editor_options.png\"),\n                                       'Customize', self.edit_parameters)\n                    a.setToolTip('Edit axis, curve and image parameters')\n\n        # Add the x,y location widget at the right side of the toolbar\n        # The stretch factor is 1 which means any resizing of the toolbar\n        # will resize this label instead of the buttons.\n        if self.coordinates:\n            self.locLabel = QtWidgets.QLabel(\"\", self)\n            self.locLabel.setAlignment(\n                    QtCore.Qt.AlignRight | QtCore.Qt.AlignTop)\n            self.locLabel.setSizePolicy(\n                QtWidgets.QSizePolicy(QtWidgets.QSizePolicy.Expanding,\n                                      QtWidgets.QSizePolicy.Ignored))\n            labelAction = self.addWidget(self.locLabel)\n            labelAction.setVisible(True)\n\n        # Esthetic adjustments - we need to set these explicitly in PyQt5\n        # otherwise the layout looks different - but we don't want to set it if\n        # not using HiDPI icons otherwise they look worse than before.\n        if is_pyqt5() and self.canvas._dpi_ratio > 1:\n            self.setIconSize(QtCore.QSize(24, 24))\n            self.layout().setSpacing(12)"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "edit_parameters",
          "class_name": "NavigationToolbar2QT",
          "code": "def edit_parameters(self):\n        axes = self.canvas.figure.get_axes()\n        if not axes:\n            QtWidgets.QMessageBox.warning(\n                self.parent, \"Error\", \"There are no axes to edit.\")\n            return\n        elif len(axes) == 1:\n            ax, = axes\n        else:\n            titles = [\n                ax.get_label() or\n                ax.get_title() or\n                \" - \".join(filter(None, [ax.get_xlabel(), ax.get_ylabel()])) or\n                f\"<anonymous {type(ax).__name__}>\"\n                for ax in axes]\n            duplicate_titles = [\n                title for title in titles if titles.count(title) > 1]\n            for i, ax in enumerate(axes):\n                if titles[i] in duplicate_titles:\n                    titles[i] += f\" (id: {id(ax):#x})\"  # Deduplicate titles.\n            item, ok = QtWidgets.QInputDialog.getItem(\n                self.parent, 'Customize', 'Select axes:', titles, 0, False)\n            if not ok:\n                return\n            ax = axes[titles.index(item)]\n        figureoptions.figure_edit(ax, self)"
        }
      ]
    },
    {
      "pr_number": 3982,
      "pr_title": "pgf can not write to `BytesIO` [back port to 1.4.x]",
      "pr_body": "Discovered as part of #3981\n",
      "issue_id": 3982,
      "issue_title": "pgf can not write to `BytesIO` [back port to 1.4.x]",
      "issue_body": "Discovered as part of #3981\n",
      "issue_closed_at": "2015-01-09T07:47:13Z",
      "base_commit": "580d9905a0cce6a51edb6e6ae3b66c8488117fa7",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FigureManagerPgf",
          "code": "def __init__(self, *args):\n        FigureManagerBase.__init__(self, *args)"
        },
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "print_pgf",
          "class_name": "FigureCanvasPgf",
          "code": "def print_pgf(self, fname_or_fh, *args, **kwargs):\n        \"\"\"\n        Output pgf commands for drawing the figure so it can be included and\n        rendered in latex documents.\n        \"\"\"\n        if kwargs.get(\"dryrun\", False):\n            self._print_pgf_to_fh(None, *args, **kwargs)\n            return\n\n        # figure out where the pgf is to be written to\n        if is_string_like(fname_or_fh):\n            with codecs.open(fname_or_fh, \"w\", encoding=\"utf-8\") as fh:\n                self._print_pgf_to_fh(fh, *args, **kwargs)\n        elif is_writable_file_like(fname_or_fh):\n            if not os.path.exists(fname_or_fh.name):\n                warnings.warn(\"streamed pgf-code does not support raster \"\n                              \"graphics, consider using the pgf-to-pdf option\",\n                              UserWarning)\n            self._print_pgf_to_fh(fname_or_fh, *args, **kwargs)\n        else:\n            raise ValueError(\"filename must be a path\")"
        },
        {
          "file": "lib/matplotlib/backends/backend_webagg_core.py",
          "type": "function",
          "name": "get_javascript",
          "class_name": "FigureManagerWebAgg",
          "code": "def get_javascript(cls, stream=None):\n        if stream is None:\n            output = io.StringIO()\n        else:\n            output = stream\n\n        with io.open(os.path.join(\n                os.path.dirname(__file__),\n                \"web_backend\",\n                \"mpl.js\"), encoding='utf8') as fd:\n            output.write(fd.read())\n\n        toolitems = []\n        for name, tooltip, image, method in cls.ToolbarCls.toolitems:\n            if name is None:\n                toolitems.append(['', '', '', ''])\n            else:\n                toolitems.append([name, tooltip, image, method])\n        output.write(\"mpl.toolbar_items = {0};\\n\\n\".format(\n            json.dumps(toolitems)))\n\n        extensions = []\n        for filetype, ext in sorted(FigureCanvasWebAggCore.\n                                    get_supported_filetypes_grouped().\n                                    items()):\n            if not ext[0] == 'pgf':  # pgf does not support BytesIO\n                extensions.append(ext[0])\n        output.write(\"mpl.extensions = {0};\\n\\n\".format(\n            json.dumps(extensions)))\n\n        output.write(\"mpl.default_extension = {0};\".format(\n            json.dumps(FigureCanvasWebAggCore.get_default_filetype())))\n\n        if stream is None:\n            return output.getvalue()"
        }
      ]
    },
    {
      "pr_number": 17163,
      "pr_title": "Fix clipping of markers in PDF backend.",
      "pr_body": "The bbox only contains the points of the marker, but the line will extend outside by half the line width. This was handled before, except when the line is at an angle to the edge, as then the line outside is wider than half a line width. The maximum is at 45\u00b0, or \u221a2.\r\n\r\nThis fixes corners on 'v', '^', '<', '>', 'p', 'h', 'H', 'D', 'd', 'X'.\r\n\r\nFixes #9829.\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n- [x] Has Pytest style unit tests\r\n- [x] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) compliant\r\n- [n/a] New features are documented, with examples if plot related\r\n- [n/a] Documentation is sphinx and numpydoc compliant\r\n- [n/a] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [n/a] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way",
      "issue_id": 9829,
      "issue_title": "Vertices clipped for certain markers when plotting more than two points and saving as pdf",
      "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\nThe vertices on certain markers ('^', 'D', 'v') are clipped when using `plot()` to plot data with more than two points and saving the plot as a PDF.\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.pyplot as plt\r\nsize = 256, 16\r\nfig,ax = plt.subplots()\r\nnum = 3\r\n\r\nfig,ax = plt.subplots()\r\nax.plot([i+1 for i in range(num)],[1 for i in range(num)], '^', markersize=3, markeredgewidth=0.7, color='green', markeredgecolor='black')\r\nax.set_xlim(0, num +1)\r\nax.set_xticks(())\r\nax.set_yticks(())\r\nfig.savefig('test.pdf') # This doesn't work properly.\r\nfig.savefig('test.svg')\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![pdfmarker](https://user-images.githubusercontent.com/8723338/33104959-e7cacfd6-cef8-11e7-880a-0e8aa8643454.png)\r\n![svgmarker](https://user-images.githubusercontent.com/8723338/33104963-ea6f906e-cef8-11e7-8742-ce5b2421c337.png)\r\n\r\nThe top image is of the marker from the .pdf file. The bottom image is of the marker from the .svg file.\r\n\r\n\r\n**Expected outcome**\r\n\r\n<!--A description of the expected outcome from the code snippet-->\r\n<!--If this used to work in an earlier version of Matplotlib, please note the version it used to work on-->\r\nThe marker in the .pdf file should look like the marker in the .svg file.\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 17.04\r\n  * Matplotlib version: 2.1.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: Python 3.5.3\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \r\n\r\nI installed matplotlib via pip.\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": "2020-06-10T20:38:57Z",
      "base_commit": "f42b24f534ce5d9eb89d95387695bb2cee017829",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_pdf.py",
          "type": "function",
          "name": "markerObject",
          "class_name": "PdfFile",
          "code": "def markerObject(self, path, trans, fill, stroke, lw, joinstyle,\n                     capstyle):\n        \"\"\"Return name of a marker XObject representing the given path.\"\"\"\n        # self.markers used by markerObject, writeMarkers, close:\n        # mapping from (path operations, fill?, stroke?) to\n        #   [name, object reference, bounding box, linewidth]\n        # This enables different draw_markers calls to share the XObject\n        # if the gc is sufficiently similar: colors etc can vary, but\n        # the choices of whether to fill and whether to stroke cannot.\n        # We need a bounding box enclosing all of the XObject path,\n        # but since line width may vary, we store the maximum of all\n        # occurring line widths in self.markers.\n        # close() is somewhat tightly coupled in that it expects the\n        # first two components of each value in self.markers to be the\n        # name and object reference.\n        pathops = self.pathOperations(path, trans, simplify=False)\n        key = (tuple(pathops), bool(fill), bool(stroke), joinstyle, capstyle)\n        result = self.markers.get(key)\n        if result is None:\n            name = Name('M%d' % len(self.markers))\n            ob = self.reserveObject('marker %d' % len(self.markers))\n            bbox = path.get_extents(trans)\n            self.markers[key] = [name, ob, bbox, lw]\n        else:\n            if result[-1] < lw:\n                result[-1] = lw\n            name = result[0]\n        return name"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}