{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful because it also involves issues concerning the handling of figures in Matplotlib—a fundamental aspect of the current bug. The error encountered in Candidate A involves improper function calls related to figures similar to the issue with `get_backend()` in the current bug report. Both bugs concern scenarios where figures become inconsistently managed, although they manifest differently in function calls. The structural similarity and shared components (e.g., `matplotlib.pyplot` functions) enhance its relevance for debugging the current issue regarding figure clearing in `Gcf`. Consequently, the insights gained from Candidate A regarding proper figure management could directly assist in resolving the current problem.",
  "instance_id": "matplotlib__matplotlib-23299",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-06-18T01:34:39Z",
  "problem_statement": "[Bug]: get_backend() clears figures from Gcf.figs if they were created under rc_context\n### Bug summary\r\n\r\ncalling `matplotlib.get_backend()` removes all figures from `Gcf` if the *first* figure in `Gcf.figs` was created in an `rc_context`.\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nfrom matplotlib import get_backend, rc_context\r\n\r\n# fig1 = plt.figure()  # <- UNCOMMENT THIS LINE AND IT WILL WORK\r\n# plt.ion()            # <- ALTERNATIVELY, UNCOMMENT THIS LINE AND IT WILL ALSO WORK\r\nwith rc_context():\r\n    fig2 = plt.figure()\r\nbefore = f'{id(plt._pylab_helpers.Gcf)} {plt._pylab_helpers.Gcf.figs!r}'\r\nget_backend()\r\nafter = f'{id(plt._pylab_helpers.Gcf)} {plt._pylab_helpers.Gcf.figs!r}'\r\n\r\nassert before == after, '\\n' + before + '\\n' + after\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n```\r\n---------------------------------------------------------------------------\r\nAssertionError                            Traceback (most recent call last)\r\n<ipython-input-1-fa4d099aa289> in <cell line: 11>()\r\n      9 after = f'{id(plt._pylab_helpers.Gcf)} {plt._pylab_helpers.Gcf.figs!r}'\r\n     10 \r\n---> 11 assert before == after, '\\n' + before + '\\n' + after\r\n     12 \r\n\r\nAssertionError: \r\n94453354309744 OrderedDict([(1, <matplotlib.backends.backend_qt.FigureManagerQT object at 0x7fb33e26c220>)])\r\n94453354309744 OrderedDict()\r\n```\r\n\r\n### Expected outcome\r\n\r\nThe figure should not be missing from `Gcf`.  Consequences of this are, e.g, `plt.close(fig2)` doesn't work because `Gcf.destroy_fig()` can't find it.\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\nXubuntu\r\n\r\n### Matplotlib Version\r\n\r\n3.5.2\r\n\r\n### Matplotlib Backend\r\n\r\nQtAgg\r\n\r\n### Python version\r\n\r\nPython 3.10.4\r\n\r\n### Jupyter version\r\n\r\nn/a\r\n\r\n### Installation\r\n\r\nconda\n",
  "patch": "diff --git a/lib/matplotlib/__init__.py b/lib/matplotlib/__init__.py\n--- a/lib/matplotlib/__init__.py\n+++ b/lib/matplotlib/__init__.py\n@@ -1059,6 +1059,8 @@ def rc_context(rc=None, fname=None):\n     \"\"\"\n     Return a context manager for temporarily changing rcParams.\n \n+    The :rc:`backend` will not be reset by the context manager.\n+\n     Parameters\n     ----------\n     rc : dict\n@@ -1087,7 +1089,8 @@ def rc_context(rc=None, fname=None):\n              plt.plot(x, y)  # uses 'print.rc'\n \n     \"\"\"\n-    orig = rcParams.copy()\n+    orig = dict(rcParams.copy())\n+    del orig['backend']\n     try:\n         if fname:\n             rc_file(fname)\n"
}