{
  "instance_id": "matplotlib__matplotlib-25442",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-03-12T21:58:08Z",
  "problem_statement": "[Bug]: Attribute Error combining matplotlib 3.7.1 and mplcursor on data selection\n### Bug summary\r\n\r\nIf you combine mplcursor and matplotlib 3.7.1, you'll get an `AttributeError: 'NoneType' object has no attribute 'canvas'` after clicking a few data points. Henceforth, selecting a new data point will trigger the same traceback. Otherwise, it works fine. \r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\nimport mplcursors as mpl\r\n\r\nx = np.arange(1, 11)    \r\ny1 = x\r\n\r\nplt.scatter(x,y1)\r\n\r\nmpl.cursor()\r\nplt.show()\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n```\r\nTraceback (most recent call last):\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\cbook\\__init__.py\", line 304, in process\r\n    func(*args, **kwargs)\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1550, in on_release\r\n    if self._check_still_parented() and self.got_artist:\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1560, in _check_still_parented\r\n    self.disconnect()\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1568, in disconnect\r\n    self.canvas.mpl_disconnect(cid)\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1517, in <lambda>\r\n    canvas = property(lambda self: self.ref_artist.figure.canvas)\r\nAttributeError: 'NoneType' object has no attribute 'canvas'\r\n```\r\n\r\n### Expected outcome\r\n\r\nNo terminal output\r\n\r\n### Additional information\r\n\r\nUsing matplotlib 3.7.0 or lower works fine. Using a conda install or pip install doesn't affect the output. \r\n\r\n### Operating system\r\n\r\nWindows 11 and Windwos 10 \r\n\r\n### Matplotlib Version\r\n\r\n3.7.1\r\n\r\n### Matplotlib Backend\r\n\r\nQtAgg\r\n\r\n### Python version\r\n\r\n3.9.16\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\nconda\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@@ -1500,16 +1500,23 @@ def __init__(self, ref_artist, use_blit=False):\n             ref_artist.set_picker(True)\n         self.got_artist = False\n         self._use_blit = use_blit and self.canvas.supports_blit\n-        self.cids = [\n-            self.canvas.callbacks._connect_picklable(\n-                'pick_event', self.on_pick),\n-            self.canvas.callbacks._connect_picklable(\n-                'button_release_event', self.on_release),\n+        callbacks = ref_artist.figure._canvas_callbacks\n+        self._disconnectors = [\n+            functools.partial(\n+                callbacks.disconnect, callbacks._connect_picklable(name, func))\n+            for name, func in [\n+                (\"pick_event\", self.on_pick),\n+                (\"button_release_event\", self.on_release),\n+                (\"motion_notify_event\", self.on_motion),\n+            ]\n         ]\n \n     # A property, not an attribute, to maintain picklability.\n     canvas = property(lambda self: self.ref_artist.figure.canvas)\n \n+    cids = property(lambda self: [\n+        disconnect.args[0] for disconnect in self._disconnectors[:2]])\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@@ -1536,16 +1543,12 @@ def on_pick(self, evt):\n                 self.ref_artist.draw(\n                     self.ref_artist.figure._get_renderer())\n                 self.canvas.blit()\n-            self._c1 = self.canvas.callbacks._connect_picklable(\n-                \"motion_notify_event\", self.on_motion)\n             self.save_offset()\n \n     def on_release(self, event):\n         if self._check_still_parented() and self.got_artist:\n             self.finalize_offset()\n             self.got_artist = False\n-            self.canvas.mpl_disconnect(self._c1)\n-\n             if self._use_blit:\n                 self.ref_artist.set_animated(False)\n \n@@ -1558,14 +1561,8 @@ def _check_still_parented(self):\n \n     def disconnect(self):\n         \"\"\"Disconnect the callbacks.\"\"\"\n-        for cid in self.cids:\n-            self.canvas.mpl_disconnect(cid)\n-        try:\n-            c1 = self._c1\n-        except AttributeError:\n-            pass\n-        else:\n-            self.canvas.mpl_disconnect(c1)\n+        for disconnector in self._disconnectors:\n+            disconnector()\n \n     def save_offset(self):\n         pass\n",
  "similar_bug_items": [
    {
      "pr_number": 23332,
      "pr_title": "Validate Text linespacing on input.",
      "pr_body": "... and also tweak test_two_2line_texts to not set up an axes (just a\r\nfigure), as it doesn't need that.\r\n\r\nCloses #19223 (transform_rotates_text is no longer a problem since #19575).\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n**Tests and Styling**\r\n- [ ] Has pytest style unit tests (and `pytest` passes).\r\n- [ ] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n\r\n**Documentation**\r\n- [ ] New features are documented, with examples if plot related.\r\n- [ ] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [ ] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\r\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\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 main, 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": 19223,
      "issue_title": "Certain non-hashable parameters to text() give cryptic error messages",
      "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\nPer the title.  See https://discourse.matplotlib.org/t/cryptic-exception-when-axes-parameters-are-not-as-expected/ as well.\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\nfigtext(.5, .5, \"foo\", rotation=[90])\r\n# or\r\nfigtext(.5, .5, \"foo\", transform_rotates_text=[0])\r\n# or\r\nfigtext(.5, .5, \"foo\", linespacing=[0])\r\n```\r\nall fail with\r\n```\r\nTypeError: unhashable type: 'list'\r\n```\r\n\r\n**Actual outcome**\r\n\r\nsee above\r\n\r\n**Expected outcome**\r\n\r\nError out in the setter instead.\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: linux\r\n  * Matplotlib version: (`import matplotlib; print(matplotlib.__version__)`): master\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): any\r\n  * Python version: 38\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": "2022-06-24T22:25:33Z",
      "base_commit": "6e5a5415ddc513606eac5c116fde492370a3f7f6",
      "changes": [
        {
          "file": "lib/matplotlib/text.py",
          "type": "line",
          "name": "line 5",
          "code": "import functools\nimport logging\nimport math\nimport numbers\nimport weakref\n\nimport numpy as np"
        },
        {
          "file": "lib/matplotlib/text.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Annotation",
          "code": "def __init__(self, text, xy,\n                 xytext=None,\n                 xycoords='data',\n                 textcoords=None,\n                 arrowprops=None,\n                 annotation_clip=None,\n                 **kwargs):\n        \"\"\"\n        Annotate the point *xy* with text *text*.\n\n        In the simplest form, the text is placed at *xy*.\n\n        Optionally, the text can be displayed in another position *xytext*.\n        An arrow pointing from the text to the annotated point *xy* can then\n        be added by defining *arrowprops*.\n\n        Parameters\n        ----------\n        text : str\n            The text of the annotation.\n\n        xy : (float, float)\n            The point *(x, y)* to annotate. The coordinate system is determined\n            by *xycoords*.\n\n        xytext : (float, float), default: *xy*\n            The position *(x, y)* to place the text at. The coordinate system\n            is determined by *textcoords*.\n\n        xycoords : str or `.Artist` or `.Transform` or callable or \\\n(float, float), default: 'data'\n\n            The coordinate system that *xy* is given in. The following types\n            of values are supported:\n\n            - One of the following strings:\n\n              ==================== ============================================\n              Value                Description\n              ==================== ============================================\n              'figure points'      Points from the lower left of the figure\n              'figure pixels'      Pixels from the lower left of the figure\n              'figure fraction'    Fraction of figure from lower left\n              'subfigure points'   Points from the lower left of the subfigure\n              'subfigure pixels'   Pixels from the lower left of the subfigure\n              'subfigure fraction' Fraction of subfigure from lower left\n              'axes points'        Points from lower left corner of axes\n              'axes pixels'        Pixels from lower left corner of axes\n              'axes fraction'      Fraction of axes from lower left\n              'data'               Use the coordinate system of the object\n                                   being annotated (default)\n              'polar'              *(theta, r)* if not native 'data'\n                                   coordinates\n              ==================== ============================================\n\n              Note that 'subfigure pixels' and 'figure pixels' are the same\n              for the parent figure, so users who want code that is usable in\n              a subfigure can use 'subfigure pixels'.\n\n            - An `.Artist`: *xy* is interpreted as a fraction of the artist's\n              `~matplotlib.transforms.Bbox`. E.g. *(0, 0)* would be the lower\n              left corner of the bounding box and *(0.5, 1)* would be the\n              center top of the bounding box.\n\n            - A `.Transform` to transform *xy* to screen coordinates.\n\n            - A function with one of the following signatures::\n\n                def transform(renderer) -> Bbox\n                def transform(renderer) -> Transform\n\n              where *renderer* is a `.RendererBase` subclass.\n\n              The result of the function is interpreted like the `.Artist` and\n              `.Transform` cases above.\n\n            - A tuple *(xcoords, ycoords)* specifying separate coordinate\n              systems for *x* and *y*. *xcoords* and *ycoords* must each be\n              of one of the above described types.\n\n            See :ref:`plotting-guide-annotation` for more details.\n\n        textcoords : str or `.Artist` or `.Transform` or callable or \\\n(float, float), default: value of *xycoords*\n            The coordinate system that *xytext* is given in.\n\n            All *xycoords* values are valid as well as the following\n            strings:\n\n            =================   =========================================\n            Value               Description\n            =================   =========================================\n            'offset points'     Offset (in points) from the *xy* value\n            'offset pixels'     Offset (in pixels) from the *xy* value\n            =================   =========================================\n\n        arrowprops : dict, optional\n            The properties used to draw a `.FancyArrowPatch` arrow between the\n            positions *xy* and *xytext*.  Defaults to None, i.e. no arrow is\n            drawn.\n\n            For historical reasons there are two different ways to specify\n            arrows, \"simple\" and \"fancy\":\n\n            **Simple arrow:**\n\n            If *arrowprops* does not contain the key 'arrowstyle' the\n            allowed keys are:\n\n            ==========   ======================================================\n            Key          Description\n            ==========   ======================================================\n            width        The width of the arrow in points\n            headwidth    The width of the base of the arrow head in points\n            headlength   The length of the arrow head in points\n            shrink       Fraction of total length to shrink from both ends\n            ?            Any key to :class:`matplotlib.patches.FancyArrowPatch`\n            ==========   ======================================================\n\n            The arrow is attached to the edge of the text box, the exact\n            position (corners or centers) depending on where it's pointing to.\n\n            **Fancy arrow:**\n\n            This is used if 'arrowstyle' is provided in the *arrowprops*.\n\n            Valid keys are the following `~matplotlib.patches.FancyArrowPatch`\n            parameters:\n\n            ===============  ==================================================\n            Key              Description\n            ===============  ==================================================\n            arrowstyle       the arrow style\n            connectionstyle  the connection style\n            relpos           see below; default is (0.5, 0.5)\n            patchA           default is bounding box of the text\n            patchB           default is None\n            shrinkA          default is 2 points\n            shrinkB          default is 2 points\n            mutation_scale   default is text size (in points)\n            mutation_aspect  default is 1.\n            ?                any key for :class:`matplotlib.patches.PathPatch`\n            ===============  ==================================================\n\n            The exact starting point position of the arrow is defined by\n            *relpos*. It's a tuple of relative coordinates of the text box,\n            where (0, 0) is the lower left corner and (1, 1) is the upper\n            right corner. Values <0 and >1 are supported and specify points\n            outside the text box. By default (0.5, 0.5) the starting point is\n            centered in the text box.\n\n        annotation_clip : bool or None, default: None\n            Whether to clip (i.e. not draw) the annotation when the annotation\n            point *xy* is outside the axes area.\n\n            - If *True*, the annotation will be clipped when *xy* is outside\n              the axes.\n            - If *False*, the annotation will always be drawn.\n            - If *None*, the annotation will be clipped when *xy* is outside\n              the axes and *xycoords* is 'data'.\n\n        **kwargs\n            Additional kwargs are passed to `~matplotlib.text.Text`.\n\n        Returns\n        -------\n        `.Annotation`\n\n        See Also\n        --------\n        :ref:`plotting-guide-annotation`\n\n        \"\"\"\n        _AnnotationBase.__init__(self,\n                                 xy,\n                                 xycoords=xycoords,\n                                 annotation_clip=annotation_clip)\n        # warn about wonky input data\n        if (xytext is None and\n                textcoords is not None and\n                textcoords != xycoords):\n            _api.warn_external(\"You have used the `textcoords` kwarg, but \"\n                               \"not the `xytext` kwarg.  This can lead to \"\n                               \"surprising results.\")\n\n        # clean up textcoords and assign default\n        if textcoords is None:\n            textcoords = self.xycoords\n        self._textcoords = textcoords\n\n        # cleanup xytext defaults\n        if xytext is None:\n            xytext = self.xy\n        x, y = xytext\n\n        self.arrowprops = arrowprops\n        if arrowprops is not None:\n            arrowprops = arrowprops.copy()\n            if \"arrowstyle\" in arrowprops:\n                self._arrow_relpos = arrowprops.pop(\"relpos\", (0.5, 0.5))\n            else:\n                # modified YAArrow API to be used with FancyArrowPatch\n                for key in [\n                        'width', 'headwidth', 'headlength', 'shrink', 'frac']:\n                    arrowprops.pop(key, None)\n            self.arrow_patch = FancyArrowPatch((0, 0), (1, 1), **arrowprops)\n        else:\n            self.arrow_patch = None\n\n        # Must come last, as some kwargs may be propagated to arrow_patch.\n        Text.__init__(self, x, y, text, **kwargs)"
        },
        {
          "file": "lib/matplotlib/text.py",
          "type": "function",
          "name": "set_linespacing",
          "class_name": "Text",
          "code": "def set_linespacing(self, spacing):\n        \"\"\"\n        Set the line spacing as a multiple of the font size.\n\n        The default line spacing is 1.2.\n\n        Parameters\n        ----------\n        spacing : float (multiple of font size)\n        \"\"\"\n        self._linespacing = spacing\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/text.py",
          "type": "function",
          "name": "set_rotation",
          "class_name": "Text",
          "code": "def set_rotation(self, s):\n        \"\"\"\n        Set the rotation of the text.\n\n        Parameters\n        ----------\n        s : float or {'vertical', 'horizontal'}\n            The rotation angle in degrees in mathematically positive direction\n            (counterclockwise). 'horizontal' equals 0, 'vertical' equals 90.\n        \"\"\"\n        if isinstance(s, numbers.Real):\n            self._rotation = float(s) % 360\n        elif cbook._str_equal(s, 'horizontal') or s is None:\n            self._rotation = 0.\n        elif cbook._str_equal(s, 'vertical'):\n            self._rotation = 90.\n        else:\n            raise ValueError(\"rotation must be 'vertical', 'horizontal' or \"\n                             f\"a number, not {s}\")\n        self.stale = True"
        }
      ]
    },
    {
      "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": 20761,
      "pr_title": "Fix suplabel autopos",
      "pr_body": "## PR Summary\r\nCloses #20760\r\n\r\nIf you specify the x-position of an supxlabel (or y position of supylabel) it should not disqualify the label from constrained_layout...\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\n\r\nfig, ax = plt.subplots(constrained_layout=True)\r\nlab = fig.supxlabel('Boo', x=0.7)\r\nplt.show()\r\n```\r\n\r\n### Before:\r\n\r\n![Before](https://user-images.githubusercontent.com/1562854/127404991-6bdbc7da-5b44-44ac-8e14-ecedf8498d36.png)\r\n\r\n\r\n### After\r\n\r\n![After](https://user-images.githubusercontent.com/1562854/127404999-c8706b78-1900-439d-9d8e-e0badf1ff84c.png)\r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n\r\n- [ ] Has pytest style unit tests (and `pytest` passes).\r\n- [ ] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [ ] New features are documented, with examples if plot related.\r\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [ ] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [ ] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [ ] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\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": 20760,
      "issue_title": "[Bug]: subfigure position shifts on y-axis when x kwarg added to supxlabel",
      "issue_body": "### Bug summary\r\n\r\nLocation of subfigure shifts lower on y-axis when 'x' kwarg is used for supxlabel for that subfigure.\r\nI've also posted to StackOverflow: https://stackoverflow.com/q/68567315/9249533\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nfig = plt.figure(constrained_layout=True, figsize=(10, 8))\r\n\r\n# create top/bottom subfigs\r\n# see https://stackoverflow.com/a/68553015/9249533\r\n(subfig_t, subfig_b) = fig.subfigures(2, 1, hspace=0.05, height_ratios=[1, 3])\r\n\r\n# put ax0 in top subfig\r\nax0 = subfig_t.subplots()\r\n\r\n# create left/right subfigs nested in bottom subfig\r\n(subfig_bl, subfig_br) = subfig_b.subfigures(1, 2, wspace=0.1, width_ratios=[3, 1])\r\n\r\n# put ax1-ax3 in gridspec of bottom-left subfig\r\ngs = subfig_bl.add_gridspec(nrows=1, ncols=9)\r\n\r\nax1 = subfig_bl.add_subplot(gs[0, :3])\r\nax2 = subfig_bl.add_subplot(gs[0, 3:6], sharey=ax1)\r\nax3 = subfig_bl.add_subplot(gs[0, 6:9], sharey=ax1)\r\n\r\n\r\nax1.set_title('Nov. 7 to Nov. 13')\r\nax2.set_title('Nov. 13 to Nov. 27')\r\nax3.set_title('Nov. 27 to Dec. 31')\r\nax2.get_yaxis().set_visible(False)\r\nax3.get_yaxis().set_visible(False)\r\n\r\nsubfig_bl.supxlabel(\"My Subfigure Label\", x=0.54, size=12, fontweight='bold')\r\n\r\n# put ax4 in bottom-right subfig\r\nax4 = subfig_br.subplots()\r\nax4.set_title('Some Other Title')\r\nsubfig_br.supxlabel('Other Subfigure SubLabel', size=12, fontweight='bold')\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\nBody of subfigure shifts downward (lower on y-axis) and covers supxlabel\r\n\r\n![image](https://user-images.githubusercontent.com/41835370/127401472-20570876-b098-4cc8-bed4-d58d5cfe9669.png)\r\n\r\n\r\n\r\n### Expected outcome\r\n\r\nsubfigure position doesn't change. supxlabel shifts to right.\r\n\r\n![image](https://user-images.githubusercontent.com/41835370/127401167-48803a9c-9d2c-4b52-b109-eec49cdc89de.png)\r\n\r\n\r\n### Operating system\r\n\r\nWindows 10 Pro\r\n\r\n### Matplotlib Version\r\n\r\n3.4.2\r\n\r\n### Matplotlib Backend\r\n\r\n_No response_\r\n\r\n### Python version\r\n\r\n3.9.5\r\n\r\n### Jupyter version\r\n\r\n3.0.16\r\n\r\n### Other libraries\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\nconda\r\n\r\n### Conda channel\r\n\r\nconda-forge",
      "issue_closed_at": "2021-07-29T06:35:53Z",
      "base_commit": "7413aa92b5be5760c73e31641ab0770f328ad546",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_suplabels",
          "class_name": "FigureBase",
          "code": "def _suplabels(self, t, info, **kwargs):\n        \"\"\"\n        Add a centered %(name)s to the figure.\n\n        Parameters\n        ----------\n        t : str\n            The %(name)s text.\n        x : float, default: %(x0)s\n            The x location of the text in figure coordinates.\n        y : float, default: %(y0)s\n            The y location of the text in figure coordinates.\n        horizontalalignment, ha : {'center', 'left', 'right'}, default: %(ha)s\n            The horizontal alignment of the text relative to (*x*, *y*).\n        verticalalignment, va : {'top', 'center', 'bottom', 'baseline'}, \\\ndefault: %(va)s\n            The vertical alignment of the text relative to (*x*, *y*).\n        fontsize, size : default: :rc:`figure.titlesize`\n            The font size of the text. See `.Text.set_size` for possible\n            values.\n        fontweight, weight : default: :rc:`figure.titleweight`\n            The font weight of the text. See `.Text.set_weight` for possible\n            values.\n\n        Returns\n        -------\n        text\n            The `.Text` instance of the %(name)s.\n\n        Other Parameters\n        ----------------\n        fontproperties : None or dict, optional\n            A dict of font properties. If *fontproperties* is given the\n            default values for font size and weight are taken from the\n            `.FontProperties` defaults. :rc:`figure.titlesize` and\n            :rc:`figure.titleweight` are ignored in this case.\n\n        **kwargs\n            Additional kwargs are `matplotlib.text.Text` properties.\n        \"\"\"\n\n        suplab = getattr(self, info['name'])\n\n        x = kwargs.pop('x', None)\n        y = kwargs.pop('y', None)\n        autopos = x is None and y is None\n        if x is None:\n            x = info['x0']\n        if y is None:\n            y = info['y0']\n\n        if 'horizontalalignment' not in kwargs and 'ha' not in kwargs:\n            kwargs['horizontalalignment'] = info['ha']\n        if 'verticalalignment' not in kwargs and 'va' not in kwargs:\n            kwargs['verticalalignment'] = info['va']\n        if 'rotation' not in kwargs:\n            kwargs['rotation'] = info['rotation']\n\n        if 'fontproperties' not in kwargs:\n            if 'fontsize' not in kwargs and 'size' not in kwargs:\n                kwargs['size'] = mpl.rcParams['figure.titlesize']\n            if 'fontweight' not in kwargs and 'weight' not in kwargs:\n                kwargs['weight'] = mpl.rcParams['figure.titleweight']\n\n        sup = self.text(x, y, t, **kwargs)\n        if suplab is not None:\n            suplab.set_text(t)\n            suplab.set_position((x, y))\n            suplab.update_from(sup)\n            sup.remove()\n        else:\n            suplab = sup\n        suplab._autopos = autopos\n        setattr(self, info['name'], suplab)\n        self.stale = True\n        return suplab"
        }
      ]
    },
    {
      "pr_number": 24870,
      "pr_title": "Better default bool contour levels.",
      "pr_body": "## PR Summary\r\n\r\nCloses #24321: for bool inputs, contour now defaults to levels=[.5] and contourf to levels=[0, .5, 1].\r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n\r\n**Documentation and Tests**\r\n- [ ] Has pytest style unit tests (and `pytest` passes)\r\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [ ] New plotting related features are documented with examples.\r\n\r\n**Release Notes**\r\n- [ ] New features are marked with a `.. versionadded::` directive in the docstring and documented in `doc/users/next_whats_new/`\r\n- [ ] API changes are marked with a `.. versionchanged::` directive in the docstring and documented in `doc/api/next_api_changes/`\r\n- [ ] Release notes conform with instructions in  `next_whats_new/README.rst` or `next_api_changes/README.rst`\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- Create a separate branch for your changes and open the PR from this branch. Please avoid working on `main`.\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": 24321,
      "issue_title": "[ENH]: Auto-detect bool arrays passed to contour()?",
      "issue_body": "### Problem\n\nI find myself fairly regularly calling\r\n```python\r\nplt.contour(boolean_2d_array, levels=[.5], ...)\r\n```\r\nto draw the boundary line between True and False regions on a boolean 2d array.  Without `levels=[.5]`, one gets the default 8 levels which go at 0, 0.15, 0.3, 0.45, 0.6, 0.75, 0.9, 1.05 resulting in all the contour lines being drawn on top of one another; but clearly(?), for boolean inputs, the only choice that makes sense is to have a single level at 0.5 (or rather, anywhere between 0 and 1).\r\n```python\r\nfrom pylab import *\r\nii, jj = np.ogrid[:100, :100]; im = (ii+jj) % 20 < 10; subplot(121).contour(im); subplot(122).contour(im, levels=[.5])\r\n```\r\n![test](https://user-images.githubusercontent.com/1322974/199115826-8746ebbc-e469-48fa-a7f0-d302750018b5.png)\r\n\n\n### Proposed solution\n\nAutodetect boolean inputs to contour, and default levels to [0.5] in that case.\r\n\r\nI guess the closest similar kind of autodetection in the library is for imshow, which auto-switches between 0-1 float RGBA arrays and 0-255 uint8 RGBA arrays (when given a 3D array as input).\r\n\r\nThoughts?",
      "issue_closed_at": "2023-01-04T15:39:25Z",
      "base_commit": "6091437be9776139d3672cde28a19cbe6c09dcd5",
      "changes": [
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_autolev",
          "class_name": "ContourSet",
          "code": "def _autolev(self, N):\n        \"\"\"\n        Select contour levels to span the data.\n\n        The target number of levels, *N*, is used only when the\n        scale is not log and default locator is used.\n\n        We need two more levels for filled contours than for\n        line contours, because for the latter we need to specify\n        the lower and upper boundary of each range. For example,\n        a single contour boundary, say at z = 0, requires only\n        one contour line, but two filled regions, and therefore\n        three levels to provide boundaries for both regions.\n        \"\"\"\n        if self.locator is None:\n            if self.logscale:\n                self.locator = ticker.LogLocator()\n            else:\n                self.locator = ticker.MaxNLocator(N + 1, min_n_ticks=1)\n\n        lev = self.locator.tick_values(self.zmin, self.zmax)\n\n        try:\n            if self.locator._symmetric:\n                return lev\n        except AttributeError:\n            pass\n\n        # Trim excess levels the locator may have supplied.\n        under = np.nonzero(lev < self.zmin)[0]\n        i0 = under[-1] if len(under) else 0\n        over = np.nonzero(lev > self.zmax)[0]\n        i1 = over[0] + 1 if len(over) else len(lev)\n        if self.extend in ('min', 'both'):\n            i0 += 1\n        if self.extend in ('max', 'both'):\n            i1 -= 1\n\n        if i1 - i0 < 3:\n            i0, i1 = 0, len(lev)\n\n        return lev[i0:i1]"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_contour_args",
          "class_name": "QuadContourSet",
          "code": "def _contour_args(self, args, kwargs):\n        if self.filled:\n            fn = 'contourf'\n        else:\n            fn = 'contour'\n        nargs = len(args)\n        if nargs <= 2:\n            z = ma.asarray(args[0], dtype=np.float64)\n            x, y = self._initialize_x_y(z)\n            args = args[1:]\n        elif nargs <= 4:\n            x, y, z = self._check_xyz(args[:3], kwargs)\n            args = args[3:]\n        else:\n            raise _api.nargs_error(fn, takes=\"from 1 to 4\", given=nargs)\n        z = ma.masked_invalid(z, copy=False)\n        self.zmax = float(z.max())\n        self.zmin = float(z.min())\n        if self.logscale and self.zmin <= 0:\n            z = ma.masked_where(z <= 0, z)\n            _api.warn_external('Log scale: values of z <= 0 have been masked')\n            self.zmin = float(z.min())\n        self._process_contour_level_args(args)\n        return (x, y, z)"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_contour_args",
          "class_name": "QuadContourSet",
          "code": "def _contour_args(self, args, kwargs):\n        if self.filled:\n            fn = 'contourf'\n        else:\n            fn = 'contour'\n        nargs = len(args)\n        if nargs <= 2:\n            z = ma.asarray(args[0], dtype=np.float64)\n            x, y = self._initialize_x_y(z)\n            args = args[1:]\n        elif nargs <= 4:\n            x, y, z = self._check_xyz(args[:3], kwargs)\n            args = args[3:]\n        else:\n            raise _api.nargs_error(fn, takes=\"from 1 to 4\", given=nargs)\n        z = ma.masked_invalid(z, copy=False)\n        self.zmax = float(z.max())\n        self.zmin = float(z.min())\n        if self.logscale and self.zmin <= 0:\n            z = ma.masked_where(z <= 0, z)\n            _api.warn_external('Log scale: values of z <= 0 have been masked')\n            self.zmin = float(z.min())\n        self._process_contour_level_args(args)\n        return (x, y, z)"
        },
        {
          "file": "lib/matplotlib/tri/_tricontour.py",
          "type": "function",
          "name": "_process_args",
          "class_name": "TriContourSet",
          "code": "def _process_args(self, *args, **kwargs):\n        \"\"\"\n        Process args and kwargs.\n        \"\"\"\n        if isinstance(args[0], TriContourSet):\n            C = args[0]._contour_generator\n            if self.levels is None:\n                self.levels = args[0].levels\n            self.zmin = args[0].zmin\n            self.zmax = args[0].zmax\n            self._mins = args[0]._mins\n            self._maxs = args[0]._maxs\n        else:\n            from matplotlib import _tri\n            tri, z = self._contour_args(args, kwargs)\n            C = _tri.TriContourGenerator(tri.get_cpp_triangulation(), z)\n            self._mins = [tri.x.min(), tri.y.min()]\n            self._maxs = [tri.x.max(), tri.y.max()]\n\n        self._contour_generator = C\n        return kwargs"
        },
        {
          "file": "lib/matplotlib/tri/_tricontour.py",
          "type": "function",
          "name": "_contour_args",
          "class_name": "TriContourSet",
          "code": "def _contour_args(self, args, kwargs):\n        tri, args, kwargs = Triangulation.get_from_args_and_kwargs(*args,\n                                                                   **kwargs)\n        z = np.ma.asarray(args[0])\n        if z.shape != tri.x.shape:\n            raise ValueError('z array must have same length as triangulation x'\n                             ' and y arrays')\n\n        # z values must be finite, only need to check points that are included\n        # in the triangulation.\n        z_check = z[np.unique(tri.get_masked_triangles())]\n        if np.ma.is_masked(z_check):\n            raise ValueError('z must not contain masked points within the '\n                             'triangulation')\n        if not np.isfinite(z_check).all():\n            raise ValueError('z array must not contain non-finite values '\n                             'within the triangulation')\n\n        z = np.ma.masked_invalid(z, copy=False)\n        self.zmax = float(z_check.max())\n        self.zmin = float(z_check.min())\n        if self.logscale and self.zmin <= 0:\n            func = 'contourf' if self.filled else 'contour'\n            raise ValueError(f'Cannot {func} log of negative values.')\n        self._process_contour_level_args(args[1:])\n        return (tri, z)"
        }
      ]
    },
    {
      "pr_number": 12461,
      "pr_title": "FIX: make add_lines work with new colorbar",
      "pr_body": "## PR Summary\r\n\r\nCloses #12458\r\n\r\nSee below for a less pathological example...\r\n\r\n`cbar.add_lines` was not operational after new colorbar changes in 3.0.0.  \r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport matplotlib as mpl\r\nimport numpy as np\r\n\r\nfig = plt.figure()\r\nax = fig.add_axes([0.05, 0.8, 0.9, 0.15])\r\ncmap = mpl.cm.cool\r\nnorm = mpl.colors.Normalize(vmin=-4, vmax=4)\r\nlevels = (-1.0, 1.0, 2.0, 3.0)\r\n\r\ncb = mpl.colorbar.ColorbarBase(ax, cmap=cmap, norm=norm)\r\ncolors_bg = np.tile(list((1.0, 1.0, 1.0, 1.0)), (len(levels), 1))\r\ncb.add_lines(levels=levels, colors=colors_bg, linewidths=1)\r\n\r\n```\r\n\r\n## Before\r\n![cbarold](https://user-images.githubusercontent.com/1562854/46681303-ce871d00-cb9f-11e8-8acd-e5defedfe90c.png)\r\n\r\n\r\n## After\r\n\r\n![cbarnew](https://user-images.githubusercontent.com/1562854/46681265-b3b4a880-cb9f-11e8-8efd-87b24d565751.png)\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\nfig, ax = plt.subplots()\r\nX = np.random.rand(10, 10)*10000\r\npcm = ax.pcolormesh(X)\r\n# add 1000 to make colors visible...\r\ncont = ax.contour(X+1000)\r\ncb = fig.colorbar(pcm)\r\ncb.add_lines(cont)\r\nplt.show()\r\n```\r\n\r\n## Before\r\n\r\n![before](https://user-images.githubusercontent.com/1562854/46701766-b4683180-cbd5-11e8-8c18-d994ba47e396.png)\r\n\r\n## After\r\n![new](https://user-images.githubusercontent.com/1562854/46701741-9d294400-cbd5-11e8-9006-708b7ef51fdc.png)\r\n\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-->",
      "issue_id": 12458,
      "issue_title": "add_lines misses lines for matplotlib.colorbar.ColorbarBase",
      "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\nFor matplotlib.colorbar.ColorbarBase(...) usage of add_lines(...) may yield missing lines/poorly adjusted lines.\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.pyplot as plt\r\nimport matplotlib as mpl\r\nimport numpy as np\r\n\r\nfig = plt.figure()\r\nax = fig.add_axes([0.05, 0.8, 0.9, 0.15])\r\ncmap = mpl.cm.cool\r\nnorm = mpl.colors.Normalize(vmin=-4, vmax=4)\r\nlevels = (-1.0, 1.0, 2.0, 3.0)\r\n\r\ncb = mpl.colorbar.ColorbarBase(ax, cmap=cmap, norm=norm)\r\ncolors_bg = np.tile(list((1.0, 1.0, 1.0, 1.0)), (len(levels), 1))\r\ncb.add_lines(levels=levels, colors=colors_bg, linewidths=7)\r\n\r\nplt.show()\r\n\r\n\r\n```\r\n\r\n**Expected outcome**\r\n\r\n4 white lines, crossing the colorbar.\r\nUsed to work on matplotlib 2.2.3!\r\n\r\n**Matplotlib version**\r\n  * Operating system: Ubuntu 18.04 LTS\r\n  * Matplotlib version: 3.0.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.0\r\n\r\nMiniconda install, fully updated.",
      "issue_closed_at": "2018-10-14T19:38:43Z",
      "base_commit": "f93222a555160bf967f02dae42cce39a03384774",
      "changes": [
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "add_lines",
          "class_name": "Colorbar",
          "code": "def add_lines(self, CS, erase=True):\n        '''\n        Add the lines from a non-filled\n        :class:`~matplotlib.contour.ContourSet` to the colorbar.\n\n        Set *erase* to False if these lines should be added to\n        any pre-existing lines.\n        '''\n        if not isinstance(CS, contour.ContourSet) or CS.filled:\n            raise ValueError('add_lines is only for a ContourSet of lines')\n        tcolors = [c[0] for c in CS.tcolors]\n        tlinewidths = [t[0] for t in CS.tlinewidths]\n        # The following was an attempt to get the colorbar lines\n        # to follow subsequent changes in the contour lines,\n        # but more work is needed: specifically, a careful\n        # look at event sequences, and at how\n        # to make one object track another automatically.\n        #tcolors = [col.get_colors()[0] for col in CS.collections]\n        #tlinewidths = [col.get_linewidth()[0] for lw in CS.collections]\n        ColorbarBase.add_lines(self, CS.levels, tcolors, tlinewidths,\n                               erase=erase)"
        }
      ]
    }
  ]
}