{
  "Selected_candidate": {
    "pr_number": 18328,
    "pr_title": "Add missing check for None in Qt toolmanager.",
    "pr_body": "(The fix is already there in all other backends.)\r\ncloses #18327; adding tests is tracked as https://github.com/matplotlib/matplotlib/issues/17999.\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/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": 18327,
    "issue_title": "Tool Manager: adding buttons to toolbar fails with matplotlib version 3.3.1 using Qt backend",
    "issue_body": "### Bug report\r\n\r\nAdding buttons on the toolbar in matplotlib version 3.3.1 produces `AttributeError` with `Qt5Agg` backend.\r\n\r\n**Code for reproduction**\r\nThe easiest way to reproduce: from the [original example gallery](https://matplotlib.org/gallery/user_interfaces/toolmanager_sgskip.html?highlight=tool%20manager):\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nplt.rcParams['toolbar'] = 'toolmanager'\r\nfrom matplotlib.backend_tools import ToolBase, ToolToggleBase\r\n\r\nplt.switch_backend(\"Qt5Agg\")\r\n\r\nclass ListTools(ToolBase):\r\n    \"\"\"List all the tools controlled by the `ToolManager`.\"\"\"\r\n    # keyboard shortcut\r\n    default_keymap = 'm'\r\n    description = 'List Tools'\r\n\r\n    def trigger(self, *args, **kwargs):\r\n        print('_' * 80)\r\n        print(\"{0:12} {1:45} {2}\".format(\r\n            'Name (id)', 'Tool description', 'Keymap'))\r\n        print('-' * 80)\r\n        tools = self.toolmanager.tools\r\n        for name in sorted(tools):\r\n            if not tools[name].description:\r\n                continue\r\n            keys = ', '.join(sorted(self.toolmanager.get_tool_keymap(name)))\r\n            print(\"{0:12} {1:45} {2}\".format(\r\n                name, tools[name].description, keys))\r\n        print('_' * 80)\r\n        print(\"Active Toggle tools\")\r\n        print(\"{0:12} {1:45}\".format(\"Group\", \"Active\"))\r\n        print('-' * 80)\r\n        for group, active in self.toolmanager.active_toggle.items():\r\n            print(\"{0:12} {1:45}\".format(str(group), str(active)))\r\n\r\n\r\nclass GroupHideTool(ToolToggleBase):\r\n    \"\"\"Show lines with a given gid.\"\"\"\r\n    default_keymap = 'G'\r\n    description = 'Show by gid'\r\n    default_toggled = True\r\n\r\n    def __init__(self, *args, gid, **kwargs):\r\n        self.gid = gid\r\n        super().__init__(*args, **kwargs)\r\n\r\n    def enable(self, *args):\r\n        self.set_lines_visibility(True)\r\n\r\n    def disable(self, *args):\r\n        self.set_lines_visibility(False)\r\n\r\n    def set_lines_visibility(self, state):\r\n        for ax in self.figure.get_axes():\r\n            for line in ax.get_lines():\r\n                if line.get_gid() == self.gid:\r\n                    line.set_visible(state)\r\n        self.figure.canvas.draw()\r\n\r\n\r\nfig = plt.figure()\r\nplt.plot([1, 2, 3], gid='mygroup')\r\nplt.plot([2, 3, 4], gid='unknown')\r\nplt.plot([3, 2, 1], gid='mygroup')\r\n\r\n# Add the custom tools that we created\r\nfig.canvas.manager.toolmanager.add_tool('List', ListTools)\r\nfig.canvas.manager.toolmanager.add_tool('Show', GroupHideTool, gid='mygroup')\r\n\r\n\r\n# Add an existing tool to new group `foo`.\r\n# It can be added as many times as we want\r\nfig.canvas.manager.toolbar.add_tool('zoom', 'foo')\r\n\r\n# Remove the forward button\r\nfig.canvas.manager.toolmanager.remove_tool('forward')\r\n\r\n# To add a custom tool to the toolbar at specific location inside\r\n# the navigation group\r\nfig.canvas.manager.toolbar.add_tool('Show', 'navigation', 1)\r\n\r\nplt.show()\r\n```\r\n\r\n**Actual outcome**\r\n\r\n```\r\nTreat the new Tool classes introduced in v1.5 as experimental for now, the API will likely change in version 2.1 and perhaps the rcParam as well\r\nE:\\mpl_toolbar_issue.py:57: UserWarning: The new Tool classes introduced in v1.5 are experimental; their API (including names) will likely change in future versions.\r\n  fig = plt.figure()\r\nE:\\mpl_toolbar_issue.py:63: UserWarning: The new Tool classes introduced in v1.5 are experimental; their API (including names) will likely change in future versions.\r\n  fig.canvas.manager.toolmanager.add_tool('List', ListTools)\r\nE:\\mpl_toolbar_issue.py:41: UserWarning: The new Tool classes introduced in v1.5 are experimental; their API (including names) will likely change in future versions.\r\n  super().__init__(*args, **kwargs)\r\nE:\\mpl_toolbar_issue.py:64: UserWarning: Key G changed from grid_minor to Show\r\n  fig.canvas.manager.toolmanager.add_tool('Show', GroupHideTool, gid='mygroup')\r\nTraceback (most recent call last):\r\n  File \"E:\\mpl_toolbar_issue.py\", line 76, in <module>\r\n    fig.canvas.manager.toolbar.add_tool('Show', 'navigation', 1)\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\matplotlib\\backend_bases.py\", line 3333, in add_tool\r\n    image, tool.description, toggle)\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\matplotlib\\backends\\backend_qt5.py\", line 919, in add_toolitem\r\n    button.setIcon(NavigationToolbar2QT._icon(self, image_file))\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\matplotlib\\backends\\backend_qt5.py\", line 711, in _icon\r\n    name = name.replace('.png', '_large.png')\r\nAttributeError: 'NoneType' object has no attribute 'replace'\r\n```\r\n\r\nHowever, running it on  for example `TkAgg` backend works fine. I also downgraded to `3.3.0` and it seems to fix the issue, so I think something is broken in release `3.3.1`.\r\n\r\n**Matplotlib version**\r\n  * Operating system: Windows 10\r\n  * Matplotlib version: 3.3.1\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: 3.7.6\r\n\r\nI installed matplotlib 3.3.1 on pip and conda as well, and it seems broken on both.\r\n",
    "issue_closed_at": "2020-08-23T15:43:19Z",
    "base_commit": "48b63273f940e9745010655f50078210afd3d0ea",
    "changes": [
      {
        "file": "lib/matplotlib/backends/backend_qt5.py",
        "type": "function",
        "name": "add_toolitem",
        "class_name": "ToolbarQt",
        "code": "def add_toolitem(\n            self, name, group, position, image_file, description, toggle):\n\n        button = QtWidgets.QToolButton(self)\n        button.setIcon(NavigationToolbar2QT._icon(self, image_file))\n        button.setText(name)\n        if description:\n            button.setToolTip(description)\n\n        def handler():\n            self.trigger_tool(name)\n        if toggle:\n            button.setCheckable(True)\n            button.toggled.connect(handler)\n        else:\n            button.clicked.connect(handler)\n\n        self._toolitems.setdefault(name, [])\n        self._add_to_group(group, name, button, position)\n        self._toolitems[name].append((button, handler))"
      }
    ]
  },
  "Justification": "Candidate B is the most relevant as it focuses on usability and interaction within the Matplotlib UI, similar to the proposed enhancement of easily comparable version info. Both attempts aim to improve user experience and implementation efficiency, providing insights into enhancing functionalities and rectifying usability issues in a consistent manner.",
  "instance_id": "matplotlib__matplotlib-18869",
  "repo": "matplotlib/matplotlib",
  "created_at": "2020-11-01T23:18:42Z",
  "problem_statement": "Add easily comparable version info to toplevel\n<!--\r\nWelcome! Thanks for thinking of a way to improve Matplotlib.\r\n\r\n\r\nBefore creating a new feature request please search the issues for relevant feature requests.\r\n-->\r\n\r\n### Problem\r\n\r\nCurrently matplotlib only exposes `__version__`.  For quick version checks, exposing either a `version_info` tuple (which can be compared with other tuples) or a `LooseVersion` instance (which can be properly compared with other strings) would be a small usability improvement.\r\n\r\n(In practice I guess boring string comparisons will work just fine until we hit mpl 3.10 or 4.10 which is unlikely to happen soon, but that feels quite dirty :))\r\n<!--\r\nProvide a clear and concise description of the problem this feature will solve. \r\n\r\nFor example:\r\n* I'm always frustrated when [...] because [...]\r\n* I would like it if [...] happened when I [...] because [...]\r\n* Here is a sample image of what I am asking for [...]\r\n-->\r\n\r\n### Proposed Solution\r\n\r\nI guess I slightly prefer `LooseVersion`, but exposing just a `version_info` tuple is much more common in other packages (and perhaps simpler to understand).  The hardest(?) part is probably just bikeshedding this point :-)\r\n<!-- Provide a clear and concise description of a way to accomplish what you want. For example:\r\n\r\n* Add an option so that when [...]  [...] will happen\r\n -->\r\n\r\n### Additional context and prior art\r\n\r\n`version_info` is a pretty common thing (citation needed).\r\n<!-- Add any other context or screenshots about the feature request here. You can also include links to examples of other programs that have something similar to your request. For example:\r\n\r\n* Another project [...] solved this by [...]\r\n-->\r\n\n",
  "patch": "diff --git a/lib/matplotlib/__init__.py b/lib/matplotlib/__init__.py\n--- a/lib/matplotlib/__init__.py\n+++ b/lib/matplotlib/__init__.py\n@@ -129,25 +129,60 @@\n   year      = 2007\n }\"\"\"\n \n+# modelled after sys.version_info\n+_VersionInfo = namedtuple('_VersionInfo',\n+                          'major, minor, micro, releaselevel, serial')\n \n-def __getattr__(name):\n-    if name == \"__version__\":\n+\n+def _parse_to_version_info(version_str):\n+    \"\"\"\n+    Parse a version string to a namedtuple analogous to sys.version_info.\n+\n+    See:\n+    https://packaging.pypa.io/en/latest/version.html#packaging.version.parse\n+    https://docs.python.org/3/library/sys.html#sys.version_info\n+    \"\"\"\n+    v = parse_version(version_str)\n+    if v.pre is None and v.post is None and v.dev is None:\n+        return _VersionInfo(v.major, v.minor, v.micro, 'final', 0)\n+    elif v.dev is not None:\n+        return _VersionInfo(v.major, v.minor, v.micro, 'alpha', v.dev)\n+    elif v.pre is not None:\n+        releaselevel = {\n+            'a': 'alpha',\n+            'b': 'beta',\n+            'rc': 'candidate'}.get(v.pre[0], 'alpha')\n+        return _VersionInfo(v.major, v.minor, v.micro, releaselevel, v.pre[1])\n+    else:\n+        # fallback for v.post: guess-next-dev scheme from setuptools_scm\n+        return _VersionInfo(v.major, v.minor, v.micro + 1, 'alpha', v.post)\n+\n+\n+def _get_version():\n+    \"\"\"Return the version string used for __version__.\"\"\"\n+    # Only shell out to a git subprocess if really needed, and not on a\n+    # shallow clone, such as those used by CI, as the latter would trigger\n+    # a warning from setuptools_scm.\n+    root = Path(__file__).resolve().parents[2]\n+    if (root / \".git\").exists() and not (root / \".git/shallow\").exists():\n         import setuptools_scm\n+        return setuptools_scm.get_version(\n+            root=root,\n+            version_scheme=\"post-release\",\n+            local_scheme=\"node-and-date\",\n+            fallback_version=_version.version,\n+        )\n+    else:  # Get the version from the _version.py setuptools_scm file.\n+        return _version.version\n+\n+\n+def __getattr__(name):\n+    if name in (\"__version__\", \"__version_info__\"):\n         global __version__  # cache it.\n-        # Only shell out to a git subprocess if really needed, and not on a\n-        # shallow clone, such as those used by CI, as the latter would trigger\n-        # a warning from setuptools_scm.\n-        root = Path(__file__).resolve().parents[2]\n-        if (root / \".git\").exists() and not (root / \".git/shallow\").exists():\n-            __version__ = setuptools_scm.get_version(\n-                root=root,\n-                version_scheme=\"post-release\",\n-                local_scheme=\"node-and-date\",\n-                fallback_version=_version.version,\n-            )\n-        else:  # Get the version from the _version.py setuptools_scm file.\n-            __version__ = _version.version\n-        return __version__\n+        __version__ = _get_version()\n+        global __version__info__  # cache it.\n+        __version_info__ = _parse_to_version_info(__version__)\n+        return __version__ if name == \"__version__\" else __version_info__\n     raise AttributeError(f\"module {__name__!r} has no attribute {name!r}\")\n \n \n"
}