{
  "instance_id": "matplotlib__matplotlib-25332",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-02-26T11:18:40Z",
  "problem_statement": "[Bug]: Unable to pickle figure with aligned labels\n### Bug summary\r\n\r\n Unable to pickle figure after calling `align_labels()`\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport pickle\r\n\r\nfig = plt.figure()\r\nax1 = fig.add_subplot(211)\r\nax2 = fig.add_subplot(212)\r\ntime=[0,1,2,3,4]\r\nspeed=[40000,4300,4500,4700,4800]\r\nacc=[10,11,12,13,14]\r\nax1.plot(time,speed)\r\nax1.set_ylabel('speed')\r\nax2.plot(time,acc)\r\nax2.set_ylabel('acc')\r\n\r\nfig.align_labels() ##pickling works after removing this line \r\n\r\npickle.dumps(fig)\r\nplt.show()\r\n```\r\n\r\n\r\n### Actual outcome\r\n```\r\nalign.py\", line 16\r\npickle.dumps(fig)\r\nTypeError: cannot pickle 'weakref.ReferenceType' object\r\n```\r\n### Expected outcome\r\n\r\nPickling successful\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\nWindows\r\n\r\n### Matplotlib Version\r\n\r\n3.7.0\r\n\r\n### Matplotlib Backend\r\n\r\n_No response_\r\n\r\n### Python version\r\n\r\n_No response_\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\nNone\n",
  "patch": "diff --git a/lib/matplotlib/cbook.py b/lib/matplotlib/cbook.py\n--- a/lib/matplotlib/cbook.py\n+++ b/lib/matplotlib/cbook.py\n@@ -788,6 +788,19 @@ class Grouper:\n     def __init__(self, init=()):\n         self._mapping = {weakref.ref(x): [weakref.ref(x)] for x in init}\n \n+    def __getstate__(self):\n+        return {\n+            **vars(self),\n+            # Convert weak refs to strong ones.\n+            \"_mapping\": {k(): [v() for v in vs] for k, vs in self._mapping.items()},\n+        }\n+\n+    def __setstate__(self, state):\n+        vars(self).update(state)\n+        # Convert strong refs to weak ones.\n+        self._mapping = {weakref.ref(k): [*map(weakref.ref, vs)]\n+                         for k, vs in self._mapping.items()}\n+\n     def __contains__(self, item):\n         return weakref.ref(item) in self._mapping\n \n",
  "similar_bug_items": [
    {
      "pr_number": 4050,
      "pr_title": "Fix masked array handling",
      "pr_body": "This fixes #4049, and is an alternative to #4008.  Unlike #4008, this fixes the problem directly in `_fast_from_vertices_and_codes` rather than at a number of call sites that call it.  While it does make that function slower, it apparently needs the masked array treatment from all places that call it anyway, so sticking it right inside the function seems fine.\n\nIn the process of fixing this bug, it was discovered that a fix to handle NaNs in draw_markers (cf43d57) was not brought over from the 1.4.x branch (probably my fault, since merges of C++ code are really hairy right now).  This missing bugfix was papered over by the fact that NaNs were no longer being passed to `draw_markers`.\n\nLastly, this adds an optimization to the log scaling to use NaNs rather than masked arrays.  The original array is not touched, so this should be effectively the same thing, just without the large time and memory overhead of masked arrays.\n\nThis includes a test I shamelessly cherry-picked from #4008.\n",
      "issue_id": 4049,
      "issue_title": "Issue with plotting zeros in log space",
      "issue_body": "The following simple example shows an issues with zero values when calling `loglog`:\n\n```\nimport matplotlib.pyplot as plt\n\nnu = [1e2, 1e3, 1e4, 1e5, 1e6, 1e7]\nfnu = [1, 1e-30, 1e-45, 0., 0., 0.]\n\nfig = plt.figure()\nax = fig.add_subplot(1,1,1)\nax.loglog(nu, fnu, color='blue')\nfig.savefig('mpl.png')\n```\n\nThe output is:\n\n![mpl](https://cloud.githubusercontent.com/assets/314716/5959827/8d0677cc-a7d1-11e4-8cd3-7227dee4a37a.png)\n\nOf course, zero is undefined in log space, but I don't understand why it defaults to plotting a value of 0.1 instead. Doing:\n\n```\nfnu = [1, 1e-30, 1e-45, np.nan, np.nan, np.nan]\n```\n\nDoes the right thing on the other hand. So maybe the best solution is to replace 0 values by np.nan inside `loglog`?\n\nI am using matplotlib 1.5.x with Python 3.4.\n",
      "issue_closed_at": "2015-01-30T23:18:44Z",
      "base_commit": "dd4b2bc76c800f2fae69953c888681fd67788192",
      "changes": [
        {
          "file": "lib/matplotlib/path.py",
          "type": "function",
          "name": "_fast_from_codes_and_verts",
          "class_name": "Path",
          "code": "def _fast_from_codes_and_verts(cls, verts, codes, internals=None):\n        \"\"\"\n        Creates a Path instance without the expense of calling the constructor\n\n        Parameters\n        ----------\n        verts : numpy array\n        codes : numpy array (may not be None)\n        internals : dict or None\n            The attributes that the resulting path should have.\n            Allowed keys are ``readonly``, ``should_simplify``,\n            ``simplify_threshold``, ``has_nonfinite`` and\n            ``interpolation_steps``.\n\n        \"\"\"\n        internals = internals or {}\n        pth = cls.__new__(cls)\n        pth._vertices = verts\n        pth._codes = codes\n        pth._readonly = internals.pop('readonly', False)\n        pth.should_simplify = internals.pop('should_simplify', True)\n        pth.simplify_threshold = (\n            internals.pop('simplify_threshold',\n                          rcParams['path.simplify_threshold'])\n        )\n        pth._has_nonfinite = internals.pop('has_nonfinite', False)\n        pth._interpolation_steps = internals.pop('interpolation_steps', 1)\n        if internals:\n            raise ValueError('Unexpected internals provided to '\n                             '_fast_from_codes_and_verts: '\n                             '{0}'.format('\\n *'.join(six.iterkeys(\n                                 internals\n                             ))))\n        return pth"
        },
        {
          "file": "lib/matplotlib/path.py",
          "type": "function",
          "name": "_fast_from_codes_and_verts",
          "class_name": "Path",
          "code": "def _fast_from_codes_and_verts(cls, verts, codes, internals=None):\n        \"\"\"\n        Creates a Path instance without the expense of calling the constructor\n\n        Parameters\n        ----------\n        verts : numpy array\n        codes : numpy array (may not be None)\n        internals : dict or None\n            The attributes that the resulting path should have.\n            Allowed keys are ``readonly``, ``should_simplify``,\n            ``simplify_threshold``, ``has_nonfinite`` and\n            ``interpolation_steps``.\n\n        \"\"\"\n        internals = internals or {}\n        pth = cls.__new__(cls)\n        pth._vertices = verts\n        pth._codes = codes\n        pth._readonly = internals.pop('readonly', False)\n        pth.should_simplify = internals.pop('should_simplify', True)\n        pth.simplify_threshold = (\n            internals.pop('simplify_threshold',\n                          rcParams['path.simplify_threshold'])\n        )\n        pth._has_nonfinite = internals.pop('has_nonfinite', False)\n        pth._interpolation_steps = internals.pop('interpolation_steps', 1)\n        if internals:\n            raise ValueError('Unexpected internals provided to '\n                             '_fast_from_codes_and_verts: '\n                             '{0}'.format('\\n *'.join(six.iterkeys(\n                                 internals\n                             ))))\n        return pth"
        },
        {
          "file": "lib/matplotlib/scale.py",
          "type": "function",
          "name": "get_transform",
          "class_name": "SymmetricalLogScale",
          "code": "def get_transform(self):\n        \"\"\"\n        Return a :class:`SymmetricalLogTransform` instance.\n        \"\"\"\n        return self._transform"
        }
      ]
    },
    {
      "pr_number": 11165,
      "pr_title": "Fix figure window icon",
      "pr_body": "## PR Summary\r\nFixes #11138 \r\n\r\n## PR Checklist\r\n\r\n- [x] Has Pytest style unit tests\r\n- [x] Code is PEP 8 compliant\r\n- [x] New features are documented, with examples if plot related\r\n- [x] Documentation is sphinx and numpydoc compliant\r\n- [ ] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [ ] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way",
      "issue_id": 11138,
      "issue_title": "Only the first figure window has mpl icon, all other figures have default tk icon.",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\nWhen plot more than one figure window, only the first figure window has the default mpl icon (come from matplotlib.ppm), and all other figure windows' icon is set to the default tk icon.\r\n\r\n**Code for reproduction**\r\n\r\ncode 1:\r\n```python\r\nimport logging\r\nimport matplotlib.pyplot as plt\r\nlogging.basicConfig(format=\"%(name)s %(levelname)s: %(message)s\", level=logging.INFO)\r\nplt.figure(1)\r\nplt.plot([1,2,3])\r\nplt.show()\r\n```\r\n\r\ncode 2:\r\n```python\r\nimport logging\r\nimport matplotlib.pyplot as plt\r\nlogging.basicConfig(format=\"%(name)s %(levelname)s: %(message)s\", level=logging.INFO)\r\nplt.figure(1)\r\nplt.plot([1,2,3])\r\nplt.figure(2)\r\nplt.plot([4,5,6])\r\nplt.show()\r\n```\r\n\r\ncode 3:\r\n```python\r\nimport logging\r\nimport matplotlib.pyplot as plt\r\nlogging.basicConfig(format=\"%(name)s %(levelname)s: %(message)s\", level=logging.INFO)\r\nplt.figure(1)\r\nplt.plot([1,2,3])\r\nplt.figure(2)\r\nplt.plot([4,5,6])\r\nplt.figure(3)\r\nplt.plot([7,8,9])\r\nplt.show()\r\n```\r\n\r\n**Actual outcome**\r\ncode 1 runs normally.\r\n\r\ncode 2 will log:\r\n```\r\nmatplotlib.backends._backend_tk INFO: Could not load matplotlib icon: can't use \"pyimage10\" as iconphoto: not a photo image\r\n```\r\nAnd the icon of figure 1 and figure 2 is different.\r\n\r\ncode 3 will log:\r\n```\r\nmatplotlib.backends._backend_tk INFO: Could not load matplotlib icon: can't use \"pyimage10\" as iconphoto: not a photo image\r\nmatplotlib.backends._backend_tk INFO: Could not load matplotlib icon: can't use \"pyimage19\" as iconphoto: not a photo image\r\n```\r\nThe icon of figure 1 is different than figure 2 and 3.\r\nScreenshot attached:\r\nIcon of window:\r\n![figure_title](https://user-images.githubusercontent.com/35422407/39393621-04eec34c-4afa-11e8-85a0-6f4aa58f8c4c.png)\r\n\r\nAnd the icon in taskbar:\r\n![taskbar](https://user-images.githubusercontent.com/35422407/39393623-0ad91244-4afa-11e8-8536-49138602fb4e.png)\r\n\r\nIf debug, the log information is coming from the exception:\r\n```\r\nfile: matplotlib/backends/_backend_tk.py\r\nclass _BackendTk()\r\nmethod new_figure_manager_given_figure()\r\n        # Put a mpl icon on the window rather than the default tk icon.\r\n        # Tkinter doesn't allow colour icons on linux systems, but tk>=8.5 has\r\n        # a iconphoto command which we call directly. Source:\r\n        # http://mail.python.org/pipermail/tkinter-discuss/2006-November/000954.html\r\n        icon_fname = os.path.join(\r\n            rcParams['datapath'], 'images', 'matplotlib.ppm')\r\n        icon_img = Tk.PhotoImage(file=icon_fname)\r\n        try:\r\n            window.tk.call('wm', 'iconphoto', window._w, icon_img)\r\n        except Exception as exc:\r\n            # log the failure (due e.g. to Tk version), but carry on\r\n            _log.info('Could not load matplotlib icon: %s', exc)\r\n```\r\n\r\n**Expected outcome**\r\nAll figure's icon are same by default.\r\n\r\n**Matplotlib version**\r\n  * Operating system: Windows 7 pro.\r\n  * Matplotlib version:  2.2.2\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: 3.6.5\r\n\r\n",
      "issue_closed_at": "2018-07-08T17:50:21Z",
      "base_commit": "c20b0c2bede5e354ac3bc90858d247c2a2d10d5e",
      "changes": [
        {
          "file": "lib/matplotlib/backends/_backend_tk.py",
          "type": "function",
          "name": "new_figure_manager_given_figure",
          "class_name": "_BackendTk",
          "code": "def new_figure_manager_given_figure(cls, num, figure):\n        \"\"\"\n        Create a new figure manager instance for the given figure.\n        \"\"\"\n        with _restore_foreground_window_at_end():\n            window = Tk.Tk(className=\"matplotlib\")\n            window.withdraw()\n\n            # Put a mpl icon on the window rather than the default tk icon.\n            # Tkinter doesn't allow colour icons on linux systems, but tk>=8.5 has\n            # a iconphoto command which we call directly. Source:\n            # http://mail.python.org/pipermail/tkinter-discuss/2006-November/000954.html\n            icon_fname = os.path.join(\n                rcParams['datapath'], 'images', 'matplotlib.ppm')\n            icon_img = Tk.PhotoImage(file=icon_fname)\n            try:\n                window.tk.call('wm', 'iconphoto', window._w, icon_img)\n            except Exception as exc:\n                # log the failure (due e.g. to Tk version), but carry on\n                _log.info('Could not load matplotlib icon: %s', exc)\n\n            canvas = cls.FigureCanvas(figure, master=window)\n            manager = cls.FigureManager(canvas, num, window)\n            if matplotlib.is_interactive():\n                manager.show()\n                canvas.draw_idle()\n            return manager"
        }
      ]
    },
    {
      "pr_number": 11601,
      "pr_title": "FIX: subplots don't mutate kwargs passed by user.",
      "pr_body": "## PR Summary\r\n\r\nCloses #11515\r\n\r\nDon't mess with the user-supplied dictionaries in `subplots`...\r\n\r\nDunno if this really merits a test..\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\n\r\nsubplot_kw = {'sharex': 'all'}\r\nprint(subplot_kw)\r\nfig, ax = plt.subplots(2, 1, subplot_kw=subplot_kw)\r\nprint(subplot_kw)\r\n```\r\n\r\n### New:\r\n\r\n```\r\n{'sharex': 'all'}\r\n{'sharex': 'all'}\r\n```\r\n\r\n### Old:\r\n\r\n```\r\n{'sharex': 'all'}\r\n{'sharex': None, 'sharey': None}\r\n```\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is PEP 8 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": 11515,
      "issue_title": "using 'sharex' once in 'subplots' function can affect subsequent calles to 'subplots'",
      "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\nUsing 'sharex' keyword argument once in the subplot function causes all subsequent axes returned by subplots to have a shared x-axis when the subplot_kw argument is also included.\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\r\nfrom pylab import *\r\n\r\n\r\ndefaults_subplots = dict()\r\n\r\nx1, y1 = range(30), randn(30)\r\nx2, y2 = range(30, 60), randn(30)\r\n\r\nclose('all')\r\n\r\n## Case 1 : single subplot (can be multiple as well)\r\n##  this behaves correctly\r\nf2, ax2 =  subplots(1,1, sharex=True, subplot_kw=defaults_subplots)\r\ntitle('figure 1')\r\nax2.plot(x2, -y2)\r\n\r\n## Case 2 : multiple subplots (single subplot causes bug to not appear)\r\nf1, (ax1, _) = subplots(2,1, sharex=True, subplot_kw=defaults_subplots)\r\ntitle('figure 2')\r\nax1.plot(x1, y1)\r\n\r\n## Case 3 : same code as case 1, however axis is shared with subplots in case 2\r\nf2, ax3 =  subplots(1,1, subplot_kw=defaults_subplots)\r\ntitle('figure 3')\r\nax3.plot(x2, -y2)\r\n\r\nshow()\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\nFigure 1 has an independent x axis, while subplots in figures 2 and 3 share an x axis. \r\n\r\n**Expected outcome**\r\nThe x axis should be independent for each figure.  The subplots in figure 2 should have a shared axis.\r\n \r\n**Fix**\r\nThe bug is the result of modifying the `subplot_kw` dictionary, following patch resolves the issue:\r\n\r\n```\r\n@@ -1171,6 +1171,9 @@\r\n                          (sharey, share_values))\r\n     if subplot_kw is None:\r\n         subplot_kw = {}\r\n+    else:\r\n+        subplot_kw = subplot_kw.copy()\r\n+\r\n     if gridspec_kw is None:\r\n         gridspec_kw = {}\r\n ```\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: Ubuntu 16.04\r\n  * Matplotlib version: 1.5.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: 3.5.2\r\nMatplotlib installed through ubuntu package manager.\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
      "issue_closed_at": "2018-07-09T19:17:19Z",
      "base_commit": "3d19f94284e15d4d1c4c1d736cf646d18856960a",
      "changes": [
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "subplots",
          "class_name": "Figure",
          "code": "def subplots(self, nrows=1, ncols=1, sharex=False, sharey=False,\n                 squeeze=True, subplot_kw=None, gridspec_kw=None):\n        \"\"\"\n        Add a set of subplots to this figure.\n\n        This utility wrapper makes it convenient to create common layouts of\n        subplots in a single call.\n\n        Parameters\n        ----------\n        nrows, ncols : int, optional, default: 1\n            Number of rows/columns of the subplot grid.\n\n        sharex, sharey : bool or {'none', 'all', 'row', 'col'}, default: False\n            Controls sharing of properties among x (`sharex`) or y (`sharey`)\n            axes:\n\n                - True or 'all': x- or y-axis will be shared among all\n                  subplots.\n                - False or 'none': each subplot x- or y-axis will be\n                  independent.\n                - 'row': each subplot row will share an x- or y-axis.\n                - 'col': each subplot column will share an x- or y-axis.\n\n            When subplots have a shared x-axis along a column, only the x tick\n            labels of the bottom subplot are created. Similarly, when subplots\n            have a shared y-axis along a row, only the y tick labels of the\n            first column subplot are created. To later turn other subplots'\n            ticklabels on, use `~matplotlib.axes.Axes.tick_params`.\n\n        squeeze : bool, optional, default: True\n            - If True, extra dimensions are squeezed out from the returned\n              array of Axes:\n\n                - if only one subplot is constructed (nrows=ncols=1), the\n                  resulting single Axes object is returned as a scalar.\n                - for Nx1 or 1xM subplots, the returned object is a 1D numpy\n                  object array of Axes objects.\n                - for NxM, subplots with N>1 and M>1 are returned\n                  as a 2D array.\n\n            - If False, no squeezing at all is done: the returned Axes object\n              is always a 2D array containing Axes instances, even if it ends\n              up being 1x1.\n\n        subplot_kw : dict, optional\n            Dict with keywords passed to the\n            :meth:`~matplotlib.figure.Figure.add_subplot` call used to create\n            each subplot.\n\n        gridspec_kw : dict, optional\n            Dict with keywords passed to the\n            `~matplotlib.gridspec.GridSpec` constructor used to create\n            the grid the subplots are placed on.\n\n        Returns\n        -------\n        ax : `~.axes.Axes` object or array of Axes objects.\n            *ax* can be either a single `~matplotlib.axes.Axes` object or\n            an array of Axes objects if more than one subplot was created. The\n            dimensions of the resulting array can be controlled with the\n            squeeze keyword, see above.\n\n        Examples\n        --------\n        ::\n\n            # First create some toy data:\n            x = np.linspace(0, 2*np.pi, 400)\n            y = np.sin(x**2)\n\n            # Create a figure\n            plt.figure(1, clear=True)\n\n            # Creates a subplot\n            ax = fig.subplots()\n            ax.plot(x, y)\n            ax.set_title('Simple plot')\n\n            # Creates two subplots and unpacks the output array immediately\n            ax1, ax2 = fig.subplots(1, 2, sharey=True)\n            ax1.plot(x, y)\n            ax1.set_title('Sharing Y axis')\n            ax2.scatter(x, y)\n\n            # Creates four polar axes, and accesses them through the\n            # returned array\n            axes = fig.subplots(2, 2, subplot_kw=dict(polar=True))\n            axes[0, 0].plot(x, y)\n            axes[1, 1].scatter(x, y)\n\n            # Share a X axis with each column of subplots\n            fig.subplots(2, 2, sharex='col')\n\n            # Share a Y axis with each row of subplots\n            fig.subplots(2, 2, sharey='row')\n\n            # Share both X and Y axes with all subplots\n            fig.subplots(2, 2, sharex='all', sharey='all')\n\n            # Note that this is the same as\n            fig.subplots(2, 2, sharex=True, sharey=True)\n\n            See Also\n            --------\n            .pyplot.subplots\n            .Figure.add_subplot\n            .pyplot.subplot\n            \"\"\"\n\n        if isinstance(sharex, bool):\n            sharex = \"all\" if sharex else \"none\"\n        if isinstance(sharey, bool):\n            sharey = \"all\" if sharey else \"none\"\n        share_values = [\"all\", \"row\", \"col\", \"none\"]\n        if sharex not in share_values:\n            # This check was added because it is very easy to type\n            # `subplots(1, 2, 1)` when `subplot(1, 2, 1)` was intended.\n            # In most cases, no error will ever occur, but mysterious behavior\n            # will result because what was intended to be the subplot index is\n            # instead treated as a bool for sharex.\n            if isinstance(sharex, Integral):\n                warnings.warn(\n                    \"sharex argument to subplots() was an integer. \"\n                    \"Did you intend to use subplot() (without 's')?\")\n\n            raise ValueError(\"sharex [%s] must be one of %s\" %\n                             (sharex, share_values))\n        if sharey not in share_values:\n            raise ValueError(\"sharey [%s] must be one of %s\" %\n                             (sharey, share_values))\n        if subplot_kw is None:\n            subplot_kw = {}\n        if gridspec_kw is None:\n            gridspec_kw = {}\n\n        if self.get_constrained_layout():\n            gs = GridSpec(nrows, ncols, figure=self, **gridspec_kw)\n        else:\n            # this should turn constrained_layout off if we don't want it\n            gs = GridSpec(nrows, ncols, figure=None, **gridspec_kw)\n\n        # Create array to hold all axes.\n        axarr = np.empty((nrows, ncols), dtype=object)\n        for row in range(nrows):\n            for col in range(ncols):\n                shared_with = {\"none\": None, \"all\": axarr[0, 0],\n                               \"row\": axarr[row, 0], \"col\": axarr[0, col]}\n                subplot_kw[\"sharex\"] = shared_with[sharex]\n                subplot_kw[\"sharey\"] = shared_with[sharey]\n                axarr[row, col] = self.add_subplot(gs[row, col], **subplot_kw)\n\n        # turn off redundant tick labeling\n        if sharex in [\"col\", \"all\"]:\n            # turn off all but the bottom row\n            for ax in axarr[:-1, :].flat:\n                ax.xaxis.set_tick_params(which='both',\n                                         labelbottom=False, labeltop=False)\n                ax.xaxis.offsetText.set_visible(False)\n        if sharey in [\"row\", \"all\"]:\n            # turn off all but the first column\n            for ax in axarr[:, 1:].flat:\n                ax.yaxis.set_tick_params(which='both',\n                                         labelleft=False, labelright=False)\n                ax.yaxis.offsetText.set_visible(False)\n\n        if squeeze:\n            # Discarding unneeded dimensions that equal 1.  If we only have one\n            # subplot, just return it instead of a 1-element array.\n            return axarr.item() if axarr.size == 1 else axarr.squeeze()\n        else:\n            # Returned axis array will be always 2-d, even if nrows=ncols=1.\n            return axarr"
        }
      ]
    },
    {
      "pr_number": 2380,
      "pr_title": "check if pyparsing <<= is broken instead of checking the version",
      "pr_body": "distributions with packaged pyparsing might have already patched\npyparsing but not bumped the version number.\ncloses #2376\n",
      "issue_id": 2376,
      "issue_title": "feature test pyparsing 2 bug instead of version check",
      "issue_body": "the last pyparsing python3 weirdness already screwed distributions, the recent revert makes it even worse.\nmatplotlib should feature test for the bug that breaks 2.0.0 instead of relying on version numbers to at least give distributions a chance of resolving the issue in a sane way.\nthis patch, if accepted, should be applied to master and the 1.3.x branch\n\n``` diff\n--- a/setupext.py\n+++ b/setupext.py\n@@ -984,14 +984,13 @@ class Pyparsing(SetupPackage):\n                 \"matplotlib requires pyparsing >= {0}\".format(\n                     '.'.join(str(x) for x in required)))\n\n-        if pyparsing.__version__ == \"2.0.0\":\n-            if PY3:\n+        # check for bug in 2.0.0\n+        f = pyparsing.Forward()\n+        f <<= pyparsing.Literal('a')\n+        if f is None:\n                 return (\n                     \"pyparsing 2.0.0 has bugs that prevent its use with \"\n                     \"matplotlib\")\n-            else:\n-                return (\n-                    \"pyparsing 2.0.0 is not compatible with Python 2.x\")\n\n         return \"using pyparsing version %s\" % pyparsing.__version__\n```\n",
      "issue_closed_at": "2013-09-06T13:14:43Z",
      "base_commit": "0f8a45216a964ff76372428f95fa449709ba7565",
      "changes": [
        {
          "file": "lib/matplotlib/__init__.py",
          "type": "function",
          "name": "compare_versions",
          "class_name": null,
          "code": "def compare_versions(a, b):\n    \"return True if a is greater than or equal to b\"\n    if a:\n        a = distutils.version.LooseVersion(a)\n        b = distutils.version.LooseVersion(b)\n        if a>=b: return True\n        else: return False\n    else: return False"
        },
        {
          "file": "lib/matplotlib/__init__.py",
          "type": "function",
          "name": "compare_versions",
          "class_name": null,
          "code": "def compare_versions(a, b):\n    \"return True if a is greater than or equal to b\"\n    if a:\n        a = distutils.version.LooseVersion(a)\n        b = distutils.version.LooseVersion(b)\n        if a>=b: return True\n        else: return False\n    else: return False"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "get_install_requires",
          "class_name": "Pyparsing",
          "code": "def get_install_requires(self):\n        return ['pyparsing>=1.5.6,!=2.0.0']"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "check",
          "class_name": "PdfToPs",
          "code": "def check(self):\n        try:\n            output = check_output('pdftops -v', shell=True,\n                                  stderr=subprocess.STDOUT)\n            for line in output.splitlines():\n                line = line.decode()\n                if 'version' in line:\n                    return \"version %s\" % line.split()[2]\n        except (IndexError, ValueError, subprocess.CalledProcessError):\n            pass\n\n        raise CheckFailed()"
        }
      ]
    },
    {
      "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\")"
        }
      ]
    }
  ]
}