{
  "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",
  "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": 5324,
      "pr_title": "Fix #5302: Proper alpha-blending for jpeg",
      "pr_body": "Saving a JPEG with an alpha channel doesn't look right -- I think pillow just removes the alpha channel altogether and uses the rest of the RGB values as-is.\n\nThis blends RGBA images onto a white background before saving.\n\nThis is JPEG-specific since it's our only raster output file format that doesn't support alpha blending.  If we ever grow another (unlikely) this will need to be duplicated there.\n",
      "issue_id": 5302,
      "issue_title": "Pixelated fonts when plot saved as jpeg",
      "issue_body": "I have noticed that when you save figures as jpeg using the nbagg backend you will get pixelated fonts for labels and tickmarks.  It appears this has to do with giving PIL raw data with transparent regions when trying to save jpeg images.  This can be avoided with setting: rcParams['nbagg.transparent'] = False.\n\nIs this behavior desired?  Perhaps there is a way we can assign default values that will be called for when you have a transparency in a figure and you are trying to save it to a file format that doesn't support transparency?\n\nPlease see http://stackoverflow.com/questions/33283274/pixelated-fonts-when-plot-is-saved-as-jpeg for more information.\n",
      "issue_closed_at": "2015-10-28T22:45:18Z",
      "base_commit": "0e6ca435d132edd6c2274770de306d80e5a8f0a8",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_agg.py",
          "type": "line",
          "name": "line 38",
          "code": "from matplotlib.mathtext import MathTextParser\nfrom matplotlib.path import Path\nfrom matplotlib.transforms import Bbox, BboxBase\n\nfrom matplotlib.backends._backend_agg import RendererAgg as _RendererAgg\nfrom matplotlib import _png"
        },
        {
          "file": "lib/matplotlib/backends/backend_agg.py",
          "type": "function",
          "name": "print_to_buffer",
          "class_name": "FigureCanvasAgg",
          "code": "def print_to_buffer(self):\n        FigureCanvasAgg.draw(self)\n        renderer = self.get_renderer()\n        original_dpi = renderer.dpi\n        renderer.dpi = self.figure.dpi\n        result = (renderer._renderer.buffer_rgba(),\n                  (int(renderer.width), int(renderer.height)))\n        renderer.dpi = original_dpi\n        return result"
        },
        {
          "file": "lib/matplotlib/backends/backend_agg.py",
          "type": "function",
          "name": "print_jpg",
          "class_name": "FigureCanvasAgg",
          "code": "def print_jpg(self, filename_or_obj, *args, **kwargs):\n            \"\"\"\n            Supported kwargs:\n\n            *quality*: The image quality, on a scale from 1 (worst) to\n                95 (best). The default is 95, if not given in the\n                matplotlibrc file in the savefig.jpeg_quality parameter.\n                Values above 95 should be avoided; 100 completely\n                disables the JPEG quantization stage.\n\n            *optimize*: If present, indicates that the encoder should\n                make an extra pass over the image in order to select\n                optimal encoder settings.\n\n            *progressive*: If present, indicates that this image\n                should be stored as a progressive JPEG file.\n            \"\"\"\n            buf, size = self.print_to_buffer()\n            if kwargs.pop(\"dryrun\", False):\n                return\n            image = Image.frombuffer('RGBA', size, buf, 'raw', 'RGBA', 0, 1)\n            options = restrict_dict(kwargs, ['quality', 'optimize',\n                                             'progressive'])\n\n            if 'quality' not in options:\n                options['quality'] = rcParams['savefig.jpeg_quality']\n\n            return image.save(filename_or_obj, format='jpeg', **options)"
        }
      ]
    },
    {
      "pr_number": 2587,
      "pr_title": "Make backend_pgf more flexible when saving to file-handles or streams (fix #1625).",
      "pr_body": "Since binary data and thus raster images cannot be embedded in pgf code, saving to a stream was prohibited in backend_pgf. This PR allows saving pgf code to streams after deactivating draw_image() and emitting a user warning.\n\nI'll merge this in a few days if there are no objections (fixing issue #1625).\n",
      "issue_id": 1625,
      "issue_title": "saving pgf to a stream is not supported",
      "issue_body": "Hi,\n\nThe pgf output does not support a stream as illustrated by this snippet\n\n``` python\nimport sys\nplt.plot([1,2])\nplt.savefig(sys.stdout, format='pgf')\n```\n\nIt returns:\n\n``` python\nValueError: saving pgf to a stream is not supported, consider using the pdf option of the pgf-backend\n```\n\nOther formats like png, eps, ps, svg support it. I don't see any good reason to do not support streams for pgf and I guess it is a bug :)\n\nThanks.\n",
      "issue_closed_at": "2013-11-12T13:20:52Z",
      "base_commit": "c91589c024aa4c95288173112d3d5c44e1703d27",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "line",
          "name": "line 12",
          "code": "import codecs\nimport atexit\nimport weakref\n\nimport matplotlib as mpl\nfrom matplotlib.backend_bases import RendererBase, GraphicsContextBase,\\"
        },
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FigureManagerPgf",
          "code": "def __init__(self, *args):\n        FigureManagerBase.__init__(self, *args)"
        },
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "print_pgf",
          "class_name": "FigureCanvasPgf",
          "code": "def print_pgf(self, fname_or_fh, *args, **kwargs):\n        \"\"\"\n        Output pgf commands for drawing the figure so it can be included and\n        rendered in latex documents.\n        \"\"\"\n        if kwargs.get(\"dryrun\", False):\n            self._print_pgf_to_fh(None, *args, **kwargs)\n            return\n\n        # figure out where the pgf is to be written to\n        if is_string_like(fname_or_fh):\n            with codecs.open(fname_or_fh, \"w\", encoding=\"utf-8\") as fh:\n                self._print_pgf_to_fh(fh, *args, **kwargs)\n        elif is_writable_file_like(fname_or_fh):\n            raise ValueError(\"saving pgf to a stream is not supported, \" +\n                             \"consider using the pdf option of the pgf-backend\")\n        else:\n            raise ValueError(\"filename must be a path\")"
        }
      ]
    },
    {
      "pr_number": 1993,
      "pr_title": "PS backend fails to savefig() pcolormesh with gouraud shading",
      "pr_body": "Following script\n\n``` python\nimport numpy as np\nfrom matplotlib.pyplot import figure, savefig\n\nn = 12\nx = np.linspace(-1.5,1.5,n)\ny = np.linspace(-1.5,1.5,n*2)\nX,Y = np.meshgrid(x,y);\n\nQx = np.cos(Y) - np.cos(X)\nQz = np.sin(Y) + np.sin(X)\nQx = (Qx + 1.1)\nZ = np.sqrt(X**2 + Y**2)/5;\nZ = (Z - Z.min()) / (Z.max() - Z.min())\n\nfig = figure()\nax = fig.add_subplot(111)\nax.pcolormesh(Qx,Qz,Z, shading='gouraud')\nsavefig('bug.eps')\n```\n\nresults in\n\n``` text\nTraceback (most recent call last):\n  File \"quad.py\", line 20, in <module>\n    savefig('bug.eps')\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/pyplot.py\", line 494, in savefig\n    return fig.savefig(*args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/figure.py\", line 1403, in savefig\n    self.canvas.print_figure(*args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_qt4agg.py\", line 161, in print_figure\n    FigureCanvasAgg.print_figure(self, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backend_bases.py\", line 2158, in print_figure\n    **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backend_bases.py\", line 1886, in print_eps\n    return ps.print_eps(*args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 974, in print_eps\n    return self._print_ps(outfile, 'eps', *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 1002, in _print_ps\n    **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 1095, in _print_figure\n    self.figure.draw(renderer)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/artist.py\", line 54, in draw_wrapper\n    draw(artist, renderer, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/figure.py\", line 1027, in draw\n    func(*args)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/artist.py\", line 54, in draw_wrapper\n    draw(artist, renderer, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/axes.py\", line 2086, in draw\n    a.draw(renderer)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/artist.py\", line 54, in draw_wrapper\n    draw(artist, renderer, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/collections.py\", line 1745, in draw\n    gc, triangles, colors, transform.frozen())\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 827, in draw_gouraud_triangles\n    points = trans.transform(points)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/transforms.py\", line 1561, in transform\n    return self.transform_affine(values)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/transforms.py\", line 1645, in transform_affine\n    return affine_transform(points, mtx)\nValueError: Invalid vertices array.\n```\n\nTested with\n\n``` text\ncommit af07100823d219e51f061cfc8e96d957fa661126\n\nBUILDING MATPLOTLIB\n            matplotlib: yes [1.3.x]\n                python: yes [2.7.4 (default, Apr 21 2013, 14:51:13)  [GCC\n                        4.7.2]]\n              platform: yes [linux2]\n\nREQUIRED DEPENDENCIES AND EXTENSIONS\n                 numpy: yes [version 1.6.2]\n              dateutil: yes [using dateutil version 2.1]\n               tornado: yes [using tornado version 2.3]\n             pyparsing: yes [using pyparsing version 1.5.7]\n                 pycxx: yes [pkg-config information for 'PyCXX' could not be\n                        found. Using local copy.]\n                libagg: yes [pkg-config information for 'libagg' could not\n                        be found. Using local copy.]\n              freetype: yes [version 16.0.10]\n                   png: yes [version 1.6.1]\n\nOPTIONAL SUBPACKAGES\n           sample_data: yes [installing]\n              toolkits: yes [installing]\n                 tests: yes [using nose version 1.1.2]\n\nOPTIONAL BACKEND EXTENSIONS\n                macosx: no  [Mac OS-X only]\n                qt4agg: yes [Qt: 4.8.2, PyQt4: 4.9.2]\n               gtk3agg: yes [version 3.3.6]\n             gtk3cairo: yes [version 3.3.6]\n                gtkagg: yes [Gtk: 2.24.17 pygtk: 2.24.0]\n                 tkagg: yes [version 81008]\n                 wxagg: yes [version 2.8.12.1]\n                   gtk: yes [Gtk: 2.24.17 pygtk: 2.24.0]\n                 qtagg: no  [pyqt not found]\n                   agg: yes [installing]\n                 cairo: yes [version 1.10.0]\n\nOPTIONAL LATEX DEPENDENCIES\n                dvipng: yes [version Larsson]\n           ghostscript: yes [version 9.06]\n                 latex: yes [version 3.1415926]\n               pdftops: yes [version 0.22.3]\n```\n",
      "issue_id": 1993,
      "issue_title": "PS backend fails to savefig() pcolormesh with gouraud shading",
      "issue_body": "Following script\n\n``` python\nimport numpy as np\nfrom matplotlib.pyplot import figure, savefig\n\nn = 12\nx = np.linspace(-1.5,1.5,n)\ny = np.linspace(-1.5,1.5,n*2)\nX,Y = np.meshgrid(x,y);\n\nQx = np.cos(Y) - np.cos(X)\nQz = np.sin(Y) + np.sin(X)\nQx = (Qx + 1.1)\nZ = np.sqrt(X**2 + Y**2)/5;\nZ = (Z - Z.min()) / (Z.max() - Z.min())\n\nfig = figure()\nax = fig.add_subplot(111)\nax.pcolormesh(Qx,Qz,Z, shading='gouraud')\nsavefig('bug.eps')\n```\n\nresults in\n\n``` text\nTraceback (most recent call last):\n  File \"quad.py\", line 20, in <module>\n    savefig('bug.eps')\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/pyplot.py\", line 494, in savefig\n    return fig.savefig(*args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/figure.py\", line 1403, in savefig\n    self.canvas.print_figure(*args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_qt4agg.py\", line 161, in print_figure\n    FigureCanvasAgg.print_figure(self, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backend_bases.py\", line 2158, in print_figure\n    **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backend_bases.py\", line 1886, in print_eps\n    return ps.print_eps(*args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 974, in print_eps\n    return self._print_ps(outfile, 'eps', *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 1002, in _print_ps\n    **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 1095, in _print_figure\n    self.figure.draw(renderer)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/artist.py\", line 54, in draw_wrapper\n    draw(artist, renderer, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/figure.py\", line 1027, in draw\n    func(*args)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/artist.py\", line 54, in draw_wrapper\n    draw(artist, renderer, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/axes.py\", line 2086, in draw\n    a.draw(renderer)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/artist.py\", line 54, in draw_wrapper\n    draw(artist, renderer, *args, **kwargs)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/collections.py\", line 1745, in draw\n    gc, triangles, colors, transform.frozen())\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/backends/backend_ps.py\", line 827, in draw_gouraud_triangles\n    points = trans.transform(points)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/transforms.py\", line 1561, in transform\n    return self.transform_affine(values)\n  File \"/tmp/matplotlib/build/lib.linux-x86_64-2.7/matplotlib/transforms.py\", line 1645, in transform_affine\n    return affine_transform(points, mtx)\nValueError: Invalid vertices array.\n```\n\nTested with\n\n``` text\ncommit af07100823d219e51f061cfc8e96d957fa661126\n\nBUILDING MATPLOTLIB\n            matplotlib: yes [1.3.x]\n                python: yes [2.7.4 (default, Apr 21 2013, 14:51:13)  [GCC\n                        4.7.2]]\n              platform: yes [linux2]\n\nREQUIRED DEPENDENCIES AND EXTENSIONS\n                 numpy: yes [version 1.6.2]\n              dateutil: yes [using dateutil version 2.1]\n               tornado: yes [using tornado version 2.3]\n             pyparsing: yes [using pyparsing version 1.5.7]\n                 pycxx: yes [pkg-config information for 'PyCXX' could not be\n                        found. Using local copy.]\n                libagg: yes [pkg-config information for 'libagg' could not\n                        be found. Using local copy.]\n              freetype: yes [version 16.0.10]\n                   png: yes [version 1.6.1]\n\nOPTIONAL SUBPACKAGES\n           sample_data: yes [installing]\n              toolkits: yes [installing]\n                 tests: yes [using nose version 1.1.2]\n\nOPTIONAL BACKEND EXTENSIONS\n                macosx: no  [Mac OS-X only]\n                qt4agg: yes [Qt: 4.8.2, PyQt4: 4.9.2]\n               gtk3agg: yes [version 3.3.6]\n             gtk3cairo: yes [version 3.3.6]\n                gtkagg: yes [Gtk: 2.24.17 pygtk: 2.24.0]\n                 tkagg: yes [version 81008]\n                 wxagg: yes [version 2.8.12.1]\n                   gtk: yes [Gtk: 2.24.17 pygtk: 2.24.0]\n                 qtagg: no  [pyqt not found]\n                   agg: yes [installing]\n                 cairo: yes [version 1.10.0]\n\nOPTIONAL LATEX DEPENDENCIES\n                dvipng: yes [version Larsson]\n           ghostscript: yes [version 9.06]\n                 latex: yes [version 3.1415926]\n               pdftops: yes [version 0.22.3]\n```\n",
      "issue_closed_at": "2013-05-16T16:07:26Z",
      "base_commit": "f3f345022481db548d7fdabcfe297925b08ca436",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "draw_gouraud_triangles",
          "class_name": "RendererPS",
          "code": "def draw_gouraud_triangles(self, gc, points, colors, trans):\n        assert len(points) == len(colors)\n        assert points.ndim == 3\n        assert points.shape[1] == 3\n        assert points.shape[2] == 2\n        assert colors.ndim == 3\n        assert colors.shape[1] == 3\n        assert colors.shape[2] == 4\n\n        points = trans.transform(points)\n\n        shape = points.shape\n        flat_points = points.reshape((shape[0] * shape[1], 2))\n        flat_colors = colors.reshape((shape[0] * shape[1], 4))\n        points_min = np.min(flat_points, axis=0) - (1 << 8)\n        points_max = np.max(flat_points, axis=0) + (1 << 8)\n        factor = float(0xffffffff) / (points_max - points_min)\n\n        xmin, ymin = points_min\n        xmax, ymax = points_max\n\n        streamarr = np.empty(\n            (shape[0] * shape[1],),\n            dtype=[('flags', 'u1'),\n                   ('points', '>u4', (2,)),\n                   ('colors', 'u1', (3,))])\n        streamarr['flags'] = 0\n        streamarr['points'] = (flat_points - points_min) * factor\n        streamarr['colors'] = flat_colors[:, :3] * 255.0\n\n        stream = quote_ps_string(streamarr.tostring())\n\n        self._pswriter.write(\"\"\"\ngsave\n<< /ShadingType 4\n   /ColorSpace [/DeviceRGB]\n   /BitsPerCoordinate 32\n   /BitsPerComponent 8\n   /BitsPerFlag 8\n   /AntiAlias true\n   /Decode [ %(xmin)f %(xmax)f %(ymin)f %(ymax)f 0 1 0 1 0 1 ]\n   /DataSource (%(stream)s)\n>>\nshfill\ngrestore\n\"\"\" % locals())"
        }
      ]
    },
    {
      "pr_number": 3083,
      "pr_title": "New rcParams to set pyplot.suptitle() defaults",
      "pr_body": "Two new rcParams are introduced:\n- figure.titlesize (xx-large)\n- figure.titleweight (normal)\n\nAllow to set the pyplot.suptitle() default font size and weight.\n\nFixes #3011\n",
      "issue_id": 3011,
      "issue_title": "Allow to customize default font size for suptitle() in matplotlibrc",
      "issue_body": "It seems that suptitle() is using the default font size and there's no way to customize the default font size, while title() is using axes.titlesize.\n\nI always have to specify the 'size' parameter when using suptitle, which is inconvenient.\n\nI'd love to have a \"figure.titlesize\", which defaults to 'xx-large' for consistency.\n",
      "issue_closed_at": "2014-11-13T17:20:52Z",
      "base_commit": "ed953827850179256c09996ac7a7a80e7d291aa6",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "suptitle",
          "class_name": "Figure",
          "code": "def suptitle(self, t, **kwargs):\n        \"\"\"\n        Add a centered title to the figure.\n\n        kwargs are :class:`matplotlib.text.Text` properties.  Using figure\n        coordinates, the defaults are:\n\n          *x* : 0.5\n            The x location of the text in figure coords\n\n          *y* : 0.98\n            The y location of the text in figure coords\n\n          *horizontalalignment* : 'center'\n            The horizontal alignment of the text\n\n          *verticalalignment* : 'top'\n            The vertical alignment of the text\n\n        A :class:`matplotlib.text.Text` instance is returned.\n\n        Example::\n\n          fig.suptitle('this is the figure title', fontsize=12)\n        \"\"\"\n        x = kwargs.pop('x', 0.5)\n        y = kwargs.pop('y', 0.98)\n        if ('horizontalalignment' not in kwargs) and ('ha' not in kwargs):\n            kwargs['horizontalalignment'] = 'center'\n\n        if ('verticalalignment' not in kwargs) and ('va' not in kwargs):\n            kwargs['verticalalignment'] = 'top'\n\n        sup = self.text(x, y, t, **kwargs)\n        if self._suptitle is not None:\n            self._suptitle.set_text(t)\n            self._suptitle.set_position((x, y))\n            self._suptitle.update_from(sup)\n            sup.remove()\n        else:\n            self._suptitle = sup\n        return self._suptitle"
        },
        {
          "file": "lib/matplotlib/rcsetup.py",
          "type": "function",
          "name": "__call__",
          "class_name": "ValidateInterval",
          "code": "def __call__(self, s):\n        try:\n            s = float(s)\n        except:\n            raise RuntimeError('Value must be a float; found \"%s\"' % s)\n\n        if self.cmin and s < self.vmin:\n            raise RuntimeError('Value must be >= %f; found \"%f\"' %\n                               (self.vmin, s))\n        elif not self.cmin and s <= self.vmin:\n            raise RuntimeError('Value must be > %f; found \"%f\"' %\n                               (self.vmin, s))\n\n        if self.cmax and s > self.vmax:\n            raise RuntimeError('Value must be <= %f; found \"%f\"' %\n                               (self.vmax, s))\n        elif not self.cmax and s >= self.vmax:\n            raise RuntimeError('Value must be < %f; found \"%f\"' %\n                               (self.vmax, s))\n        return s"
        }
      ]
    }
  ]
}