{
  "Selected_candidate": {
    "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)"
      }
    ]
  },
  "Justification": "Candidate E is the most relevant as it also concerns functionality within the TkAgg backend which is being used in the CURRENT bug. The issue reported in Candidate E deals with interactive window elements in Matplotlib, similar to the problems encountered when attempting to pickle a figure with a draggable legend in the CURRENT bug. Both reports involve interaction aspects of Matplotlib's rendering system and can lead to insights about handling backend-specific behaviors and ensuring functionality works as intended. The structural and symptom similarities make this candidate valuable for debugging the CURRENT bug.",
  "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"
}