{
  "instance_id": "matplotlib__matplotlib-25311",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-02-23T21:04:12Z",
  "problem_statement": "[Bug]: Unable to pickle figure with draggable legend\n### Bug summary\r\n\r\nI am unable to pickle figure with draggable legend. Same error comes for draggable annotations.\r\n\r\n\r\n\r\n\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport pickle\r\n\r\nfig = plt.figure()\r\nax = fig.add_subplot(111)\r\n\r\ntime=[0,1,2,3,4]\r\nspeed=[40,43,45,47,48]\r\n\r\nax.plot(time,speed,label=\"speed\")\r\n\r\nleg=ax.legend()\r\nleg.set_draggable(True) #pickling works after removing this line \r\n\r\npickle.dumps(fig)\r\nplt.show()\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n`TypeError: cannot pickle 'FigureCanvasQTAgg' object`\r\n\r\n### Expected outcome\r\n\r\nPickling successful\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\nWindows 10\r\n\r\n### Matplotlib Version\r\n\r\n3.7.0\r\n\r\n### Matplotlib Backend\r\n\r\n_No response_\r\n\r\n### Python version\r\n\r\n3.10\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\npip\n",
  "patch": "diff --git a/lib/matplotlib/offsetbox.py b/lib/matplotlib/offsetbox.py\n--- a/lib/matplotlib/offsetbox.py\n+++ b/lib/matplotlib/offsetbox.py\n@@ -1505,7 +1505,6 @@ def __init__(self, ref_artist, use_blit=False):\n         if not ref_artist.pickable():\n             ref_artist.set_picker(True)\n         self.got_artist = False\n-        self.canvas = self.ref_artist.figure.canvas\n         self._use_blit = use_blit and self.canvas.supports_blit\n         self.cids = [\n             self.canvas.callbacks._connect_picklable(\n@@ -1514,6 +1513,9 @@ def __init__(self, ref_artist, use_blit=False):\n                 'button_release_event', self.on_release),\n         ]\n \n+    # A property, not an attribute, to maintain picklability.\n+    canvas = property(lambda self: self.ref_artist.figure.canvas)\n+\n     def on_motion(self, evt):\n         if self._check_still_parented() and self.got_artist:\n             dx = evt.x - self.mouse_x\n",
  "similar_bug_items": [
    {
      "pr_number": 11165,
      "pr_title": "Fix figure window icon",
      "pr_body": "## PR Summary\r\nFixes #11138 \r\n\r\n## PR Checklist\r\n\r\n- [x] Has Pytest style unit tests\r\n- [x] Code is PEP 8 compliant\r\n- [x] New features are documented, with examples if plot related\r\n- [x] 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",
      "issue_id": 11138,
      "issue_title": "Only the first figure window has mpl icon, all other figures have default tk icon.",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\nWhen plot more than one figure window, only the first figure window has the default mpl icon (come from matplotlib.ppm), and all other figure windows' icon is set to the default tk icon.\r\n\r\n**Code for reproduction**\r\n\r\ncode 1:\r\n```python\r\nimport logging\r\nimport matplotlib.pyplot as plt\r\nlogging.basicConfig(format=\"%(name)s %(levelname)s: %(message)s\", level=logging.INFO)\r\nplt.figure(1)\r\nplt.plot([1,2,3])\r\nplt.show()\r\n```\r\n\r\ncode 2:\r\n```python\r\nimport logging\r\nimport matplotlib.pyplot as plt\r\nlogging.basicConfig(format=\"%(name)s %(levelname)s: %(message)s\", level=logging.INFO)\r\nplt.figure(1)\r\nplt.plot([1,2,3])\r\nplt.figure(2)\r\nplt.plot([4,5,6])\r\nplt.show()\r\n```\r\n\r\ncode 3:\r\n```python\r\nimport logging\r\nimport matplotlib.pyplot as plt\r\nlogging.basicConfig(format=\"%(name)s %(levelname)s: %(message)s\", level=logging.INFO)\r\nplt.figure(1)\r\nplt.plot([1,2,3])\r\nplt.figure(2)\r\nplt.plot([4,5,6])\r\nplt.figure(3)\r\nplt.plot([7,8,9])\r\nplt.show()\r\n```\r\n\r\n**Actual outcome**\r\ncode 1 runs normally.\r\n\r\ncode 2 will log:\r\n```\r\nmatplotlib.backends._backend_tk INFO: Could not load matplotlib icon: can't use \"pyimage10\" as iconphoto: not a photo image\r\n```\r\nAnd the icon of figure 1 and figure 2 is different.\r\n\r\ncode 3 will log:\r\n```\r\nmatplotlib.backends._backend_tk INFO: Could not load matplotlib icon: can't use \"pyimage10\" as iconphoto: not a photo image\r\nmatplotlib.backends._backend_tk INFO: Could not load matplotlib icon: can't use \"pyimage19\" as iconphoto: not a photo image\r\n```\r\nThe icon of figure 1 is different than figure 2 and 3.\r\nScreenshot attached:\r\nIcon of window:\r\n![figure_title](https://user-images.githubusercontent.com/35422407/39393621-04eec34c-4afa-11e8-85a0-6f4aa58f8c4c.png)\r\n\r\nAnd the icon in taskbar:\r\n![taskbar](https://user-images.githubusercontent.com/35422407/39393623-0ad91244-4afa-11e8-8536-49138602fb4e.png)\r\n\r\nIf debug, the log information is coming from the exception:\r\n```\r\nfile: matplotlib/backends/_backend_tk.py\r\nclass _BackendTk()\r\nmethod new_figure_manager_given_figure()\r\n        # Put a mpl icon on the window rather than the default tk icon.\r\n        # Tkinter doesn't allow colour icons on linux systems, but tk>=8.5 has\r\n        # a iconphoto command which we call directly. Source:\r\n        # http://mail.python.org/pipermail/tkinter-discuss/2006-November/000954.html\r\n        icon_fname = os.path.join(\r\n            rcParams['datapath'], 'images', 'matplotlib.ppm')\r\n        icon_img = Tk.PhotoImage(file=icon_fname)\r\n        try:\r\n            window.tk.call('wm', 'iconphoto', window._w, icon_img)\r\n        except Exception as exc:\r\n            # log the failure (due e.g. to Tk version), but carry on\r\n            _log.info('Could not load matplotlib icon: %s', exc)\r\n```\r\n\r\n**Expected outcome**\r\nAll figure's icon are same by default.\r\n\r\n**Matplotlib version**\r\n  * Operating system: Windows 7 pro.\r\n  * Matplotlib version:  2.2.2\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: 3.6.5\r\n\r\n",
      "issue_closed_at": "2018-07-08T17:50:21Z",
      "base_commit": "c20b0c2bede5e354ac3bc90858d247c2a2d10d5e",
      "changes": [
        {
          "file": "lib/matplotlib/backends/_backend_tk.py",
          "type": "function",
          "name": "new_figure_manager_given_figure",
          "class_name": "_BackendTk",
          "code": "def new_figure_manager_given_figure(cls, num, figure):\n        \"\"\"\n        Create a new figure manager instance for the given figure.\n        \"\"\"\n        with _restore_foreground_window_at_end():\n            window = Tk.Tk(className=\"matplotlib\")\n            window.withdraw()\n\n            # Put a mpl icon on the window rather than the default tk icon.\n            # Tkinter doesn't allow colour icons on linux systems, but tk>=8.5 has\n            # a iconphoto command which we call directly. Source:\n            # http://mail.python.org/pipermail/tkinter-discuss/2006-November/000954.html\n            icon_fname = os.path.join(\n                rcParams['datapath'], 'images', 'matplotlib.ppm')\n            icon_img = Tk.PhotoImage(file=icon_fname)\n            try:\n                window.tk.call('wm', 'iconphoto', window._w, icon_img)\n            except Exception as exc:\n                # log the failure (due e.g. to Tk version), but carry on\n                _log.info('Could not load matplotlib icon: %s', exc)\n\n            canvas = cls.FigureCanvas(figure, master=window)\n            manager = cls.FigureManager(canvas, num, window)\n            if matplotlib.is_interactive():\n                manager.show()\n                canvas.draw_idle()\n            return manager"
        }
      ]
    },
    {
      "pr_number": 25193,
      "pr_title": "Fix displacement of colorbar for eps with bbox_inches='tight'",
      "pr_body": "Co-authored-by: Antony Lee <anntzer.lee@gmail.com>\n\nCloses #25176\n\n## PR Summary\n\n## PR Checklist\n\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\n\n**Documentation and Tests**\n- [ ] Has pytest style unit tests (and `pytest` passes)\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\n- [ ] New plotting related features are documented with examples.\n\n**Release Notes**\n- [ ] New features are marked with a `.. versionadded::` directive in the docstring and documented in `doc/users/next_whats_new/`\n- [ ] API changes are marked with a `.. versionchanged::` directive in the docstring and documented in `doc/api/next_api_changes/`\n- [ ] Release notes conform with instructions in  `next_whats_new/README.rst` or `next_api_changes/README.rst`\n\n<!--\nThank you so much for your PR!  To help us review your contribution, please\nconsider the following points:\n\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\n\n- Help with git and github is available at\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\n\n- Create a separate branch for your changes and open the PR from this branch. Please avoid working on `main`.\n\n- The PR title should summarize the changes, for example \"Raise ValueError on\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\n  \"Addresses issue #8576\".\n\n- The summary should provide at least 1-2 sentences describing the pull request\n  in detail (Why is this change required?  What problem does it solve?) and\n  link to any relevant issues.\n\n- If you are contributing fixes to docstrings, please pay attention to\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\n  note the difference between using single backquotes, double backquotes, and\n  asterisks in the markup.\n\nWe understand that PRs can sometimes be overwhelming, especially as the\nreviews start coming in.  Please let us know if the reviews are unclear or\nthe recommended next step seems overly demanding, if you would like help in\naddressing a reviewer's comments, or if you have been waiting too long to hear\nback on your PR.\n-->\n",
      "issue_id": 25176,
      "issue_title": "[Bug]: Colorbar is displaced when saving as .eps with bbox_inches='tight'",
      "issue_body": "### Bug summary\r\nI am plotting with `imshow()` and create a colorbar. When saving the figure as .eps with `savefig()` and option `bbox_inches='tight'`, the colorbar (without the spines) is displaced.\r\n\r\n### Code for reproduction\r\n```python\r\nimport matplotlib.pyplot as plt\r\n\r\nfig = plt.figure()\r\nplt.imshow([[1, 1], [1, 1]])\r\nplt.colorbar()\r\nplt.savefig(\"test.eps\", bbox_inches='tight')\r\nplt.show()\r\n```\r\n\r\n### Expected outcome\r\nColorbar should be placed inside the spines as when using `plt.show()`. Also, saving as .png instead of .eps works correctly.\r\n\r\n### Actual outcome\r\n![test](https://user-images.githubusercontent.com/12721580/217641953-4a6b5cdd-f777-436f-849f-6580244031b8.png)\r\n\r\n### Additional information\r\nThe bug does not happen when using Matplotlib version 5.2.3. \r\n\r\n### Operating system\r\nWindows 10\r\n\r\n### Matplotlib version\r\n3.6.3\r\n\r\n### Matplotlib backend\r\nmodule://backend_interagg\r\n\r\n### Python version\r\n3.11.0",
      "issue_closed_at": "2023-02-11T11:13:58Z",
      "base_commit": "73909bcb408886a22e2b84581d6b9e6d9907c813",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "get_default_filetype",
          "class_name": "FigureCanvasPS",
          "code": "def get_default_filetype(self):\n        return 'ps'"
        },
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "_print_ps",
          "class_name": "FigureCanvasPS",
          "code": "def _print_ps(\n            self, fmt, outfile, *,\n            metadata=None, papertype=None, orientation='portrait',\n            **kwargs):\n\n        dpi = self.figure.dpi\n        self.figure.dpi = 72  # Override the dpi kwarg\n\n        dsc_comments = {}\n        if isinstance(outfile, (str, os.PathLike)):\n            filename = pathlib.Path(outfile).name\n            dsc_comments[\"Title\"] = \\\n                filename.encode(\"ascii\", \"replace\").decode(\"ascii\")\n        dsc_comments[\"Creator\"] = (metadata or {}).get(\n            \"Creator\",\n            f\"Matplotlib v{mpl.__version__}, https://matplotlib.org/\")\n        # See https://reproducible-builds.org/specs/source-date-epoch/\n        source_date_epoch = os.getenv(\"SOURCE_DATE_EPOCH\")\n        dsc_comments[\"CreationDate\"] = (\n            datetime.datetime.utcfromtimestamp(\n                int(source_date_epoch)).strftime(\"%a %b %d %H:%M:%S %Y\")\n            if source_date_epoch\n            else time.ctime())\n        dsc_comments = \"\\n\".join(\n            f\"%%{k}: {v}\" for k, v in dsc_comments.items())\n\n        if papertype is None:\n            papertype = mpl.rcParams['ps.papersize']\n        papertype = papertype.lower()\n        _api.check_in_list(['auto', *papersize], papertype=papertype)\n\n        orientation = _api.check_getitem(\n            _Orientation, orientation=orientation.lower())\n\n        printer = (self._print_figure_tex\n                   if mpl.rcParams['text.usetex'] else\n                   self._print_figure)\n        printer(fmt, outfile, dpi=dpi, dsc_comments=dsc_comments,\n                orientation=orientation, papertype=papertype, **kwargs)"
        }
      ]
    },
    {
      "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": 2380,
      "pr_title": "check if pyparsing <<= is broken instead of checking the version",
      "pr_body": "distributions with packaged pyparsing might have already patched\npyparsing but not bumped the version number.\ncloses #2376\n",
      "issue_id": 2376,
      "issue_title": "feature test pyparsing 2 bug instead of version check",
      "issue_body": "the last pyparsing python3 weirdness already screwed distributions, the recent revert makes it even worse.\nmatplotlib should feature test for the bug that breaks 2.0.0 instead of relying on version numbers to at least give distributions a chance of resolving the issue in a sane way.\nthis patch, if accepted, should be applied to master and the 1.3.x branch\n\n``` diff\n--- a/setupext.py\n+++ b/setupext.py\n@@ -984,14 +984,13 @@ class Pyparsing(SetupPackage):\n                 \"matplotlib requires pyparsing >= {0}\".format(\n                     '.'.join(str(x) for x in required)))\n\n-        if pyparsing.__version__ == \"2.0.0\":\n-            if PY3:\n+        # check for bug in 2.0.0\n+        f = pyparsing.Forward()\n+        f <<= pyparsing.Literal('a')\n+        if f is None:\n                 return (\n                     \"pyparsing 2.0.0 has bugs that prevent its use with \"\n                     \"matplotlib\")\n-            else:\n-                return (\n-                    \"pyparsing 2.0.0 is not compatible with Python 2.x\")\n\n         return \"using pyparsing version %s\" % pyparsing.__version__\n```\n",
      "issue_closed_at": "2013-09-06T13:14:43Z",
      "base_commit": "0f8a45216a964ff76372428f95fa449709ba7565",
      "changes": [
        {
          "file": "lib/matplotlib/__init__.py",
          "type": "function",
          "name": "compare_versions",
          "class_name": null,
          "code": "def compare_versions(a, b):\n    \"return True if a is greater than or equal to b\"\n    if a:\n        a = distutils.version.LooseVersion(a)\n        b = distutils.version.LooseVersion(b)\n        if a>=b: return True\n        else: return False\n    else: return False"
        },
        {
          "file": "lib/matplotlib/__init__.py",
          "type": "function",
          "name": "compare_versions",
          "class_name": null,
          "code": "def compare_versions(a, b):\n    \"return True if a is greater than or equal to b\"\n    if a:\n        a = distutils.version.LooseVersion(a)\n        b = distutils.version.LooseVersion(b)\n        if a>=b: return True\n        else: return False\n    else: return False"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "get_install_requires",
          "class_name": "Pyparsing",
          "code": "def get_install_requires(self):\n        return ['pyparsing>=1.5.6,!=2.0.0']"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "check",
          "class_name": "PdfToPs",
          "code": "def check(self):\n        try:\n            output = check_output('pdftops -v', shell=True,\n                                  stderr=subprocess.STDOUT)\n            for line in output.splitlines():\n                line = line.decode()\n                if 'version' in line:\n                    return \"version %s\" % line.split()[2]\n        except (IndexError, ValueError, subprocess.CalledProcessError):\n            pass\n\n        raise CheckFailed()"
        }
      ]
    },
    {
      "pr_number": 18306,
      "pr_title": "Fix configure subplots",
      "pr_body": "## PR Summary\r\n\r\nFixes #18299. Draft for the moment because I still need to see if we can keep #17025 from haunting us. For the moment the reference to the subplot_tool is kept on the navbar; I think that makes more sense than an artificial cycle, as the navbar is responsible for spawning the tool anyway.\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/next_api_changes/* 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": 18299,
      "issue_title": "New configure_subplots behaves badly on TkAgg backend",
      "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\nAfter #16818, with the TkAgg backend, configure subplots window can't be closed and the sliders do not respond to input. The window close issue is straightforward to understand: the button response calls `Gcf.destroy` but the figure is intentionally not registered, so the `FigureManagerTk.destory` is never called. I suspect (but have not confirmed) that the slider events are being lost due to the new figure manager making a new `tkinter.Tk` instance [here](https://github.com/anntzer/matplotlib/blob/c55ca9fd64f74c4bb9ad57ea371ae8a1ec07b7d0/lib/matplotlib/backends/_backend_tk.py#L849). [EDIT: NOPE, see comment]\r\n\r\nCC @anntzer \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\r\nimport matplotlib.pyplot as plt\r\n\r\nmatplotlib.use(\"TkAgg\")\r\nplt.plot([1,2,3],[1,2,5])\r\n#  if not interactive\r\nplt.show()\r\n# click Configure Subplots\r\n```\r\n\r\n**Actual outcome**\r\n\r\ninteractive Tk window but sliders don't work and close window button is not responsive\r\n\r\n**Expected outcome**\r\n\r\nprevious behavior\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: Win10\r\n  * Matplotlib version: master\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: 3.8.5\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \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": "2020-08-24T15:22:34Z",
      "base_commit": "68408879105e53eb1377d15b2ecaf7538c22afb4",
      "changes": [
        {
          "file": "lib/matplotlib/_pylab_helpers.py",
          "type": "function",
          "name": "destroy",
          "class_name": "Gcf",
          "code": "def destroy(cls, num):\n        \"\"\"\n        Destroy manager *num* -- either a manager instance or a manager number.\n\n        In the interactive backends, this is bound to the window \"destroy\" and\n        \"delete\" events.\n\n        It is recommended to pass a manager instance, to avoid confusion when\n        two managers share the same number.\n        \"\"\"\n        if all(hasattr(num, attr) for attr in [\"num\", \"_cidgcf\", \"destroy\"]):\n            manager = num\n            if cls.figs.get(manager.num) is manager:\n                cls.figs.pop(manager.num)\n            else:\n                return\n        else:\n            try:\n                manager = cls.figs.pop(num)\n            except KeyError:\n                return\n        manager.canvas.mpl_disconnect(manager._cidgcf)\n        manager.destroy()\n        gc.collect(1)"
        },
        {
          "file": "lib/matplotlib/backend_bases.py",
          "type": "function",
          "name": "_update_view",
          "class_name": "NavigationToolbar2",
          "code": "def _update_view(self):\n        \"\"\"\n        Update the viewlim and position from the view and position stack for\n        each axes.\n        \"\"\"\n        nav_info = self._nav_stack()\n        if nav_info is None:\n            return\n        # Retrieve all items at once to avoid any risk of GC deleting an Axes\n        # while in the middle of the loop below.\n        items = list(nav_info.items())\n        for ax, (view, (pos_orig, pos_active)) in items:\n            ax._set_view(view)\n            # Restore both the original and modified positions\n            ax._set_position(pos_orig, 'original')\n            ax._set_position(pos_active, 'active')\n        self.canvas.draw_idle()"
        },
        {
          "file": "lib/matplotlib/backends/backend_nbagg.py",
          "type": "function",
          "name": "new_figure_manager_given_figure",
          "class_name": "_BackendNbAgg",
          "code": "def new_figure_manager_given_figure(num, figure):\n        canvas = FigureCanvasNbAgg(figure)\n        manager = FigureManagerNbAgg(canvas, num)\n        if is_interactive():\n            manager.show()\n            figure.canvas.draw_idle()\n        canvas.mpl_connect('close_event', lambda event: Gcf.destroy(manager))\n        return manager"
        },
        {
          "file": "lib/matplotlib/pyplot.py",
          "type": "function",
          "name": "subplot_tool",
          "class_name": null,
          "code": "def subplot_tool(targetfig=None):\n    \"\"\"\n    Launch a subplot tool window for a figure.\n\n    A `matplotlib.widgets.SubplotTool` instance is returned.\n    \"\"\"\n    if targetfig is None:\n        targetfig = gcf()\n    with rc_context({\"toolbar\": \"none\"}):  # No navbar for the toolfig.\n        # Use new_figure_manager() instead of figure() so that the figure\n        # doesn't get registered with pyplot.\n        manager = new_figure_manager(-1, (6, 3))\n    manager.set_window_title(\"Subplot configuration tool\")\n    tool_fig = manager.canvas.figure\n    tool_fig.subplots_adjust(top=0.9)\n    manager.show()\n    return SubplotTool(targetfig, tool_fig)"
        }
      ]
    }
  ]
}