{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 16392,
      "pr_title": "FIX colorbars for Norms that do not have a scale.",
      "pr_body": "## PR Summary\r\n\r\nIf a norm that did not have a corresponding scale is used, then the colorbar was not working.  This PR (thanks @dstansby) now keeps track of the scale associated with the colorbar and if it is \"manual\" then falls back to the manual method of creating the colorbar.  \r\n\r\nReplaces #16286\r\n\r\nCloses #16280 \r\n\r\nNote there are still issues with SymLogNorm, but this fix is regardless of that.  \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": 16280,
      "issue_title": "SymLogNorm colorbar incorrect on master",
      "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\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n\r\nThe colormap/tick locations within the colorbar are not being updated properly with a SymLogNorm on master. I think this is different than #16269 because it is actually creating the colorbar, just a wrong one.\r\n\r\nCompare the previous correct version:\r\nhttps://matplotlib.org/3.1.0/gallery/userdemo/colormap_normalizations_symlognorm.html\r\n\r\nwith a fresh checkout of master:\r\n\r\n![test](https://user-images.githubusercontent.com/12417828/72822693-c38abb80-3c2f-11ea-89f9-a101191d7bb3.png)\r\n\r\nDifference in the colorbar of the top axis.",
      "issue_closed_at": "2020-02-10T20:35:02Z",
      "base_commit": "cb904b187370119e965e5fe811f796772647e7df",
      "changes": [
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ColorbarPatch",
          "code": "def __init__(self, ax, mappable, **kw):\n        # we do not want to override the behaviour of solids\n        # so add a new attribute which will be a list of the\n        # colored patches in the colorbar\n        self.solids_patches = []\n        Colorbar.__init__(self, ax, mappable, **kw)"
        },
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "_use_auto_colorbar_locator",
          "class_name": "ColorbarBase",
          "code": "def _use_auto_colorbar_locator(self):\n        \"\"\"\n        Return if we should use an adjustable tick locator or a fixed\n        one.  (check is used twice so factored out here...)\n        \"\"\"\n        contouring = self.boundaries is not None and self.spacing == 'uniform'\n        return (type(self.norm) in [colors.Normalize, colors.LogNorm]\n                and not contouring)"
        },
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "_reset_locator_formatter_scale",
          "class_name": "ColorbarBase",
          "code": "def _reset_locator_formatter_scale(self):\n        \"\"\"\n        Reset the locator et al to defaults.  Any user-hardcoded changes\n        need to be re-entered if this gets called (either at init, or when\n        the mappable normal gets changed: Colorbar.update_normal)\n        \"\"\"\n        self.locator = None\n        self.formatter = None\n        if isinstance(self.norm, colors.LogNorm):\n            # *both* axes are made log so that determining the\n            # mid point is easier.\n            self.ax.set_xscale('log')\n            self.ax.set_yscale('log')\n            self.minorticks_on()\n        else:\n            self.ax.set_xscale('linear')\n            self.ax.set_yscale('linear')"
        },
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "_mesh",
          "class_name": "ColorbarBase",
          "code": "def _mesh(self):\n        \"\"\"\n        Return ``(X, Y)``, the coordinate arrays for the colorbar pcolormesh.\n        These are suitable for a vertical colorbar; swapping and transposition\n        for a horizontal colorbar are done outside this function.\n\n        These are scaled between vmin and vmax.\n        \"\"\"\n        # copy the norm and change the vmin and vmax to the vmin and\n        # vmax of the colorbar, not the norm.  This allows the situation\n        # where the colormap has a narrower range than the colorbar, to\n        # accomodate extra contours:\n        norm = copy.copy(self.norm)\n        norm.vmin = self.vmin\n        norm.vmax = self.vmax\n        x = np.array([0.0, 1.0])\n        if self.spacing == 'uniform':\n            y = self._uniform_y(self._central_N())\n        else:\n            y = self._proportional_y()\n        xmid = np.array([0.5])\n        try:\n            y = norm.inverse(y)\n            x = norm.inverse(x)\n            xmid = norm.inverse(xmid)\n        except ValueError:\n            # occurs for norms that don't have an inverse, in\n            # which case manually scale:\n            dv = self.vmax - self.vmin\n            x = x * dv + self.vmin\n            y = y * dv + self.vmin\n            xmid = xmid * dv + self.vmin\n        self._y = y\n        X, Y = np.meshgrid(x, y)\n        if self._extend_lower() and not self.extendrect:\n            X[0, :] = xmid\n        if self._extend_upper() and not self.extendrect:\n            X[-1, :] = xmid\n        return X, Y"
        }
      ]
    },
    {
      "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))"
        }
      ]
    },
    {
      "pr_number": 18322,
      "pr_title": "Disable FH4 so that we don't require VCRUNTIME140_1.dll.",
      "pr_body": "## PR Summary\r\n\r\nFor more details, see:\r\nhttps://devblogs.microsoft.com/cppblog/making-cpp-exception-handling-smaller-x64/\r\nhttps://github.com/joerick/cibuildwheel/issues/423#issuecomment-677763904\r\n\r\nFixes #18292.\r\n\r\n## PR Checklist\r\n\r\n- [n/a] Has Pytest style unit tests\r\n- [n/a] 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/next_api_changes/* if API changed in a backward-incompatible way",
      "issue_id": 18292,
      "issue_title": "ImportError: DLL load failed with Matplotlib 3.3.1 on Windows",
      "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\nOn some Windows machines (including Windows 10 and Windows 7), we experienced `ImportError: DLL load failed: The specific module could not be found` after upgrading from Matplotlib 3.3.0 to 3.3.1. \r\n\r\n**Code for reproduction**\r\n\r\n`import matplotlib` would trigger the import error while trying to import ft2font.\r\n\r\n\r\n**Actual outcome**\r\n\r\n```python\r\nIn [1]: import matplotlib\r\n---------------------------------------------------------------------------\r\nImportError                               Traceback (most recent call last)\r\n<ipython-input-1-0484cd13f94d> in <module>\r\n----> 1 import matplotlib\r\n\r\nc:\\users\\nzlab\\envs\\nta-1465\\lib\\site-packages\\matplotlib\\__init__.py in <module>\r\n    172\r\n    173\r\n--> 174 _check_versions()\r\n    175\r\n    176\r\n\r\nc:\\users\\nzlab\\envs\\nta-1465\\lib\\site-packages\\matplotlib\\__init__.py in _check_versions()\r\n    157     # Quickfix to ensure Microsoft Visual C++ redistributable\r\n    158     # DLLs are loaded before importing kiwisolver\r\n--> 159     from . import ft2font\r\n    160\r\n    161     for modname, minver in [\r\n\r\nImportError: DLL load failed: The specified module could not be found.\r\n\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\n\r\nNo error should be presented. \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: Windows 7 (64bit), Windows 10 (64bit)\r\n  * Matplotlib version: 3.3.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: Python 3.7.6\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: Microsoft Visual C++ 2015-2019 Redistributable (x64) - 14.27.29016 **NOT INSTALLED**\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**Further diagnose shows: **\r\n\r\n- ft2font.cp37-win_amd64.pyd (Matplotlib 3.3.0) links VCRUNTIME140.dll\r\n- ft2font.cp37-win_amd64.pyd (Matplotlib 3.3.1) links VCRUNTIME140_1.dll\r\n\r\nThose computers which have DLL load failed don't have VCRUNTIME140_1.dll in DLL search path. \r\n\r\n**Solution**\r\nInstalling the latest [Visual Studio 2015, 2019 and 2019 redistributable](https://support.microsoft.com/en-nz/help/2977003/the-latest-supported-visual-c-downloads) should address the issue. \r\n\r\n**Question**\r\nIs changing in VC++ API (from 14 to 14.1) an intentional move on Matplotlib 3.3.1 for Windows release? If so, should the VC++ dependency to be added to Installation Guide? ",
      "issue_closed_at": "2020-08-24T00:49:39Z",
      "base_commit": "5102676244c24622639601b693c552dad0b6259e",
      "changes": [
        {
          "file": "setup.py",
          "type": "function",
          "name": "build_extensions",
          "class_name": "BuildExtraLibraries",
          "code": "def build_extensions(self):\n        # Remove the -Wstrict-prototypes option, it's not valid for C++.  Fixed\n        # in Py3.7 as bpo-5755.\n        try:\n            self.compiler.compiler_so.remove('-Wstrict-prototypes')\n        except (ValueError, AttributeError):\n            pass\n\n        env = self.add_optimization_flags()\n        for package in good_packages:\n            package.do_custom_build(env)\n        return super().build_extensions()"
        }
      ]
    },
    {
      "pr_number": 15140,
      "pr_title": "Fix ScalarFormatter formatting of masked values",
      "pr_body": "Fixes #15103. Could probably do with a test. Possibly an API change since previously a masked value would be formatted as `nan`?",
      "issue_id": 15103,
      "issue_title": "Colorbar for imshow messes interactive cursor with masked data",
      "issue_body": "### Bug report\r\n\r\nAdding a colorbar when the mappable is the result of imshow for a masked array changes the behavior of the interactive z-data cursor and triggers a UserWarning.\r\n\r\n**Code for reproduction**\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\nmask = np.zeros((4, 4), bool)\r\nmask[:2] = True\r\nA = np.ma.array(np.random.rand(4, 4), mask=mask)\r\n\r\nfig, ax = plt.subplots(1, 1)\r\nim = ax.imshow(A)\r\nfig.colorbar(im)\r\n```\r\n\r\n**Actual outcome**\r\n```\r\n~/.anaconda3/lib/python3.7/site-packages/matplotlib/ticker.py:632: UserWarning: Warning: converting a masked element to nan.\r\n  return '%-12g' % value\r\n```\r\n![Screenshot from 2019-08-22 18-22-23](https://user-images.githubusercontent.com/52320542/63531938-e9d5ef00-c509-11e9-83ef-7ce9bc3f3c59.png)\r\n\r\n**Expected outcome**\r\nBefore adding the colorbar, the zdata cursor shows a blank (`[]`).\r\n\r\n**Matplotlib version**\r\n  * Operating system: Linux\r\n  * Matplotlib version: 3.1.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.3\r\n  * IPython version: 7.2.0\r\n  * Other libraries: \r\nMatplotlib installed from pip.\r\n\r\n\r\n\r\n",
      "issue_closed_at": "2019-08-28T03:46:25Z",
      "base_commit": "40dfc353aa66b93fd0fbc55ca1f51701202c0549",
      "changes": [
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "format_data_short",
          "class_name": "LogitFormatter",
          "code": "def format_data_short(self, value):\n        \"\"\"\n        Return a short formatted string representation of a number.\n        \"\"\"\n        # thresholds choosen for use scienfic notation if and only if exponent\n        # is less or equal than -2.\n        if value < 0.1:\n            return \"{:e}\".format(value)\n        if value < 0.9:\n            return \"{:f}\".format(value)\n        return \"1-{:e}\".format(1 - value)"
        }
      ]
    },
    {
      "pr_number": 4186,
      "pr_title": "Close clipped paths",
      "pr_body": "Fix #4185.\n",
      "issue_id": 4185,
      "issue_title": "Colorbar outline has broken path in vector backends",
      "issue_body": "Hi again,\nIn all of the vector graphics backends that I tested (pdf, eps, ps, svg), the colorbar outline is drawn with a broken path at the corner where the path starts and ends.  This does not happen with the raster graphic or interactive backends.  Here is a minimal working example:\n\n``` python\nimport numpy as np\nimport matplotlib as mpl\nimport matplotlib.pyplot as plt\n\nmpl.rcParams.update({'axes.linewidth':10})\n\nfig = plt.figure(figsize=(8,6))\nax1 = fig.add_axes([0.05, 0.85, 0.9, 0.1])\nax2 = fig.add_axes([0.05, 0.65, 0.9, 0.1])\nax3 = fig.add_axes([0.05, 0.45, 0.9, 0.1])\nax4 = fig.add_axes([0.05, 0.25, 0.9, 0.1])\nax5 = fig.add_axes([0.05, 0.05, 0.9, 0.1])\n\ncmap = mpl.cm.jet\ncmap.set_under('w')\ncmap.set_over('w')\n\nim = ax1.pcolormesh(np.linspace(0,10,16).reshape((4,4)))\n\nplt.colorbar(im,cax=ax2,cmap=cmap,orientation='horizontal',\n             extend='both',extendfrac=0.5)\nplt.colorbar(im,cax=ax3,cmap=cmap,orientation='horizontal',\n             extend='both',)\nplt.colorbar(im,cax=ax4,cmap=cmap,orientation='horizontal',\n             extend='both',extendrect=True)\nplt.colorbar(im,cax=ax5,cmap=cmap,orientation='horizontal',\n             extend='neither')\n\nplt.savefig('colorbar_tip.pdf')\nplt.savefig('colorbar_tip.eps')\nplt.savefig('colorbar_tip.svg')\n```\n\nTested with matplotlib 1.4.3, and python 2.7, on Linux and Windows. I have exaggerated the axes linewidth to make the problem more visible.  Here is a screenshot of the PDF output:\n\n![colorbar_tip](https://cloud.githubusercontent.com/assets/8595627/6471312/c556da04-c1b7-11e4-8711-25ba893eb043.png)\n\nI have been able to fix this by making the following change to the `ColorbarBase` class in colorbar.py:\n\n``` python\ndef _config_axes(self, X, Y):\n\n    ...\n\n    if self.outline is not None:\n        self.outline.remove()\n\n    xy2 = np.append(xy,xy[1:2,:],axis=0) # <-- added\n\n    self.outline = mpatches.Polygon(\n        xy2, edgecolor=mpl.rcParams['axes.edgecolor'], # <-- modified\n        facecolor='none',\n        linewidth=mpl.rcParams['axes.linewidth'],\n        closed=True,\n        zorder=2)\n\n    ...\n```\n\nThis essentially just copies the second point of the path to the end of the path.  However, I don't know if this is the best fix.  I don't know anything about vector graphics drawing directives, but it seems to me that there should be a builtin function to have paths wrap-around enough so that it creates a closed loop, including drawing the correct corner.  Clearly the `closed=True` option of the `mpatches.Polygon` is not doing this.\n\nAgain, thanks to anyone who looks into this.\n",
      "issue_closed_at": "2015-03-04T22:48:51Z",
      "base_commit": "6387fcaaeeff4777373429a54d79ca270cbd2522",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "print_figure_impl",
          "class_name": null,
          "code": "def print_figure_impl():\n            # write the PostScript headers\n            if isEPSF: print(\"%!PS-Adobe-3.0 EPSF-3.0\", file=fh)\n            else: print(\"%!PS-Adobe-3.0\", file=fh)\n            if title: print(\"%%Title: \"+title, file=fh)\n            print((\"%%Creator: matplotlib version \"\n                         +__version__+\", http://matplotlib.org/\"), file=fh)\n            print(\"%%CreationDate: \"+time.ctime(time.time()), file=fh)\n            print(\"%%Orientation: \" + orientation, file=fh)\n            if not isEPSF: print(\"%%DocumentPaperSizes: \"+papertype, file=fh)\n            print(\"%%%%BoundingBox: %d %d %d %d\" % bbox, file=fh)\n            if not isEPSF: print(\"%%Pages: 1\", file=fh)\n            print(\"%%EndComments\", file=fh)\n\n            Ndict = len(psDefs)\n            print(\"%%BeginProlog\", file=fh)\n            if not rcParams['ps.useafm']:\n                Ndict += len(ps_renderer.used_characters)\n            print(\"/mpldict %d dict def\"%Ndict, file=fh)\n            print(\"mpldict begin\", file=fh)\n            for d in psDefs:\n                d=d.strip()\n                for l in d.split('\\n'):\n                    print(l.strip(), file=fh)\n            if not rcParams['ps.useafm']:\n                for font_filename, chars in six.itervalues(ps_renderer.used_characters):\n                    if len(chars):\n                        font = FT2Font(font_filename)\n                        cmap = font.get_charmap()\n                        glyph_ids = []\n                        for c in chars:\n                            gind = cmap.get(c) or 0\n                            glyph_ids.append(gind)\n\n                        fonttype = rcParams['ps.fonttype']\n\n                        # Can not use more than 255 characters from a\n                        # single font for Type 3\n                        if len(glyph_ids) > 255:\n                            fonttype = 42\n\n                        # The ttf to ps (subsetting) support doesn't work for\n                        # OpenType fonts that are Postscript inside (like the\n                        # STIX fonts).  This will simply turn that off to avoid\n                        # errors.\n                        if is_opentype_cff_font(font_filename):\n                            raise RuntimeError(\"OpenType CFF fonts can not be saved using the internal Postscript backend at this time.\\nConsider using the Cairo backend.\")\n                        else:\n                            fh.flush()\n                            convert_ttf_to_ps(\n                                font_filename.encode(sys.getfilesystemencoding()),\n                                fh, fonttype, glyph_ids)\n            print(\"end\", file=fh)\n            print(\"%%EndProlog\", file=fh)\n\n            if not isEPSF: print(\"%%Page: 1 1\", file=fh)\n            print(\"mpldict begin\", file=fh)\n            #print >>fh, \"gsave\"\n            print(\"%s translate\"%_nums_to_str(xo, yo), file=fh)\n            if rotation: print(\"%d rotate\"%rotation, file=fh)\n            print(\"%s clipbox\"%_nums_to_str(width*72, height*72, 0, 0), file=fh)\n\n            # write the figure\n            content = self._pswriter.getvalue()\n            if not isinstance(content, six.text_type):\n                content = content.decode('ascii')\n            print(content, file=fh)\n\n            # write the trailer\n            #print >>fh, \"grestore\"\n            print(\"end\", file=fh)\n            print(\"showpage\", file=fh)\n            if not isEPSF: print(\"%%EOF\", file=fh)\n            fh.flush()"
        },
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "write",
          "class_name": "NullWriter",
          "code": "def write(self, *kl, **kwargs):\n                    pass"
        },
        {
          "file": "lib/matplotlib/backends/backend_svg.py",
          "type": "function",
          "name": "_write_default_style",
          "class_name": "RendererSVG",
          "code": "def _write_default_style(self):\n        writer = self.writer\n        default_style = generate_css({\n            'stroke-linejoin': 'round',\n            'stroke-linecap': 'butt'})\n        writer.start('defs')\n        writer.start('style', type='text/css')\n        writer.data('*{%s}\\n' % default_style)\n        writer.end('style')\n        writer.end('defs')"
        }
      ]
    }
  ]
}