{
  "instance_id": "matplotlib__matplotlib-22835",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-04-12T23:13:58Z",
  "problem_statement": "[Bug]: scalar mappable format_cursor_data crashes on BoundarNorm\n### Bug summary\r\n\r\nIn 3.5.0 if you do:\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nimport matplotlib as mpl\r\n\r\nfig, ax = plt.subplots()\r\nnorm = mpl.colors.BoundaryNorm(np.linspace(-4, 4, 5), 256)\r\nX = np.random.randn(10, 10)\r\npc = ax.imshow(X, cmap='RdBu_r', norm=norm)\r\n```\r\n\r\nand mouse over the image, it crashes with\r\n\r\n```\r\nFile \"/Users/jklymak/matplotlib/lib/matplotlib/artist.py\", line 1282, in format_cursor_data\r\n    neighbors = self.norm.inverse(\r\n  File \"/Users/jklymak/matplotlib/lib/matplotlib/colors.py\", line 1829, in inverse\r\n    raise ValueError(\"BoundaryNorm is not invertible\")\r\nValueError: BoundaryNorm is not invertible\r\n```\r\n\r\nand interaction stops.  \r\n\r\nNot sure if we should have a special check here, a try-except, or actually just make BoundaryNorm approximately invertible.  \r\n\r\n\r\n### Matplotlib Version\r\n\r\nmain 3.5.0\r\n\r\n\n[Bug]: scalar mappable format_cursor_data crashes on BoundarNorm\n### Bug summary\r\n\r\nIn 3.5.0 if you do:\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nimport matplotlib as mpl\r\n\r\nfig, ax = plt.subplots()\r\nnorm = mpl.colors.BoundaryNorm(np.linspace(-4, 4, 5), 256)\r\nX = np.random.randn(10, 10)\r\npc = ax.imshow(X, cmap='RdBu_r', norm=norm)\r\n```\r\n\r\nand mouse over the image, it crashes with\r\n\r\n```\r\nFile \"/Users/jklymak/matplotlib/lib/matplotlib/artist.py\", line 1282, in format_cursor_data\r\n    neighbors = self.norm.inverse(\r\n  File \"/Users/jklymak/matplotlib/lib/matplotlib/colors.py\", line 1829, in inverse\r\n    raise ValueError(\"BoundaryNorm is not invertible\")\r\nValueError: BoundaryNorm is not invertible\r\n```\r\n\r\nand interaction stops.  \r\n\r\nNot sure if we should have a special check here, a try-except, or actually just make BoundaryNorm approximately invertible.  \r\n\r\n\r\n### Matplotlib Version\r\n\r\nmain 3.5.0\r\n\r\n\n",
  "patch": "diff --git a/lib/matplotlib/artist.py b/lib/matplotlib/artist.py\n--- a/lib/matplotlib/artist.py\n+++ b/lib/matplotlib/artist.py\n@@ -12,6 +12,7 @@\n \n import matplotlib as mpl\n from . import _api, cbook\n+from .colors import BoundaryNorm\n from .cm import ScalarMappable\n from .path import Path\n from .transforms import (Bbox, IdentityTransform, Transform, TransformedBbox,\n@@ -1303,10 +1304,20 @@ def format_cursor_data(self, data):\n                 return \"[]\"\n             normed = self.norm(data)\n             if np.isfinite(normed):\n-                # Midpoints of neighboring color intervals.\n-                neighbors = self.norm.inverse(\n-                    (int(self.norm(data) * n) + np.array([0, 1])) / n)\n-                delta = abs(neighbors - data).max()\n+                if isinstance(self.norm, BoundaryNorm):\n+                    # not an invertible normalization mapping\n+                    cur_idx = np.argmin(np.abs(self.norm.boundaries - data))\n+                    neigh_idx = max(0, cur_idx - 1)\n+                    # use max diff to prevent delta == 0\n+                    delta = np.diff(\n+                        self.norm.boundaries[neigh_idx:cur_idx + 2]\n+                    ).max()\n+\n+                else:\n+                    # Midpoints of neighboring color intervals.\n+                    neighbors = self.norm.inverse(\n+                        (int(normed * n) + np.array([0, 1])) / n)\n+                    delta = abs(neighbors - data).max()\n                 g_sig_digits = cbook._g_sig_digits(data, delta)\n             else:\n                 g_sig_digits = 3  # Consistent with default below.\n",
  "similar_bug_items": [
    {
      "pr_number": 14014,
      "pr_title": "Disallow figure argument for pyplot.subplot() and Figure.add_subplot()",
      "pr_body": "## PR Summary\r\n\r\nCloses #14011.\r\n\r\n`Figure.add_subplot()` and `pyplot.subplot()` have technically accepted a `figure` keyword argument by allowing all keywords from the `Axes` constructor.\r\n\r\nIn this context, supplying a figure does not make sense since the axes should be bound to `self` or the current figure respecively.\r\n\r\nI'm daring to go without a deprecation since this anyway did only work so far if the user supplied the same figure that would have been used without the parameter. Please let me know if that's too bold and we should deprecate this first.",
      "issue_id": 14011,
      "issue_title": "TypeError on plt.subplot(figure=plt.figure())",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\nWhen calling `plt.subplot` with a `figure` argument, we get:\r\n```\r\nTypeError: process_projection_requirements() got multiple values for argument 'figure'\r\n```\r\nThe argument 'figure' is specified in the [doc](https://matplotlib.org/api/_as_gen/matplotlib.pyplot.subplot.html)\r\n\r\nThe function `plt.subplot` uses `plt.gcf()` to get the current figure. It should check whether a figure was provided, eg. `fig = kwargs.pop('figure') if kwargs.get('figure') else plt.gcf()`\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nplt.subplot(figure=plt.figure())\r\n# OR EVEN\r\nplt.subplot(figure=None)\r\n```\r\n\r\n**Actual outcome**\r\n```                              \r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n<ipython-input-30-bdfa817205a7> in <module>\r\n----> 1 plt.subplot(figure=None)\r\n\r\n~/bin/anaconda3/lib/python3.7/site-packages/matplotlib/pyplot.py in subplot(*args, **kwargs)\r\n   1082 \r\n   1083     fig = gcf()\r\n-> 1084     a = fig.add_subplot(*args, **kwargs)\r\n   1085     bbox = a.bbox\r\n   1086     byebye = []\r\n\r\n~/bin/anaconda3/lib/python3.7/site-packages/matplotlib/figure.py in add_subplot(self, *args, **kwargs)\r\n   1347         else:\r\n   1348             projection_class, kwargs, key = process_projection_requirements(\r\n-> 1349                 self, *args, **kwargs)\r\n   1350 \r\n   1351             # try to find the axes with this key in the stack\r\n\r\nTypeError: process_projection_requirements() got multiple values for argument 'figure'\r\n```\r\n\r\n**Matplotlib version**\r\n  * Operating system: Linux\r\n  * Matplotlib version: 3.0.2 (with conda and with pip)\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: tested on 3.7.2 and 3.5\r\n\r\n",
      "issue_closed_at": "2019-05-28T22:25:29Z",
      "base_commit": "559925e3ec43a5cbe1697a4496482d38d8489f68",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "add_subplot",
          "class_name": "Figure",
          "code": "def add_subplot(self, *args, **kwargs):\n        \"\"\"\n        Add an `~.axes.Axes` to the figure as part of a subplot arrangement.\n\n        Call signatures::\n\n           add_subplot(nrows, ncols, index, **kwargs)\n           add_subplot(pos, **kwargs)\n           add_subplot(ax)\n           add_subplot()\n\n        Parameters\n        ----------\n        *args\n            Either a 3-digit integer or three separate integers\n            describing the position of the subplot. If the three\n            integers are *nrows*, *ncols*, and *index* in order, the\n            subplot will take the *index* position on a grid with *nrows*\n            rows and *ncols* columns. *index* starts at 1 in the upper left\n            corner and increases to the right.\n\n            *pos* is a three digit integer, where the first digit is the\n            number of rows, the second the number of columns, and the third\n            the index of the subplot. i.e. fig.add_subplot(235) is the same as\n            fig.add_subplot(2, 3, 5). Note that all integers must be less than\n            10 for this form to work.\n\n            If no positional arguments are passed, defaults to (1, 1, 1).\n\n        projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n            The projection type of the subplot (`~.axes.Axes`). *str* is the\n            name of a custom projection, see `~matplotlib.projections`. The\n            default None results in a 'rectilinear' projection.\n\n        polar : boolean, optional\n            If True, equivalent to projection='polar'.\n\n        sharex, sharey : `~.axes.Axes`, optional\n            Share the x or y `~matplotlib.axis` with sharex and/or sharey.\n            The axis will have the same limits, ticks, and scale as the axis\n            of the shared axes.\n\n        label : str\n            A label for the returned axes.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            This method also takes the keyword arguments for\n            the returned axes base class. The keyword arguments for the\n            rectilinear base class `~.axes.Axes` can be found in\n            the following table but there might also be other keyword\n            arguments if another projection is used.\n            %(Axes)s\n\n        Returns\n        -------\n        axes : an `.axes.SubplotBase` subclass of `~.axes.Axes` (or a \\\n               subclass of `~.axes.Axes`)\n\n            The axes of the subplot. The returned axes base class depends on\n            the projection used. It is `~.axes.Axes` if rectilinear projection\n            are used and `.projections.polar.PolarAxes` if polar projection\n            are used. The returned axes is then a subplot subclass of the\n            base class.\n\n        Notes\n        -----\n        If the figure already has a subplot with key (*args*,\n        *kwargs*) then it will simply make that subplot current and\n        return it.  This behavior is deprecated. Meanwhile, if you do\n        not want this behavior (i.e., you want to force the creation of a\n        new subplot), you must use a unique set of args and kwargs.  The axes\n        *label* attribute has been exposed for this purpose: if you want\n        two subplots that are otherwise identical to be added to the figure,\n        make sure you give them unique labels.\n\n        In rare circumstances, `.add_subplot` may be called with a single\n        argument, a subplot axes instance already created in the\n        present figure but not in the figure's list of axes.\n\n        See Also\n        --------\n        .Figure.add_axes\n        .pyplot.subplot\n        .pyplot.axes\n        .Figure.subplots\n        .pyplot.subplots\n\n        Examples\n        --------\n        ::\n\n            fig = plt.figure()\n            fig.add_subplot(221)\n\n            # equivalent but more general\n            ax1 = fig.add_subplot(2, 2, 1)\n\n            # add a subplot with no frame\n            ax2 = fig.add_subplot(222, frameon=False)\n\n            # add a polar subplot\n            fig.add_subplot(223, projection='polar')\n\n            # add a red subplot that share the x-axis with ax1\n            fig.add_subplot(224, sharex=ax1, facecolor='red')\n\n            #delete x2 from the figure\n            fig.delaxes(ax2)\n\n            #add x2 to the figure again\n            fig.add_subplot(ax2)\n        \"\"\"\n        if not len(args):\n            args = (1, 1, 1)\n\n        if len(args) == 1 and isinstance(args[0], Integral):\n            if not 100 <= args[0] <= 999:\n                raise ValueError(\"Integer subplot specification must be a \"\n                                 \"three-digit number, not {}\".format(args[0]))\n            args = tuple(map(int, str(args[0])))\n\n        if isinstance(args[0], SubplotBase):\n\n            a = args[0]\n            if a.get_figure() is not self:\n                raise ValueError(\n                    \"The Subplot must have been created in the present figure\")\n            # make a key for the subplot (which includes the axes object id\n            # in the hash)\n            key = self._make_key(*args, **kwargs)\n        else:\n            projection_class, kwargs, key = \\\n                self._process_projection_requirements(*args, **kwargs)\n\n            # try to find the axes with this key in the stack\n            ax = self._axstack.get(key)\n\n            if ax is not None:\n                if isinstance(ax, projection_class):\n                    # the axes already existed, so set it as active & return\n                    self.sca(ax)\n                    return ax\n                else:\n                    # Undocumented convenience behavior:\n                    # subplot(111); subplot(111, projection='polar')\n                    # will replace the first with the second.\n                    # Without this, add_subplot would be simpler and\n                    # more similar to add_axes.\n                    self._axstack.remove(ax)\n\n            a = subplot_class_factory(projection_class)(self, *args, **kwargs)\n\n        return self._add_axes_internal(key, a)"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "add_subplot",
          "class_name": "Figure",
          "code": "def add_subplot(self, *args, **kwargs):\n        \"\"\"\n        Add an `~.axes.Axes` to the figure as part of a subplot arrangement.\n\n        Call signatures::\n\n           add_subplot(nrows, ncols, index, **kwargs)\n           add_subplot(pos, **kwargs)\n           add_subplot(ax)\n           add_subplot()\n\n        Parameters\n        ----------\n        *args\n            Either a 3-digit integer or three separate integers\n            describing the position of the subplot. If the three\n            integers are *nrows*, *ncols*, and *index* in order, the\n            subplot will take the *index* position on a grid with *nrows*\n            rows and *ncols* columns. *index* starts at 1 in the upper left\n            corner and increases to the right.\n\n            *pos* is a three digit integer, where the first digit is the\n            number of rows, the second the number of columns, and the third\n            the index of the subplot. i.e. fig.add_subplot(235) is the same as\n            fig.add_subplot(2, 3, 5). Note that all integers must be less than\n            10 for this form to work.\n\n            If no positional arguments are passed, defaults to (1, 1, 1).\n\n        projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n            The projection type of the subplot (`~.axes.Axes`). *str* is the\n            name of a custom projection, see `~matplotlib.projections`. The\n            default None results in a 'rectilinear' projection.\n\n        polar : boolean, optional\n            If True, equivalent to projection='polar'.\n\n        sharex, sharey : `~.axes.Axes`, optional\n            Share the x or y `~matplotlib.axis` with sharex and/or sharey.\n            The axis will have the same limits, ticks, and scale as the axis\n            of the shared axes.\n\n        label : str\n            A label for the returned axes.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            This method also takes the keyword arguments for\n            the returned axes base class. The keyword arguments for the\n            rectilinear base class `~.axes.Axes` can be found in\n            the following table but there might also be other keyword\n            arguments if another projection is used.\n            %(Axes)s\n\n        Returns\n        -------\n        axes : an `.axes.SubplotBase` subclass of `~.axes.Axes` (or a \\\n               subclass of `~.axes.Axes`)\n\n            The axes of the subplot. The returned axes base class depends on\n            the projection used. It is `~.axes.Axes` if rectilinear projection\n            are used and `.projections.polar.PolarAxes` if polar projection\n            are used. The returned axes is then a subplot subclass of the\n            base class.\n\n        Notes\n        -----\n        If the figure already has a subplot with key (*args*,\n        *kwargs*) then it will simply make that subplot current and\n        return it.  This behavior is deprecated. Meanwhile, if you do\n        not want this behavior (i.e., you want to force the creation of a\n        new subplot), you must use a unique set of args and kwargs.  The axes\n        *label* attribute has been exposed for this purpose: if you want\n        two subplots that are otherwise identical to be added to the figure,\n        make sure you give them unique labels.\n\n        In rare circumstances, `.add_subplot` may be called with a single\n        argument, a subplot axes instance already created in the\n        present figure but not in the figure's list of axes.\n\n        See Also\n        --------\n        .Figure.add_axes\n        .pyplot.subplot\n        .pyplot.axes\n        .Figure.subplots\n        .pyplot.subplots\n\n        Examples\n        --------\n        ::\n\n            fig = plt.figure()\n            fig.add_subplot(221)\n\n            # equivalent but more general\n            ax1 = fig.add_subplot(2, 2, 1)\n\n            # add a subplot with no frame\n            ax2 = fig.add_subplot(222, frameon=False)\n\n            # add a polar subplot\n            fig.add_subplot(223, projection='polar')\n\n            # add a red subplot that share the x-axis with ax1\n            fig.add_subplot(224, sharex=ax1, facecolor='red')\n\n            #delete x2 from the figure\n            fig.delaxes(ax2)\n\n            #add x2 to the figure again\n            fig.add_subplot(ax2)\n        \"\"\"\n        if not len(args):\n            args = (1, 1, 1)\n\n        if len(args) == 1 and isinstance(args[0], Integral):\n            if not 100 <= args[0] <= 999:\n                raise ValueError(\"Integer subplot specification must be a \"\n                                 \"three-digit number, not {}\".format(args[0]))\n            args = tuple(map(int, str(args[0])))\n\n        if isinstance(args[0], SubplotBase):\n\n            a = args[0]\n            if a.get_figure() is not self:\n                raise ValueError(\n                    \"The Subplot must have been created in the present figure\")\n            # make a key for the subplot (which includes the axes object id\n            # in the hash)\n            key = self._make_key(*args, **kwargs)\n        else:\n            projection_class, kwargs, key = \\\n                self._process_projection_requirements(*args, **kwargs)\n\n            # try to find the axes with this key in the stack\n            ax = self._axstack.get(key)\n\n            if ax is not None:\n                if isinstance(ax, projection_class):\n                    # the axes already existed, so set it as active & return\n                    self.sca(ax)\n                    return ax\n                else:\n                    # Undocumented convenience behavior:\n                    # subplot(111); subplot(111, projection='polar')\n                    # will replace the first with the second.\n                    # Without this, add_subplot would be simpler and\n                    # more similar to add_axes.\n                    self._axstack.remove(ax)\n\n            a = subplot_class_factory(projection_class)(self, *args, **kwargs)\n\n        return self._add_axes_internal(key, a)"
        },
        {
          "file": "lib/matplotlib/pyplot.py",
          "type": "function",
          "name": "subplot",
          "class_name": null,
          "code": "def subplot(*args, **kwargs):\n    \"\"\"\n    Add a subplot to the current figure.\n\n    Wrapper of `.Figure.add_subplot` with a difference in behavior\n    explained in the notes section.\n\n    Call signatures::\n\n       subplot(nrows, ncols, index, **kwargs)\n       subplot(pos, **kwargs)\n       subplot(ax)\n\n    Parameters\n    ----------\n    *args\n        Either a 3-digit integer or three separate integers\n        describing the position of the subplot. If the three\n        integers are *nrows*, *ncols*, and *index* in order, the\n        subplot will take the *index* position on a grid with *nrows*\n        rows and *ncols* columns. *index* starts at 1 in the upper left\n        corner and increases to the right.\n\n        *pos* is a three digit integer, where the first digit is the\n        number of rows, the second the number of columns, and the third\n        the index of the subplot. i.e. fig.add_subplot(235) is the same as\n        fig.add_subplot(2, 3, 5). Note that all integers must be less than\n        10 for this form to work.\n\n    projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n        The projection type of the subplot (`~.axes.Axes`). *str* is the name\n        of a costum projection, see `~matplotlib.projections`. The default\n        None results in a 'rectilinear' projection.\n\n    polar : boolean, optional\n        If True, equivalent to projection='polar'.\n\n    sharex, sharey : `~.axes.Axes`, optional\n        Share the x or y `~matplotlib.axis` with sharex and/or sharey. The\n        axis will have the same limits, ticks, and scale as the axis of the\n        shared axes.\n\n    label : str\n        A label for the returned axes.\n\n    Other Parameters\n    ----------------\n    **kwargs\n        This method also takes the keyword arguments for\n        the returned axes base class. The keyword arguments for the\n        rectilinear base class `~.axes.Axes` can be found in\n        the following table but there might also be other keyword\n        arguments if another projection is used.\n        %(Axes)s\n\n    Returns\n    -------\n    axes : an `.axes.SubplotBase` subclass of `~.axes.Axes` (or a subclass \\\n    of `~.axes.Axes`)\n\n        The axes of the subplot. The returned axes base class depends on\n        the projection used. It is `~.axes.Axes` if rectilinear projection\n        are used and `.projections.polar.PolarAxes` if polar projection\n        are used. The returned axes is then a subplot subclass of the\n        base class.\n\n    Notes\n    -----\n    Creating a subplot will delete any pre-existing subplot that overlaps\n    with it beyond sharing a boundary::\n\n        import matplotlib.pyplot as plt\n        # plot a line, implicitly creating a subplot(111)\n        plt.plot([1,2,3])\n        # now create a subplot which represents the top plot of a grid\n        # with 2 rows and 1 column. Since this subplot will overlap the\n        # first, the plot (and its axes) previously created, will be removed\n        plt.subplot(211)\n\n    If you do not want this behavior, use the `.Figure.add_subplot` method\n    or the `.pyplot.axes` function instead.\n\n    If the figure already has a subplot with key (*args*,\n    *kwargs*) then it will simply make that subplot current and\n    return it.  This behavior is deprecated. Meanwhile, if you do\n    not want this behavior (i.e., you want to force the creation of a\n    new subplot), you must use a unique set of args and kwargs.  The axes\n    *label* attribute has been exposed for this purpose: if you want\n    two subplots that are otherwise identical to be added to the figure,\n    make sure you give them unique labels.\n\n    In rare circumstances, `.add_subplot` may be called with a single\n    argument, a subplot axes instance already created in the\n    present figure but not in the figure's list of axes.\n\n    See Also\n    --------\n    .Figure.add_subplot\n    .pyplot.subplots\n    .pyplot.axes\n    .Figure.subplots\n\n    Examples\n    --------\n    ::\n\n        plt.subplot(221)\n\n        # equivalent but more general\n        ax1=plt.subplot(2, 2, 1)\n\n        # add a subplot with no frame\n        ax2=plt.subplot(222, frameon=False)\n\n        # add a polar subplot\n        plt.subplot(223, projection='polar')\n\n        # add a red subplot that shares the x-axis with ax1\n        plt.subplot(224, sharex=ax1, facecolor='red')\n\n        # delete ax2 from the figure\n        plt.delaxes(ax2)\n\n        # add ax2 to the figure again\n        plt.subplot(ax2)\n        \"\"\"\n\n    # if subplot called without arguments, create subplot(1,1,1)\n    if len(args) == 0:\n        args = (1, 1, 1)\n\n    # This check was added because it is very easy to type\n    # subplot(1, 2, False) when subplots(1, 2, False) was intended\n    # (sharex=False, that is). In most cases, no error will\n    # ever occur, but mysterious behavior can result because what was\n    # intended to be the sharex argument is instead treated as a\n    # subplot index for subplot()\n    if len(args) >= 3 and isinstance(args[2], bool):\n        cbook._warn_external(\"The subplot index argument to subplot() appears \"\n                             \"to be a boolean. Did you intend to use \"\n                             \"subplots()?\")\n\n    fig = gcf()\n    a = fig.add_subplot(*args, **kwargs)\n    bbox = a.bbox\n    byebye = []\n    for other in fig.axes:\n        if other == a:\n            continue\n        if bbox.fully_overlaps(other.bbox):\n            byebye.append(other)\n    for ax in byebye:\n        delaxes(ax)\n\n    return a"
        }
      ]
    },
    {
      "pr_number": 14719,
      "pr_title": "Make Qt navtoolbar more robust against removal of either pan or zoom.",
      "pr_body": "... by using a QButtonGroup to ensure that a button gets unchecked when\r\nthe other gets checked.\r\n\r\nCloses #12893.\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/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": 12893,
      "issue_title": "[PyQt] NavigationToolbar2QT : Error when removing tools",
      "issue_body": "### 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\n**Code**\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\nclass FigureWidget(QtWidgets.QWidget):\r\n    \"\"\"Figure widget used by different views\r\n\r\n    Parameters\r\n    ----------\r\n    parent : Qt parent\r\n       Qt parent object\r\n    canvas : FigureCanvas\r\n       the figure canvas class\r\n    exclude_toolbar_items : tuple\r\n       elements to exclude from the toolbar\r\n    \"\"\"\r\n    def __init__(self, parent=None, canvas=None, exclude_toolbar_items=()):\r\n        super().__init__(parent)\r\n\r\n        self.canvas = canvas\r\n        self.layout = QtWidgets.QVBoxLayout()\r\n        self.layout.addWidget(self.canvas)\r\n\r\n        if isinstance(canvas, FigureCanvas):\r\n          class _SimpleNavigationToolbar(NavigationToolbar):\r\n            # only display the buttons we need\r\n            toolitems = [t for t in NavigationToolbar.toolitems if\r\n                        t[0] not in exclude_toolbar_items]\r\n          self.toolbar = _SimpleNavigationToolbar(self.canvas, self.canvas)\r\n\r\n          self.layout.addWidget(self.toolbar)\r\n      else:\r\n          self.toolbar = None\r\n\r\n      self.setLayout(self.layout)\r\n```\r\n\r\n**Error example**\r\n\r\nWhen **exclude_toolbar_items=('Pan',))**, we get the following result on our window :\r\n\r\n![toolbar_error](https://user-images.githubusercontent.com/39213808/49034667-e5ff8100-f1b2-11e8-80d2-25ad2ca06d02.png)\r\n\r\nBut if we click on the **Zoom** button (_encircled in red_), we get the error message on the log and written just below as if the **Pan** button stills exists.\r\n\r\n```\r\n2018-11-26 19:35:01.566 [ERROR]\t[utils.py:28]\t\tUncaught exception\r\nTraceback (most recent call last):\r\n  File \"C:\\Users\\Renaud Gautier\\Anaconda3\\envs\\some_env\\lib\\site-packages\\matplotlib\\backends\\backend_qt5.py\", line 803, in zoom\r\n    self._update_buttons_checked()\r\n  File \"C:\\Users\\Renaud Gautier\\Anaconda3\\envs\\some_env\\lib\\site-packages\\matplotlib\\backends\\backend_qt5.py\", line 794, in _update_buttons_checked\r\n    self._actions['pan'].setChecked(self._active == 'PAN')\r\nKeyError: 'pan'\r\n```\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating system: Windows 10\r\n  * Matplotlib version: 3.0.2\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.1\r\n  * Other libraries: PyQt 5 (PySide)\r\n\r\n\r\n",
      "issue_closed_at": "2019-07-17T18:39:21Z",
      "base_commit": "9f1c7306d23cd9eac858897b3c873c8314e300cf",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "_icon",
          "class_name": "ToolbarQt",
          "code": "def _icon(self, name):\n        pm = QtGui.QPixmap(name)\n        if hasattr(pm, 'setDevicePixelRatio'):\n            pm.setDevicePixelRatio(self.toolmanager.canvas._dpi_ratio)\n        return QtGui.QIcon(pm)"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "_init_toolbar",
          "class_name": "NavigationToolbar2QT",
          "code": "def _init_toolbar(self):\n        self.basedir = str(cbook._get_data_path('images'))\n\n        for text, tooltip_text, image_file, callback in self.toolitems:\n            if text is None:\n                self.addSeparator()\n            else:\n                a = self.addAction(self._icon(image_file + '.png'),\n                                   text, getattr(self, callback))\n                self._actions[callback] = a\n                if callback in ['zoom', 'pan']:\n                    a.setCheckable(True)\n                if tooltip_text is not None:\n                    a.setToolTip(tooltip_text)\n                if text == 'Subplots':\n                    a = self.addAction(self._icon(\"qt4_editor_options.png\"),\n                                       'Customize', self.edit_parameters)\n                    a.setToolTip('Edit axis, curve and image parameters')\n\n        # Add the x,y location widget at the right side of the toolbar\n        # The stretch factor is 1 which means any resizing of the toolbar\n        # will resize this label instead of the buttons.\n        if self.coordinates:\n            self.locLabel = QtWidgets.QLabel(\"\", self)\n            self.locLabel.setAlignment(\n                    QtCore.Qt.AlignRight | QtCore.Qt.AlignTop)\n            self.locLabel.setSizePolicy(\n                QtWidgets.QSizePolicy(QtWidgets.QSizePolicy.Expanding,\n                                      QtWidgets.QSizePolicy.Ignored))\n            labelAction = self.addWidget(self.locLabel)\n            labelAction.setVisible(True)\n\n        # Esthetic adjustments - we need to set these explicitly in PyQt5\n        # otherwise the layout looks different - but we don't want to set it if\n        # not using HiDPI icons otherwise they look worse than before.\n        if is_pyqt5() and self.canvas._dpi_ratio > 1:\n            self.setIconSize(QtCore.QSize(24, 24))\n            self.layout().setSpacing(12)"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "edit_parameters",
          "class_name": "NavigationToolbar2QT",
          "code": "def edit_parameters(self):\n        axes = self.canvas.figure.get_axes()\n        if not axes:\n            QtWidgets.QMessageBox.warning(\n                self.parent, \"Error\", \"There are no axes to edit.\")\n            return\n        elif len(axes) == 1:\n            ax, = axes\n        else:\n            titles = [\n                ax.get_label() or\n                ax.get_title() or\n                \" - \".join(filter(None, [ax.get_xlabel(), ax.get_ylabel()])) or\n                f\"<anonymous {type(ax).__name__}>\"\n                for ax in axes]\n            duplicate_titles = [\n                title for title in titles if titles.count(title) > 1]\n            for i, ax in enumerate(axes):\n                if titles[i] in duplicate_titles:\n                    titles[i] += f\" (id: {id(ax):#x})\"  # Deduplicate titles.\n            item, ok = QtWidgets.QInputDialog.getItem(\n                self.parent, 'Customize', 'Select axes:', titles, 0, False)\n            if not ok:\n                return\n            ax = axes[titles.index(item)]\n        figureoptions.figure_edit(ax, self)"
        }
      ]
    },
    {
      "pr_number": 19964,
      "pr_title": "FIX: add subplot_mosaic axes in the order the user gave them to us",
      "pr_body": "## PR Summary\r\n\r\nThe order the Axes are added to the Figure (and hence the order they are in\r\nthe returned dictionary and fig.axes) is now based on the first time we see the\r\nkey if the layout were recursively unraveled as a c-style array.\r\n\r\nCloses #19736\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- [x] Has pytest style unit tests (and `pytest` passes).\r\n- [x] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [x] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [ ] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\r\n\r\nI don't think this needs an API change note as the order was previously unreliable.",
      "issue_id": 19736,
      "issue_title": "subplot_mosaic axes are not added in consistent order",
      "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\nThe axes of a subplot_mosaic show up in a random order in `fig.axes` (likely due to the use of a `set` for uniquification in `_identify_keys_and_nested`).\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```sh\r\nfor _ in $(seq 10); do python -c 'from pylab import *; fig, axs = subplot_mosaic(\"ab\"); print(fig.axes.index(axs[\"a\"]))'; done\r\n```\r\n\r\n**Actual outcome**\r\n\r\n<!--The output produced by the above code, which may be a screenshot, console output, etc.-->\r\n\r\n```\r\n1\r\n0\r\n1\r\n1\r\n1\r\n0\r\n0\r\n0\r\n0\r\n1\r\n```\r\n\r\n**Expected outcome**\r\n\r\nAxes should be added in a consistent order.  I guess a reasonable one would be as if iterating the spec in C order (dropping duplicates).\r\n\r\nNot release critical (especially as the order was not fixed, so fixing an order is not a backcompat break), but would be nice to get this sorted out before the API moves out of being experimental.\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__)`): head\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): any\r\n  * Python version: 39\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---\r\n\r\nNote: the simple solution of replacing `unique_ids = set()` by `unique_ids = cbook._OrderedSet()` is good enough for the non-nested case, but doesn't handle nested layouts such as `subplot_mosaic([[\"a\", [[\"b1\", \"b2\"], [\"b3\", \"b4\"]]], [\"c\", \"d\"]])` because currently the nested submosaic is always added after all the non-nested axes.",
      "issue_closed_at": "2021-04-16T21:43:45Z",
      "base_commit": "a051169c1b51a36c270b7b696ec86fae8f2f23e8",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_identify_keys_and_nested",
          "class_name": "FigureBase",
          "code": "def _identify_keys_and_nested(layout):\n            \"\"\"\n            Given a 2D object array, identify unique IDs and nested layouts\n\n            Parameters\n            ----------\n            layout : 2D numpy object array\n\n            Returns\n            -------\n            unique_ids : set\n                The unique non-sub layout entries in this layout\n            nested : dict[tuple[int, int]], 2D object array\n            \"\"\"\n            unique_ids = set()\n            nested = {}\n            for j, row in enumerate(layout):\n                for k, v in enumerate(row):\n                    if v == empty_sentinel:\n                        continue\n                    elif not cbook.is_scalar_or_string(v):\n                        nested[(j, k)] = _make_array(v)\n                    else:\n                        unique_ids.add(v)\n\n            return unique_ids, nested"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_identify_keys_and_nested",
          "class_name": "FigureBase",
          "code": "def _identify_keys_and_nested(layout):\n            \"\"\"\n            Given a 2D object array, identify unique IDs and nested layouts\n\n            Parameters\n            ----------\n            layout : 2D numpy object array\n\n            Returns\n            -------\n            unique_ids : set\n                The unique non-sub layout entries in this layout\n            nested : dict[tuple[int, int]], 2D object array\n            \"\"\"\n            unique_ids = set()\n            nested = {}\n            for j, row in enumerate(layout):\n                for k, v in enumerate(row):\n                    if v == empty_sentinel:\n                        continue\n                    elif not cbook.is_scalar_or_string(v):\n                        nested[(j, k)] = _make_array(v)\n                    else:\n                        unique_ids.add(v)\n\n            return unique_ids, nested"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_do_layout",
          "class_name": "FigureBase",
          "code": "def _do_layout(gs, layout, unique_ids, nested):\n            \"\"\"\n            Recursively do the layout.\n\n            Parameters\n            ----------\n            gs : GridSpec\n            layout : 2D object array\n                The input converted to a 2D numpy array for this level.\n            unique_ids : set\n                The identified scalar labels at this level of nesting.\n            nested : dict[tuple[int, int]], 2D object array\n                The identified nested layouts, if any.\n\n            Returns\n            -------\n            dict[label, Axes]\n                A flat dict of all of the Axes created.\n            \"\"\"\n            rows, cols = layout.shape\n            output = dict()\n\n            # create the Axes at this level of nesting\n            for name in unique_ids:\n                indx = np.argwhere(layout == name)\n                start_row, start_col = np.min(indx, axis=0)\n                end_row, end_col = np.max(indx, axis=0) + 1\n                slc = (slice(start_row, end_row), slice(start_col, end_col))\n\n                if (layout[slc] != name).any():\n                    raise ValueError(\n                        f\"While trying to layout\\n{layout!r}\\n\"\n                        f\"we found that the label {name!r} specifies a \"\n                        \"non-rectangular or non-contiguous area.\")\n\n                ax = self.add_subplot(\n                    gs[slc], **{'label': str(name), **subplot_kw}\n                )\n                output[name] = ax\n\n            # do any sub-layouts\n            for (j, k), nested_layout in nested.items():\n                rows, cols = nested_layout.shape\n                nested_output = _do_layout(\n                    gs[j, k].subgridspec(rows, cols, **gridspec_kw),\n                    nested_layout,\n                    *_identify_keys_and_nested(nested_layout)\n                )\n                overlap = set(output) & set(nested_output)\n                if overlap:\n                    raise ValueError(f\"There are duplicate keys {overlap} \"\n                                     f\"between the outer layout\\n{layout!r}\\n\"\n                                     f\"and the nested layout\\n{nested_layout}\")\n                output.update(nested_output)\n            return output"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "_do_layout",
          "class_name": "FigureBase",
          "code": "def _do_layout(gs, layout, unique_ids, nested):\n            \"\"\"\n            Recursively do the layout.\n\n            Parameters\n            ----------\n            gs : GridSpec\n            layout : 2D object array\n                The input converted to a 2D numpy array for this level.\n            unique_ids : set\n                The identified scalar labels at this level of nesting.\n            nested : dict[tuple[int, int]], 2D object array\n                The identified nested layouts, if any.\n\n            Returns\n            -------\n            dict[label, Axes]\n                A flat dict of all of the Axes created.\n            \"\"\"\n            rows, cols = layout.shape\n            output = dict()\n\n            # create the Axes at this level of nesting\n            for name in unique_ids:\n                indx = np.argwhere(layout == name)\n                start_row, start_col = np.min(indx, axis=0)\n                end_row, end_col = np.max(indx, axis=0) + 1\n                slc = (slice(start_row, end_row), slice(start_col, end_col))\n\n                if (layout[slc] != name).any():\n                    raise ValueError(\n                        f\"While trying to layout\\n{layout!r}\\n\"\n                        f\"we found that the label {name!r} specifies a \"\n                        \"non-rectangular or non-contiguous area.\")\n\n                ax = self.add_subplot(\n                    gs[slc], **{'label': str(name), **subplot_kw}\n                )\n                output[name] = ax\n\n            # do any sub-layouts\n            for (j, k), nested_layout in nested.items():\n                rows, cols = nested_layout.shape\n                nested_output = _do_layout(\n                    gs[j, k].subgridspec(rows, cols, **gridspec_kw),\n                    nested_layout,\n                    *_identify_keys_and_nested(nested_layout)\n                )\n                overlap = set(output) & set(nested_output)\n                if overlap:\n                    raise ValueError(f\"There are duplicate keys {overlap} \"\n                                     f\"between the outer layout\\n{layout!r}\\n\"\n                                     f\"and the nested layout\\n{nested_layout}\")\n                output.update(nested_output)\n            return output"
        }
      ]
    },
    {
      "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)"
        }
      ]
    },
    {
      "pr_number": 17142,
      "pr_title": "BUGFIX: conditional for add_axes arg deprecation",
      "pr_body": "## PR Summary\r\n\r\nFixes #17114.\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": 17114,
      "issue_title": "`add_axes` shows deprecation warning when called with only `kwarg`s",
      "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\nCalling `add_axes` without args is deprecated. However, if a kwarg is passed, the deprecation warning still shows.\r\n\r\n\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug.\r\nPlease make sure to minimize the number of dependencies required, and provide\r\nany necessary plotted data.\r\nAvoid using threads, as Matplotlib is (explicitly) not thread-safe.-->\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nfig = plt.figure()\r\nax = fig.add_axes(rect=[0, 0, 1, 1])\r\n```\r\n\r\n**Actual outcome**\r\n\r\n<!--The output produced by the above code, which may be a screenshot, console output, etc.-->\r\n\r\n```\r\n<ipython-input-128-1ef4234f1c1f>:3: MatplotlibDeprecationWarning: Calling add_axes() without argument is deprecated. You may want to use add_subplot() instead.\r\n  ax = fig.add_axes(rect=[0, 0, 1, 1])\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 warning should be shown.\r\n\r\n**Discussion**\r\n\r\n`figure.add_axes` currently starts with\r\n\r\n```\r\n        if not len(args):\r\n            cbook.warn_deprecated(\r\n                \"3.3\",\r\n                message=\"Calling add_axes() without argument is \"\r\n                \"deprecated. You may want to use add_subplot() \"\r\n                \"instead.\")\r\n            return\r\n```\r\n\r\nBecause the function forwards `kwargs`, the intended behavior is a bit ambiguous to me, but I'm assuming from the docstring that the intention is to force the user to pass `rect` specifically? \r\n\r\n```\r\nif not len(args) and 'rect' not in kwargs:\r\n    ...\r\n```\r\n\r\nI can make this a PR if necessary, but if I'm not sure if we continue to update 3.2...And I don't know what the deprecation schedule even looks like for removing the \"ability to use no args\", so I don't know if this will even be an issue after 3.3....just wanted to point it out.\r\n\r\nFeel free to close if this is a non-issue.",
      "issue_closed_at": "2020-04-15T08:44:02Z",
      "base_commit": "e55e79b97d0b3ff772a4f7aaa0fb15193d84300c",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "add_axes",
          "class_name": "Figure",
          "code": "def add_axes(self, *args, **kwargs):\n        \"\"\"\n        Add an axes to the figure.\n\n        Call signatures::\n\n            add_axes(rect, projection=None, polar=False, **kwargs)\n            add_axes(ax)\n\n        Parameters\n        ----------\n        rect : sequence of float\n            The dimensions [left, bottom, width, height] of the new axes. All\n            quantities are in fractions of figure width and height.\n\n        projection : {None, 'aitoff', 'hammer', 'lambert', 'mollweide', \\\n'polar', 'rectilinear', str}, optional\n            The projection type of the `~.axes.Axes`. *str* is the name of\n            a custom projection, see `~matplotlib.projections`. The default\n            None results in a 'rectilinear' projection.\n\n        polar : bool, default: False\n            If True, equivalent to projection='polar'.\n\n        sharex, sharey : `~.axes.Axes`, optional\n            Share the x or y `~matplotlib.axis` with sharex and/or sharey.\n            The axis will have the same limits, ticks, and scale as the axis\n            of the shared axes.\n\n        label : str\n            A label for the returned axes.\n\n        Returns\n        -------\n        `~.axes.Axes` (or a subclass of `~.axes.Axes`)\n            The returned axes class depends on the projection used. It is\n            `~.axes.Axes` if rectilinear projection are used and\n            `.projections.polar.PolarAxes` if polar projection\n            are used.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            This method also takes the keyword arguments for\n            the returned axes class. The keyword arguments for the\n            rectilinear axes class `~.axes.Axes` can be found in\n            the following table but there might also be other keyword\n            arguments if another projection is used, see the actual axes\n            class.\n\n            %(Axes)s\n\n        Notes\n        -----\n        If the figure already has an axes with key (*args*,\n        *kwargs*) then it will simply make that axes current and\n        return it.  This behavior is deprecated. Meanwhile, if you do\n        not want this behavior (i.e., you want to force the creation of a\n        new axes), you must use a unique set of args and kwargs.  The axes\n        *label* attribute has been exposed for this purpose: if you want\n        two axes that are otherwise identical to be added to the figure,\n        make sure you give them unique labels.\n\n        In rare circumstances, `.add_axes` may be called with a single\n        argument, a axes instance already created in the present figure but\n        not in the figure's list of axes.\n\n        See Also\n        --------\n        .Figure.add_subplot\n        .pyplot.subplot\n        .pyplot.axes\n        .Figure.subplots\n        .pyplot.subplots\n\n        Examples\n        --------\n        Some simple examples::\n\n            rect = l, b, w, h\n            fig = plt.figure()\n            fig.add_axes(rect, label=label1)\n            fig.add_axes(rect, label=label2)\n            fig.add_axes(rect, frameon=False, facecolor='g')\n            fig.add_axes(rect, polar=True)\n            ax = fig.add_axes(rect, projection='polar')\n            fig.delaxes(ax)\n            fig.add_axes(ax)\n        \"\"\"\n\n        if not len(args):\n            cbook.warn_deprecated(\n                \"3.3\",\n                message=\"Calling add_axes() without argument is \"\n                \"deprecated. You may want to use add_subplot() \"\n                \"instead.\")\n            return\n\n        # shortcut the projection \"key\" modifications later on, if an axes\n        # with the exact args/kwargs exists, return it immediately.\n        key = self._make_key(*args, **kwargs)\n        ax = self._axstack.get(key)\n        if ax is not None:\n            self.sca(ax)\n            return ax\n\n        if isinstance(args[0], Axes):\n            a = args[0]\n            if a.get_figure() is not self:\n                raise ValueError(\n                    \"The Axes must have been created in the present figure\")\n        else:\n            rect = args[0]\n            if not np.isfinite(rect).all():\n                raise ValueError('all entries in rect must be finite '\n                                 'not {}'.format(rect))\n            projection_class, kwargs, key = \\\n                self._process_projection_requirements(*args, **kwargs)\n\n            # check that an axes of this type doesn't already exist, if it\n            # does, set it as active and return it\n            ax = self._axstack.get(key)\n            if isinstance(ax, projection_class):\n                self.sca(ax)\n                return ax\n\n            # create the new axes using the axes class given\n            a = projection_class(self, rect, **kwargs)\n\n        return self._add_axes_internal(key, a)"
        }
      ]
    }
  ]
}