{
    "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."
}