{
  "instance_id": "matplotlib__matplotlib-23913",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-09-16T21:51:24Z",
  "problem_statement": "legend draggable as keyword\n<!--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### Feature request\r\n\r\n**There is not keyword to make legend draggable at creation**\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n\r\nIs there a code reason why one can not add a \"draggable=True\" keyword to the __init__ function for Legend?  This would be more handy than having to call it after legend creation.  And, naively, it would seem simple to do.  But maybe there is a reason why it would not work?\n",
  "patch": "diff --git a/lib/matplotlib/legend.py b/lib/matplotlib/legend.py\n--- a/lib/matplotlib/legend.py\n+++ b/lib/matplotlib/legend.py\n@@ -286,6 +286,9 @@ def _update_bbox_to_anchor(self, loc_in_canvas):\n     The custom dictionary mapping instances or types to a legend\n     handler. This *handler_map* updates the default handler map\n     found at `matplotlib.legend.Legend.get_legend_handler_map`.\n+\n+draggable : bool, default: False\n+    Whether the legend can be dragged with the mouse.\n \"\"\")\n \n \n@@ -342,7 +345,8 @@ def __init__(\n         title_fontproperties=None,  # properties for the legend title\n         alignment=\"center\",       # control the alignment within the legend box\n         *,\n-        ncol=1  # synonym for ncols (backward compatibility)\n+        ncol=1,  # synonym for ncols (backward compatibility)\n+        draggable=False  # whether the legend can be dragged with the mouse\n     ):\n         \"\"\"\n         Parameters\n@@ -537,7 +541,9 @@ def val_or_rc(val, rc_name):\n             title_prop_fp.set_size(title_fontsize)\n \n         self.set_title(title, prop=title_prop_fp)\n+\n         self._draggable = None\n+        self.set_draggable(state=draggable)\n \n         # set the text color\n \n",
  "similar_bug_items": [
    {
      "pr_number": 10142,
      "pr_title": "FIX: prevent recursive draws with plt.ion not in IPython with qt5agg",
      "pr_body": "## PR Summary\r\n\r\nIn 9b8a944db954c8240e14413e517e1b781881b80b the latch logic was moved\r\nabove the call to `self.draw` to de-latch the call to\r\n`__draw_idle_agg` in `paintEvent` to protect against `blit` calls\r\nduring a draw.\r\n\r\nHowever, this now fails to properly latch the base draw so with\r\n`plt.ion` (not in IPython) the stale callback will cause a (possibly\r\ninfinite) recursion in the draw due to stale being set as part of the\r\ndraw call (which will re-trigger the draw).\r\n\r\nThis adds a second flag to track if we are currently rendering and\r\nbail.\r\n\r\ncloses #10140\r\n\r\n\r\nI suspect a better fix is to re-think how the stale callbacks propagate or how the non-ipython stale callback works, but I think this is the minimal change which fixes a critical issue.\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
      "issue_id": 10140,
      "issue_title": "Qt5Agg eats 100% CPU when plotting with block=True in interactive mode",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\nWhen using Qt5Agg, Matplotlib consumes a full CPU core while an interactive window is open.  The cursor repeatedly shows a \"spinner\" when I mouse over the canvas.  This didn't used to be the case.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nplt.plot([1, 2], [3, 4])\r\nplt.ion()\r\nplt.show(block=True)\r\n```\r\n\r\n**Actual outcome**\r\n\r\nOne of the CPU cores is pegged at 100%.  Interface is very slow and barely responsive.\r\n\r\n**Expected outcome**\r\n\r\nShould not use up 100% of a CPU core when nothing is happening.\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating system: Linux 4.14.8\r\n  * Matplotlib version: 2.1.1+864.g92a93d698.dirty\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: 3.6.4\r\n\r\nThis was reproduced on both XMonad and XFCE4.  Affects both `python-matplotlib` (2.1.1) from official Arch repos and `python-matplotlib-git` from AUR.",
      "issue_closed_at": "2018-01-04T20:28:55Z",
      "base_commit": "008da385fe73614059430073a722d674ade55014",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def __init__(self, figure):\n        super(FigureCanvasQTAggBase, self).__init__(figure=figure)\n        self.setAttribute(QtCore.Qt.WA_OpaquePaintEvent)\n        self._agg_draw_pending = False\n        self._bbox_queue = []\n        self._drawRect = None"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "draw",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def draw(self):\n        \"\"\"Draw the figure with Agg, and queue a request for a Qt draw.\n        \"\"\"\n        # The Agg draw is done here; delaying causes problems with code that\n        # uses the result of the draw() to update plot elements.\n        super(FigureCanvasQTAggBase, self).draw()\n        self.update()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "draw_idle",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def draw_idle(self):\n        \"\"\"Queue redraw of the Agg buffer and request Qt paintEvent.\n        \"\"\"\n        # The Agg draw needs to be handled by the same thread matplotlib\n        # modifies the scene graph from. Post Agg draw request to the\n        # current event loop in order to ensure thread affinity and to\n        # accumulate multiple draw requests from event handling.\n        # TODO: queued signal connection might be safer than singleShot\n        if not self._agg_draw_pending:\n            self._agg_draw_pending = True\n            QtCore.QTimer.singleShot(0, self.__draw_idle_agg)"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5agg.py",
          "type": "function",
          "name": "__draw_idle_agg",
          "class_name": "FigureCanvasQTAggBase",
          "code": "def __draw_idle_agg(self, *args):\n        # if nothing to do, bail\n        if not self._agg_draw_pending:\n            return\n        # we have now tried this function at least once, do not run\n        # again until re-armed.  Doing this here rather than after\n        # protects against recursive calls triggered through self.draw\n        self._agg_draw_pending = False\n        # if negative size, bail\n        if self.height() < 0 or self.width() < 0:\n            return\n        try:\n            # actually do the drawing\n            self.draw()\n        except Exception:\n            # Uncaught exceptions are fatal for PyQt5, so catch them instead.\n            traceback.print_exc()"
        }
      ]
    },
    {
      "pr_number": 9999,
      "pr_title": "improve legend docstring",
      "pr_body": "## PR Summary\r\n\r\nThis improves the legend docstrings:\r\n- describe handles and labels\r\n- add call signatures to `Axes.legend()`\r\n- Structure and reorder `Axes.legend()` so that the most common use case is at the top.\r\n\r\nalso fixes #8668",
      "issue_id": 8668,
      "issue_title": "handles keyword argument not documented in the help of legend",
      "issue_body": "<!--To help us understand and resolve your issue, please fill out the form to the best of your ability.-->\r\n<!--You can feel free to delete the sections that do not apply.-->\r\n\r\n### Bug report\r\n\r\n**Bug summary**\r\n\r\nThe following on-line documentation has an example where the `handles` keyword is used in a call to `plt.legend`: <http://matplotlib.org/users/legend_guide.html#controlling-the-legend-entries>\r\n\r\nThis keyword argument is not present in the internal function documentation.\r\n\r\n**Matplotlib version**\r\n  * Operating System: Linux (Xubuntu 16.04)\r\n  * Matplotlib Version: 2.0.0\r\n  * Python Version: 3.6\r\n\r\n",
      "issue_closed_at": "2017-12-18T15:29:15Z",
      "base_commit": "7a6bab7ca74308de0afc9656573b9c7de851c236",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_axes.py",
          "type": "function",
          "name": "legend",
          "class_name": "Axes",
          "code": "def legend(self, *args, **kwargs):\n        \"\"\"\n        Places a legend on the axes.\n\n        To make a legend for lines which already exist on the axes\n        (via plot for instance), simply call this function with an iterable\n        of strings, one for each legend item. For example::\n\n            ax.plot([1, 2, 3])\n            ax.legend(['A simple line'])\n\n        However, in order to keep the \"label\" and the legend element\n        instance together, it is preferable to specify the label either at\n        artist creation, or by calling the\n        :meth:`~matplotlib.artist.Artist.set_label` method on the artist::\n\n            line, = ax.plot([1, 2, 3], label='Inline label')\n            # Overwrite the label by calling the method.\n            line.set_label('Label via method')\n            ax.legend()\n\n        Specific lines can be excluded from the automatic legend element\n        selection by defining a label starting with an underscore.\n        This is default for all artists, so calling :meth:`legend` without\n        any arguments and without setting the labels manually will result in\n        no legend being drawn.\n\n        For full control of which artists have a legend entry, it is possible\n        to pass an iterable of legend artists followed by an iterable of\n        legend labels respectively::\n\n           legend((line1, line2, line3), ('label1', 'label2', 'label3'))\n\n        Parameters\n        ----------\n\n        loc : int or string or pair of floats, default: 'upper right'\n            The location of the legend. Possible codes are:\n\n                ===============   =============\n                Location String   Location Code\n                ===============   =============\n                'best'            0\n                'upper right'     1\n                'upper left'      2\n                'lower left'      3\n                'lower right'     4\n                'right'           5\n                'center left'     6\n                'center right'    7\n                'lower center'    8\n                'upper center'    9\n                'center'          10\n                ===============   =============\n\n\n            Alternatively can be a 2-tuple giving ``x, y`` of the lower-left\n            corner of the legend in axes coordinates (in which case\n            ``bbox_to_anchor`` will be ignored).\n\n        bbox_to_anchor : `~.BboxBase` or pair of floats\n            Specify any arbitrary location for the legend in `bbox_transform`\n            coordinates (default Axes coordinates).\n\n            For example, to put the legend's upper right hand corner in the\n            center of the axes the following keywords can be used::\n\n               loc='upper right', bbox_to_anchor=(0.5, 0.5)\n\n        ncol : integer\n            The number of columns that the legend has. Default is 1.\n\n        prop : None or :class:`matplotlib.font_manager.FontProperties` or dict\n            The font properties of the legend. If None (default), the current\n            :data:`matplotlib.rcParams` will be used.\n\n        fontsize : int or float or {'xx-small', 'x-small', 'small', 'medium', \\\n'large', 'x-large', 'xx-large'}\n            Controls the font size of the legend. If the value is numeric the\n            size will be the absolute font size in points. String values are\n            relative to the current default font size. This argument is only\n            used if `prop` is not specified.\n\n        numpoints : None or int\n            The number of marker points in the legend when creating a legend\n            entry for a line/:class:`matplotlib.lines.Line2D`.\n            Default is ``None`` which will take the value from the\n            ``legend.numpoints`` :data:`rcParam<matplotlib.rcParams>`.\n\n        scatterpoints : None or int\n            The number of marker points in the legend when creating a legend\n            entry for a scatter plot/\n            :class:`matplotlib.collections.PathCollection`.\n            Default is ``None`` which will take the value from the\n            ``legend.scatterpoints`` :data:`rcParam<matplotlib.rcParams>`.\n\n        scatteryoffsets : iterable of floats\n            The vertical offset (relative to the font size) for the markers\n            created for a scatter plot legend entry. 0.0 is at the base the\n            legend text, and 1.0 is at the top. To draw all markers at the\n            same height, set to ``[0.5]``. Default ``[0.375, 0.5, 0.3125]``.\n\n        markerscale : None or int or float\n            The relative size of legend markers compared with the originally\n            drawn ones. Default is ``None`` which will take the value from\n            the ``legend.markerscale`` :data:`rcParam <matplotlib.rcParams>`.\n\n        markerfirst : bool\n            If *True*, legend marker is placed to the left of the legend label.\n            If *False*, legend marker is placed to the right of the legend\n            label.\n            Default is *True*.\n\n        frameon : None or bool\n            Control whether the legend should be drawn on a patch (frame).\n            Default is ``None`` which will take the value from the\n            ``legend.frameon`` :data:`rcParam<matplotlib.rcParams>`.\n\n        fancybox : None or bool\n            Control whether round edges should be enabled around\n            the :class:`~matplotlib.patches.FancyBboxPatch` which\n            makes up the legend's background.\n            Default is ``None`` which will take the value from the\n            ``legend.fancybox`` :data:`rcParam<matplotlib.rcParams>`.\n\n        shadow : None or bool\n            Control whether to draw a shadow behind the legend.\n            Default is ``None`` which will take the value from the\n            ``legend.shadow`` :data:`rcParam<matplotlib.rcParams>`.\n\n        framealpha : None or float\n            Control the alpha transparency of the legend's background.\n            Default is ``None`` which will take the value from the\n            ``legend.framealpha`` :data:`rcParam<matplotlib.rcParams>`.\n            If shadow is activated and framealpha is ``None`` the\n            default value is being ignored.\n\n        facecolor : None or \"inherit\" or a color spec\n            Control the legend's background color.\n            Default is ``None`` which will take the value from the\n            ``legend.facecolor`` :data:`rcParam<matplotlib.rcParams>`.\n            If ``\"inherit\"``, it will take the ``axes.facecolor``\n            :data:`rcParam<matplotlib.rcParams>`.\n\n        edgecolor : None or \"inherit\" or a color spec\n            Control the legend's background patch edge color.\n            Default is ``None`` which will take the value from the\n            ``legend.edgecolor`` :data:`rcParam<matplotlib.rcParams>`.\n            If ``\"inherit\"``, it will take the ``axes.edgecolor``\n            :data:`rcParam<matplotlib.rcParams>`.\n\n        mode : {\"expand\", None}\n            If `mode` is set to ``\"expand\"`` the legend will be horizontally\n            expanded to fill the axes area (or `bbox_to_anchor` if defines\n            the legend's size).\n\n        bbox_transform : None or :class:`matplotlib.transforms.Transform`\n            The transform for the bounding box (`bbox_to_anchor`). For a value\n            of ``None`` (default) the Axes'\n            :data:`~matplotlib.axes.Axes.transAxes` transform will be used.\n\n        title : str or None\n            The legend's title. Default is no title (``None``).\n\n        borderpad : float or None\n            The fractional whitespace inside the legend border.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.borderpad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        labelspacing : float or None\n            The vertical space between the legend entries.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.labelspacing`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handlelength : float or None\n            The length of the legend handles.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.handlelength`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handletextpad : float or None\n            The pad between the legend handle and text.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.handletextpad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        borderaxespad : float or None\n            The pad between the axes and legend border.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.borderaxespad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        columnspacing : float or None\n            The spacing between columns.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.columnspacing`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handler_map : dict or None\n            The custom dictionary mapping instances or types to a legend\n            handler. This `handler_map` updates the default handler map\n            found at :func:`matplotlib.legend.Legend.get_legend_handler_map`.\n\n        Returns\n        -------\n\n        :class:`matplotlib.legend.Legend` instance\n\n        Notes\n        -----\n\n        Not all kinds of artist are supported by the legend command. See\n        :ref:`sphx_glr_tutorials_intermediate_legend_guide.py` for details.\n\n        Examples\n        --------\n\n        .. plot:: gallery/api/legend.py\n\n        \"\"\"\n        handles, labels, extra_args, kwargs = mlegend._parse_legend_args(\n                [self],\n                *args,\n                **kwargs)\n        if len(extra_args):\n            raise TypeError('legend only accepts two non-keyword arguments')\n        self.legend_ = mlegend.Legend(self, handles, labels, **kwargs)\n        self.legend_._remove_method = lambda h: setattr(self, 'legend_', None)\n        return self.legend_"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "legend",
          "class_name": "Figure",
          "code": "def legend(self, *args, **kwargs):\n        \"\"\"\n        Place a legend on the figure.\n\n        To make a legend from existing artists on every axes::\n\n          legend()\n\n        To make a legend for a list of lines and labels::\n\n          legend( (line1, line2, line3),\n                  ('label1', 'label2', 'label3'),\n                  loc='upper right')\n\n        These can also be specified by keyword::\n\n          legend(handles=(line1, line2, line3),\n                labels=('label1', 'label2', 'label3'),\n                loc='upper right')\n\n        Parameters\n        ----------\n\n        loc : int or string or pair of floats, default: 'upper right'\n            The location of the legend. Possible codes are:\n\n                ===============   =============\n                Location String   Location Code\n                ===============   =============\n                'best'            0\n                'upper right'     1\n                'upper left'      2\n                'lower left'      3\n                'lower right'     4\n                'right'           5\n                'center left'     6\n                'center right'    7\n                'lower center'    8\n                'upper center'    9\n                'center'          10\n                ===============   =============\n\n\n            Alternatively can be a 2-tuple giving ``x, y`` of the lower-left\n            corner of the legend in axes coordinates (in which case\n            ``bbox_to_anchor`` will be ignored).\n\n        bbox_to_anchor : `~.BboxBase` or pair of floats\n            Specify any arbitrary location for the legend in `bbox_transform`\n            coordinates (default Axes coordinates).\n\n            For example, to put the legend's upper right hand corner in the\n            center of the axes the following keywords can be used::\n\n               loc='upper right', bbox_to_anchor=(0.5, 0.5)\n\n        ncol : integer\n            The number of columns that the legend has. Default is 1.\n\n        prop : None or :class:`matplotlib.font_manager.FontProperties` or dict\n            The font properties of the legend. If None (default), the current\n            :data:`matplotlib.rcParams` will be used.\n\n        fontsize : int or float or {'xx-small', 'x-small', 'small', 'medium', \\\n'large', 'x-large', 'xx-large'}\n            Controls the font size of the legend. If the value is numeric the\n            size will be the absolute font size in points. String values are\n            relative to the current default font size. This argument is only\n            used if `prop` is not specified.\n\n        numpoints : None or int\n            The number of marker points in the legend when creating a legend\n            entry for a line/:class:`matplotlib.lines.Line2D`.\n            Default is ``None`` which will take the value from the\n            ``legend.numpoints`` :data:`rcParam<matplotlib.rcParams>`.\n\n        scatterpoints : None or int\n            The number of marker points in the legend when creating a legend\n            entry for a scatter plot/\n            :class:`matplotlib.collections.PathCollection`.\n            Default is ``None`` which will take the value from the\n            ``legend.scatterpoints`` :data:`rcParam<matplotlib.rcParams>`.\n\n        scatteryoffsets : iterable of floats\n            The vertical offset (relative to the font size) for the markers\n            created for a scatter plot legend entry. 0.0 is at the base the\n            legend text, and 1.0 is at the top. To draw all markers at the\n            same height, set to ``[0.5]``. Default ``[0.375, 0.5, 0.3125]``.\n\n        markerscale : None or int or float\n            The relative size of legend markers compared with the originally\n            drawn ones. Default is ``None`` which will take the value from\n            the ``legend.markerscale`` :data:`rcParam <matplotlib.rcParams>`.\n\n        markerfirst : bool\n            If *True*, legend marker is placed to the left of the legend label.\n            If *False*, legend marker is placed to the right of the legend\n            label.\n            Default is *True*.\n\n        frameon : None or bool\n            Control whether the legend should be drawn on a patch (frame).\n            Default is ``None`` which will take the value from the\n            ``legend.frameon`` :data:`rcParam<matplotlib.rcParams>`.\n\n        fancybox : None or bool\n            Control whether round edges should be enabled around\n            the :class:`~matplotlib.patches.FancyBboxPatch` which\n            makes up the legend's background.\n            Default is ``None`` which will take the value from the\n            ``legend.fancybox`` :data:`rcParam<matplotlib.rcParams>`.\n\n        shadow : None or bool\n            Control whether to draw a shadow behind the legend.\n            Default is ``None`` which will take the value from the\n            ``legend.shadow`` :data:`rcParam<matplotlib.rcParams>`.\n\n        framealpha : None or float\n            Control the alpha transparency of the legend's background.\n            Default is ``None`` which will take the value from the\n            ``legend.framealpha`` :data:`rcParam<matplotlib.rcParams>`.\n            If shadow is activated and framealpha is ``None`` the\n            default value is being ignored.\n\n        facecolor : None or \"inherit\" or a color spec\n            Control the legend's background color.\n            Default is ``None`` which will take the value from the\n            ``legend.facecolor`` :data:`rcParam<matplotlib.rcParams>`.\n            If ``\"inherit\"``, it will take the ``axes.facecolor``\n            :data:`rcParam<matplotlib.rcParams>`.\n\n        edgecolor : None or \"inherit\" or a color spec\n            Control the legend's background patch edge color.\n            Default is ``None`` which will take the value from the\n            ``legend.edgecolor`` :data:`rcParam<matplotlib.rcParams>`.\n            If ``\"inherit\"``, it will take the ``axes.edgecolor``\n            :data:`rcParam<matplotlib.rcParams>`.\n\n        mode : {\"expand\", None}\n            If `mode` is set to ``\"expand\"`` the legend will be horizontally\n            expanded to fill the axes area (or `bbox_to_anchor` if defines\n            the legend's size).\n\n        bbox_transform : None or :class:`matplotlib.transforms.Transform`\n            The transform for the bounding box (`bbox_to_anchor`). For a value\n            of ``None`` (default) the Axes'\n            :data:`~matplotlib.axes.Axes.transAxes` transform will be used.\n\n        title : str or None\n            The legend's title. Default is no title (``None``).\n\n        borderpad : float or None\n            The fractional whitespace inside the legend border.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.borderpad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        labelspacing : float or None\n            The vertical space between the legend entries.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.labelspacing`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handlelength : float or None\n            The length of the legend handles.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.handlelength`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handletextpad : float or None\n            The pad between the legend handle and text.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.handletextpad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        borderaxespad : float or None\n            The pad between the axes and legend border.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.borderaxespad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        columnspacing : float or None\n            The spacing between columns.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.columnspacing`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handler_map : dict or None\n            The custom dictionary mapping instances or types to a legend\n            handler. This `handler_map` updates the default handler map\n            found at :func:`matplotlib.legend.Legend.get_legend_handler_map`.\n\n        Returns\n        -------\n        :class:`matplotlib.legend.Legend` instance\n\n        Notes\n        -----\n        Not all kinds of artist are supported by the legend command. See\n        :ref:`sphx_glr_tutorials_intermediate_legend_guide.py` for details.\n        \"\"\"\n\n        handles, labels, extra_args, kwargs = mlegend._parse_legend_args(\n                self.axes,\n                *args,\n                **kwargs)\n        # check for third arg\n        if len(extra_args):\n            # cbook.warn_deprecated(\n            #     \"2.1\",\n            #     \"Figure.legend will accept no more than two \"\n            #     \"positional arguments in the future.  Use \"\n            #     \"'fig.legend(handles, labels, loc=location)' \"\n            #     \"instead.\")\n            # kwargs['loc'] = extra_args[0]\n            # extra_args = extra_args[1:]\n            pass\n        l = mlegend.Legend(self, handles, labels, *extra_args, **kwargs)\n        self.legends.append(l)\n        l._remove_method = lambda h: self.legends.remove(h)\n        self.stale = True\n        return l"
        },
        {
          "file": "lib/matplotlib/legend.py",
          "type": "function",
          "name": "_update_bbox_to_anchor",
          "class_name": "DraggableLegend",
          "code": "def _update_bbox_to_anchor(self, loc_in_canvas):\n\n        tr = self.legend.axes.transAxes\n        loc_in_bbox = tr.transform_point(loc_in_canvas)\n\n        self.legend.set_bbox_to_anchor(loc_in_bbox)"
        },
        {
          "file": "lib/matplotlib/legend.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Legend",
          "code": "def __init__(self, parent, handles, labels,\n                 loc=None,\n                 numpoints=None,    # the number of points in the legend line\n                 markerscale=None,  # the relative size of legend markers\n                                    # vs. original\n                 markerfirst=True,  # controls ordering (left-to-right) of\n                                    # legend marker and label\n                 scatterpoints=None,    # number of scatter points\n                 scatteryoffsets=None,\n                 prop=None,          # properties for the legend texts\n                 fontsize=None,        # keyword to set font size directly\n\n                 # spacing & pad defined as a fraction of the font-size\n                 borderpad=None,      # the whitespace inside the legend border\n                 labelspacing=None,   # the vertical space between the legend\n                                      # entries\n                 handlelength=None,   # the length of the legend handles\n                 handleheight=None,   # the height of the legend handles\n                 handletextpad=None,  # the pad between the legend handle\n                                      # and text\n                 borderaxespad=None,  # the pad between the axes and legend\n                                      # border\n                 columnspacing=None,  # spacing between columns\n\n                 ncol=1,     # number of columns\n                 mode=None,  # mode for horizontal distribution of columns.\n                             # None, \"expand\"\n\n                 fancybox=None,  # True use a fancy box, false use a rounded\n                                 # box, none use rc\n                 shadow=None,\n                 title=None,  # set a title for the legend\n\n                 framealpha=None,  # set frame alpha\n                 edgecolor=None,  # frame patch edgecolor\n                 facecolor=None,  # frame patch facecolor\n\n                 bbox_to_anchor=None,  # bbox that the legend will be anchored.\n                 bbox_transform=None,  # transform for the bbox\n                 frameon=None,  # draw frame\n                 handler_map=None,\n                 ):\n        \"\"\"\n        - *parent*: the artist that contains the legend\n        - *handles*: a list of artists (lines, patches) to be added to the\n                      legend\n        - *labels*: a list of strings to label the legend\n\n        Parameters\n        ----------\n\n        loc : int or string or pair of floats, default: 'upper right'\n            The location of the legend. Possible codes are:\n\n                ===============   =============\n                Location String   Location Code\n                ===============   =============\n                'best'            0\n                'upper right'     1\n                'upper left'      2\n                'lower left'      3\n                'lower right'     4\n                'right'           5\n                'center left'     6\n                'center right'    7\n                'lower center'    8\n                'upper center'    9\n                'center'          10\n                ===============   =============\n\n\n            Alternatively can be a 2-tuple giving ``x, y`` of the lower-left\n            corner of the legend in axes coordinates (in which case\n            ``bbox_to_anchor`` will be ignored).\n\n        bbox_to_anchor : `~.BboxBase` or pair of floats\n            Specify any arbitrary location for the legend in `bbox_transform`\n            coordinates (default Axes coordinates).\n\n            For example, to put the legend's upper right hand corner in the\n            center of the axes the following keywords can be used::\n\n               loc='upper right', bbox_to_anchor=(0.5, 0.5)\n\n        ncol : integer\n            The number of columns that the legend has. Default is 1.\n\n        prop : None or :class:`matplotlib.font_manager.FontProperties` or dict\n            The font properties of the legend. If None (default), the current\n            :data:`matplotlib.rcParams` will be used.\n\n        fontsize : int or float or {'xx-small', 'x-small', 'small', 'medium', \\\n'large', 'x-large', 'xx-large'}\n            Controls the font size of the legend. If the value is numeric the\n            size will be the absolute font size in points. String values are\n            relative to the current default font size. This argument is only\n            used if `prop` is not specified.\n\n        numpoints : None or int\n            The number of marker points in the legend when creating a legend\n            entry for a line/:class:`matplotlib.lines.Line2D`.\n            Default is ``None`` which will take the value from the\n            ``legend.numpoints`` :data:`rcParam<matplotlib.rcParams>`.\n\n        scatterpoints : None or int\n            The number of marker points in the legend when creating a legend\n            entry for a scatter plot/\n            :class:`matplotlib.collections.PathCollection`.\n            Default is ``None`` which will take the value from the\n            ``legend.scatterpoints`` :data:`rcParam<matplotlib.rcParams>`.\n\n        scatteryoffsets : iterable of floats\n            The vertical offset (relative to the font size) for the markers\n            created for a scatter plot legend entry. 0.0 is at the base the\n            legend text, and 1.0 is at the top. To draw all markers at the\n            same height, set to ``[0.5]``. Default ``[0.375, 0.5, 0.3125]``.\n\n        markerscale : None or int or float\n            The relative size of legend markers compared with the originally\n            drawn ones. Default is ``None`` which will take the value from\n            the ``legend.markerscale`` :data:`rcParam <matplotlib.rcParams>`.\n\n        markerfirst : bool\n            If *True*, legend marker is placed to the left of the legend label.\n            If *False*, legend marker is placed to the right of the legend\n            label.\n            Default is *True*.\n\n        frameon : None or bool\n            Control whether the legend should be drawn on a patch (frame).\n            Default is ``None`` which will take the value from the\n            ``legend.frameon`` :data:`rcParam<matplotlib.rcParams>`.\n\n        fancybox : None or bool\n            Control whether round edges should be enabled around\n            the :class:`~matplotlib.patches.FancyBboxPatch` which\n            makes up the legend's background.\n            Default is ``None`` which will take the value from the\n            ``legend.fancybox`` :data:`rcParam<matplotlib.rcParams>`.\n\n        shadow : None or bool\n            Control whether to draw a shadow behind the legend.\n            Default is ``None`` which will take the value from the\n            ``legend.shadow`` :data:`rcParam<matplotlib.rcParams>`.\n\n        framealpha : None or float\n            Control the alpha transparency of the legend's background.\n            Default is ``None`` which will take the value from the\n            ``legend.framealpha`` :data:`rcParam<matplotlib.rcParams>`.\n            If shadow is activated and framealpha is ``None`` the\n            default value is being ignored.\n\n        facecolor : None or \"inherit\" or a color spec\n            Control the legend's background color.\n            Default is ``None`` which will take the value from the\n            ``legend.facecolor`` :data:`rcParam<matplotlib.rcParams>`.\n            If ``\"inherit\"``, it will take the ``axes.facecolor``\n            :data:`rcParam<matplotlib.rcParams>`.\n\n        edgecolor : None or \"inherit\" or a color spec\n            Control the legend's background patch edge color.\n            Default is ``None`` which will take the value from the\n            ``legend.edgecolor`` :data:`rcParam<matplotlib.rcParams>`.\n            If ``\"inherit\"``, it will take the ``axes.edgecolor``\n            :data:`rcParam<matplotlib.rcParams>`.\n\n        mode : {\"expand\", None}\n            If `mode` is set to ``\"expand\"`` the legend will be horizontally\n            expanded to fill the axes area (or `bbox_to_anchor` if defines\n            the legend's size).\n\n        bbox_transform : None or :class:`matplotlib.transforms.Transform`\n            The transform for the bounding box (`bbox_to_anchor`). For a value\n            of ``None`` (default) the Axes'\n            :data:`~matplotlib.axes.Axes.transAxes` transform will be used.\n\n        title : str or None\n            The legend's title. Default is no title (``None``).\n\n        borderpad : float or None\n            The fractional whitespace inside the legend border.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.borderpad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        labelspacing : float or None\n            The vertical space between the legend entries.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.labelspacing`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handlelength : float or None\n            The length of the legend handles.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.handlelength`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handletextpad : float or None\n            The pad between the legend handle and text.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.handletextpad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        borderaxespad : float or None\n            The pad between the axes and legend border.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.borderaxespad`` :data:`rcParam<matplotlib.rcParams>`.\n\n        columnspacing : float or None\n            The spacing between columns.\n            Measured in font-size units.\n            Default is ``None`` which will take the value from the\n            ``legend.columnspacing`` :data:`rcParam<matplotlib.rcParams>`.\n\n        handler_map : dict or None\n            The custom dictionary mapping instances or types to a legend\n            handler. This `handler_map` updates the default handler map\n            found at :func:`matplotlib.legend.Legend.get_legend_handler_map`.\n\n        Notes\n        -----\n\n        Users can specify any arbitrary location for the legend using the\n        *bbox_to_anchor* keyword argument. bbox_to_anchor can be an instance\n        of BboxBase(or its derivatives) or a tuple of 2 or 4 floats.\n        See :meth:`set_bbox_to_anchor` for more detail.\n\n        The legend location can be specified by setting *loc* with a tuple of\n        2 floats, which is interpreted as the lower-left corner of the legend\n        in the normalized axes coordinate.\n        \"\"\"\n        # local import only to avoid circularity\n        from matplotlib.axes import Axes\n        from matplotlib.figure import Figure\n\n        Artist.__init__(self)\n\n        if prop is None:\n            if fontsize is not None:\n                self.prop = FontProperties(size=fontsize)\n            else:\n                self.prop = FontProperties(size=rcParams[\"legend.fontsize\"])\n        elif isinstance(prop, dict):\n            self.prop = FontProperties(**prop)\n            if \"size\" not in prop:\n                self.prop.set_size(rcParams[\"legend.fontsize\"])\n        else:\n            self.prop = prop\n\n        self._fontsize = self.prop.get_size_in_points()\n\n        self.texts = []\n        self.legendHandles = []\n        self._legend_title_box = None\n\n        #: A dictionary with the extra handler mappings for this Legend\n        #: instance.\n        self._custom_handler_map = handler_map\n\n        locals_view = locals()\n        for name in [\"numpoints\", \"markerscale\", \"shadow\", \"columnspacing\",\n                     \"scatterpoints\", \"handleheight\", 'borderpad',\n                     'labelspacing', 'handlelength', 'handletextpad',\n                     'borderaxespad']:\n            if locals_view[name] is None:\n                value = rcParams[\"legend.\" + name]\n            else:\n                value = locals_view[name]\n            setattr(self, name, value)\n        del locals_view\n        # trim handles and labels if illegal label...\n        for label, handle in zip(labels[:], handles[:]):\n                if (isinstance(label, six.string_types) and\n                        label.startswith('_')):\n                    warnings.warn('The handle {!r} has a label of {!r} which '\n                                  'cannot be automatically added to the '\n                                  'legend.'.format(handle, label))\n                    labels.remove(label)\n                    handles.remove(handle)\n\n        handles = list(handles)\n        if len(handles) < 2:\n            ncol = 1\n        self._ncol = ncol\n\n        if self.numpoints <= 0:\n            raise ValueError(\"numpoints must be > 0; it was %d\" % numpoints)\n\n        # introduce y-offset for handles of the scatter plot\n        if scatteryoffsets is None:\n            self._scatteryoffsets = np.array([3. / 8., 4. / 8., 2.5 / 8.])\n        else:\n            self._scatteryoffsets = np.asarray(scatteryoffsets)\n        reps = self.scatterpoints // len(self._scatteryoffsets) + 1\n        self._scatteryoffsets = np.tile(self._scatteryoffsets,\n                                        reps)[:self.scatterpoints]\n\n        # _legend_box is an OffsetBox instance that contains all\n        # legend items and will be initialized from _init_legend_box()\n        # method.\n        self._legend_box = None\n\n        if isinstance(parent, Axes):\n            self.isaxes = True\n            self.axes = parent\n            self.set_figure(parent.figure)\n        elif isinstance(parent, Figure):\n            self.isaxes = False\n            self.set_figure(parent)\n        else:\n            raise TypeError(\"Legend needs either Axes or Figure as parent\")\n        self.parent = parent\n\n        if loc is None:\n            loc = rcParams[\"legend.loc\"]\n            if not self.isaxes and loc in [0, 'best']:\n                loc = 'upper right'\n        if isinstance(loc, six.string_types):\n            if loc not in self.codes:\n                if self.isaxes:\n                    warnings.warn('Unrecognized location \"%s\". Falling back '\n                                  'on \"best\"; valid locations are\\n\\t%s\\n'\n                                  % (loc, '\\n\\t'.join(self.codes)))\n                    loc = 0\n                else:\n                    warnings.warn('Unrecognized location \"%s\". Falling back '\n                                  'on \"upper right\"; '\n                                  'valid locations are\\n\\t%s\\n'\n                                  % (loc, '\\n\\t'.join(self.codes)))\n                    loc = 1\n            else:\n                loc = self.codes[loc]\n        if not self.isaxes and loc == 0:\n            warnings.warn('Automatic legend placement (loc=\"best\") not '\n                          'implemented for figure legend. '\n                          'Falling back on \"upper right\".')\n            loc = 1\n\n        self._mode = mode\n        self.set_bbox_to_anchor(bbox_to_anchor, bbox_transform)\n\n        # We use FancyBboxPatch to draw a legend frame. The location\n        # and size of the box will be updated during the drawing time.\n\n        if facecolor is None:\n            facecolor = rcParams[\"legend.facecolor\"]\n        if facecolor == 'inherit':\n            facecolor = rcParams[\"axes.facecolor\"]\n\n        if edgecolor is None:\n            edgecolor = rcParams[\"legend.edgecolor\"]\n        if edgecolor == 'inherit':\n            edgecolor = rcParams[\"axes.edgecolor\"]\n\n        self.legendPatch = FancyBboxPatch(\n            xy=(0.0, 0.0), width=1., height=1.,\n            facecolor=facecolor,\n            edgecolor=edgecolor,\n            mutation_scale=self._fontsize,\n            snap=True\n            )\n\n        # The width and height of the legendPatch will be set (in the\n        # draw()) to the length that includes the padding. Thus we set\n        # pad=0 here.\n        if fancybox is None:\n            fancybox = rcParams[\"legend.fancybox\"]\n\n        if fancybox:\n            self.legendPatch.set_boxstyle(\"round\", pad=0,\n                                          rounding_size=0.2)\n        else:\n            self.legendPatch.set_boxstyle(\"square\", pad=0)\n\n        self._set_artist_props(self.legendPatch)\n\n        self._drawFrame = frameon\n        if frameon is None:\n            self._drawFrame = rcParams[\"legend.frameon\"]\n\n        # init with null renderer\n        self._init_legend_box(handles, labels, markerfirst)\n\n        # If shadow is activated use framealpha if not\n        # explicitly passed. See Issue 8943\n        if framealpha is None:\n            if shadow:\n                self.get_frame().set_alpha(1)\n            else:\n                self.get_frame().set_alpha(rcParams[\"legend.framealpha\"])\n        else:\n            self.get_frame().set_alpha(framealpha)\n\n        self._loc = loc\n        self.set_title(title)\n        self._last_fontsize_points = self._fontsize\n        self._draggable = None"
        },
        {
          "file": "lib/matplotlib/pyplot.py",
          "type": "function",
          "name": "pause",
          "class_name": null,
          "code": "def pause(interval):\n    \"\"\"\n    Pause for *interval* seconds.\n\n    If there is an active figure, it will be updated and displayed before the\n    pause, and the GUI event loop (if any) will run during the pause.\n\n    This can be used for crude animation.  For more complex animation, see\n    :mod:`matplotlib.animation`.\n\n    Note\n    ----\n    This function is experimental; its behavior may be changed or extended in a\n    future release.\n    \"\"\"\n    manager = _pylab_helpers.Gcf.get_active()\n    if manager is not None:\n        canvas = manager.canvas\n        if canvas.figure.stale:\n            canvas.draw_idle()\n        show(block=False)\n        canvas.start_event_loop(interval)\n    else:\n        time.sleep(interval)"
        }
      ]
    },
    {
      "pr_number": 15140,
      "pr_title": "Fix ScalarFormatter formatting of masked values",
      "pr_body": "Fixes #15103. Could probably do with a test. Possibly an API change since previously a masked value would be formatted as `nan`?",
      "issue_id": 15103,
      "issue_title": "Colorbar for imshow messes interactive cursor with masked data",
      "issue_body": "### Bug report\r\n\r\nAdding a colorbar when the mappable is the result of imshow for a masked array changes the behavior of the interactive z-data cursor and triggers a UserWarning.\r\n\r\n**Code for reproduction**\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\nmask = np.zeros((4, 4), bool)\r\nmask[:2] = True\r\nA = np.ma.array(np.random.rand(4, 4), mask=mask)\r\n\r\nfig, ax = plt.subplots(1, 1)\r\nim = ax.imshow(A)\r\nfig.colorbar(im)\r\n```\r\n\r\n**Actual outcome**\r\n```\r\n~/.anaconda3/lib/python3.7/site-packages/matplotlib/ticker.py:632: UserWarning: Warning: converting a masked element to nan.\r\n  return '%-12g' % value\r\n```\r\n![Screenshot from 2019-08-22 18-22-23](https://user-images.githubusercontent.com/52320542/63531938-e9d5ef00-c509-11e9-83ef-7ce9bc3f3c59.png)\r\n\r\n**Expected outcome**\r\nBefore adding the colorbar, the zdata cursor shows a blank (`[]`).\r\n\r\n**Matplotlib version**\r\n  * Operating system: Linux\r\n  * Matplotlib version: 3.1.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.3\r\n  * IPython version: 7.2.0\r\n  * Other libraries: \r\nMatplotlib installed from pip.\r\n\r\n\r\n\r\n",
      "issue_closed_at": "2019-08-28T03:46:25Z",
      "base_commit": "40dfc353aa66b93fd0fbc55ca1f51701202c0549",
      "changes": [
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "format_data_short",
          "class_name": "LogitFormatter",
          "code": "def format_data_short(self, value):\n        \"\"\"\n        Return a short formatted string representation of a number.\n        \"\"\"\n        # thresholds choosen for use scienfic notation if and only if exponent\n        # is less or equal than -2.\n        if value < 0.1:\n            return \"{:e}\".format(value)\n        if value < 0.9:\n            return \"{:f}\".format(value)\n        return \"1-{:e}\".format(1 - value)"
        }
      ]
    },
    {
      "pr_number": 6110,
      "pr_title": "Fixes matplotlib/matplotlib#1235",
      "pr_body": "Credit to @dashed, fixes #1235, supercedes #2857 \n\nThe problem is that _find_best_position does not receive positions of (transformed) points from _auto_legend_data. We attempted to fix it by adding the offsets of points, after preparing the points and transforming them, and calling legendBox.count_contains on offsets.\n",
      "issue_id": 1235,
      "issue_title": "Legend placement bug",
      "issue_body": "This script produces a legend that covers the red point:\n\n``` python\nimport matplotlib.pyplot as plt\n\ncolors = ['b','g','r']\nfor n in range(3):\n    plt.scatter([n,],[n,],color=colors[n])\nplt.legend(['foo','foo','foo'],loc='best')\nplt.gca().set_xlim(-0.5, 2.2)\nplt.gca().set_ylim(-0.5, 2.2)\nplt.show()\n```\n",
      "issue_closed_at": "2016-03-07T14:11:57Z",
      "base_commit": "963e51d4df44fa58bf89840b19f2d0f12a0e605a",
      "changes": [
        {
          "file": "lib/matplotlib/legend.py",
          "type": "function",
          "name": "_auto_legend_data",
          "class_name": "Legend",
          "code": "def _auto_legend_data(self):\n        \"\"\"\n        Returns list of vertices and extents covered by the plot.\n\n        Returns a two long list.\n\n        First element is a list of (x, y) vertices (in\n        display-coordinates) covered by all the lines and line\n        collections, in the legend's handles.\n\n        Second element is a list of bounding boxes for all the patches in\n        the legend's handles.\n        \"\"\"\n        # should always hold because function is only called internally\n        assert self.isaxes\n\n        ax = self.parent\n        bboxes = []\n        lines = []\n\n        for handle in ax.lines:\n            assert isinstance(handle, Line2D)\n            path = handle.get_path()\n            trans = handle.get_transform()\n            tpath = trans.transform_path(path)\n            lines.append(tpath)\n\n        for handle in ax.patches:\n            assert isinstance(handle, Patch)\n\n            if isinstance(handle, Rectangle):\n                transform = handle.get_data_transform()\n                bboxes.append(handle.get_bbox().transformed(transform))\n            else:\n                transform = handle.get_transform()\n                bboxes.append(handle.get_path().get_extents(transform))\n\n        try:\n            vertices = np.concatenate([l.vertices for l in lines])\n        except ValueError:\n            vertices = np.array([])\n\n        return [vertices, bboxes, lines]"
        },
        {
          "file": "lib/matplotlib/legend.py",
          "type": "function",
          "name": "_auto_legend_data",
          "class_name": "Legend",
          "code": "def _auto_legend_data(self):\n        \"\"\"\n        Returns list of vertices and extents covered by the plot.\n\n        Returns a two long list.\n\n        First element is a list of (x, y) vertices (in\n        display-coordinates) covered by all the lines and line\n        collections, in the legend's handles.\n\n        Second element is a list of bounding boxes for all the patches in\n        the legend's handles.\n        \"\"\"\n        # should always hold because function is only called internally\n        assert self.isaxes\n\n        ax = self.parent\n        bboxes = []\n        lines = []\n\n        for handle in ax.lines:\n            assert isinstance(handle, Line2D)\n            path = handle.get_path()\n            trans = handle.get_transform()\n            tpath = trans.transform_path(path)\n            lines.append(tpath)\n\n        for handle in ax.patches:\n            assert isinstance(handle, Patch)\n\n            if isinstance(handle, Rectangle):\n                transform = handle.get_data_transform()\n                bboxes.append(handle.get_bbox().transformed(transform))\n            else:\n                transform = handle.get_transform()\n                bboxes.append(handle.get_path().get_extents(transform))\n\n        try:\n            vertices = np.concatenate([l.vertices for l in lines])\n        except ValueError:\n            vertices = np.array([])\n\n        return [vertices, bboxes, lines]"
        },
        {
          "file": "lib/matplotlib/legend.py",
          "type": "function",
          "name": "_find_best_position",
          "class_name": "Legend",
          "code": "def _find_best_position(self, width, height, renderer, consider=None):\n        \"\"\"\n        Determine the best location to place the legend.\n\n        `consider` is a list of (x, y) pairs to consider as a potential\n        lower-left corner of the legend. All are display coords.\n        \"\"\"\n        # should always hold because function is only called internally\n        assert self.isaxes\n\n        verts, bboxes, lines = self._auto_legend_data()\n\n        bbox = Bbox.from_bounds(0, 0, width, height)\n        if consider is None:\n            consider = [self._get_anchored_bbox(x, bbox,\n                                                self.get_bbox_to_anchor(),\n                                                renderer)\n                        for x in range(1, len(self.codes))]\n\n#       tx, ty = self.legendPatch.get_x(), self.legendPatch.get_y()\n\n        candidates = []\n        for l, b in consider:\n            legendBox = Bbox.from_bounds(l, b, width, height)\n            badness = 0\n            # XXX TODO: If markers are present, it would be good to\n            # take their into account when checking vertex overlaps in\n            # the next line.\n            badness = legendBox.count_contains(verts)\n            badness += legendBox.count_overlaps(bboxes)\n            for line in lines:\n                # FIXME: the following line is ill-suited for lines\n                # that 'spiral' around the center, because the bbox\n                # may intersect with the legend even if the line\n                # itself doesn't. One solution would be to break up\n                # the line into its straight-segment components, but\n                # this may (or may not) result in a significant\n                # slowdown if lines with many vertices are present.\n                if line.intersects_bbox(legendBox):\n                    badness += 1\n\n            ox, oy = l, b\n            if badness == 0:\n                return ox, oy\n\n            candidates.append((badness, (l, b)))\n\n        # rather than use min() or list.sort(), do this so that we are assured\n        # that in the case of two equal badnesses, the one first considered is\n        # returned.\n        # NOTE: list.sort() is stable.But leave as it is for now. -JJL\n        minCandidate = candidates[0]\n        for candidate in candidates:\n            if candidate[0] < minCandidate[0]:\n                minCandidate = candidate\n\n        ox, oy = minCandidate[1]\n\n        return ox, oy"
        },
        {
          "file": "lib/matplotlib/legend.py",
          "type": "function",
          "name": "_find_best_position",
          "class_name": "Legend",
          "code": "def _find_best_position(self, width, height, renderer, consider=None):\n        \"\"\"\n        Determine the best location to place the legend.\n\n        `consider` is a list of (x, y) pairs to consider as a potential\n        lower-left corner of the legend. All are display coords.\n        \"\"\"\n        # should always hold because function is only called internally\n        assert self.isaxes\n\n        verts, bboxes, lines = self._auto_legend_data()\n\n        bbox = Bbox.from_bounds(0, 0, width, height)\n        if consider is None:\n            consider = [self._get_anchored_bbox(x, bbox,\n                                                self.get_bbox_to_anchor(),\n                                                renderer)\n                        for x in range(1, len(self.codes))]\n\n#       tx, ty = self.legendPatch.get_x(), self.legendPatch.get_y()\n\n        candidates = []\n        for l, b in consider:\n            legendBox = Bbox.from_bounds(l, b, width, height)\n            badness = 0\n            # XXX TODO: If markers are present, it would be good to\n            # take their into account when checking vertex overlaps in\n            # the next line.\n            badness = legendBox.count_contains(verts)\n            badness += legendBox.count_overlaps(bboxes)\n            for line in lines:\n                # FIXME: the following line is ill-suited for lines\n                # that 'spiral' around the center, because the bbox\n                # may intersect with the legend even if the line\n                # itself doesn't. One solution would be to break up\n                # the line into its straight-segment components, but\n                # this may (or may not) result in a significant\n                # slowdown if lines with many vertices are present.\n                if line.intersects_bbox(legendBox):\n                    badness += 1\n\n            ox, oy = l, b\n            if badness == 0:\n                return ox, oy\n\n            candidates.append((badness, (l, b)))\n\n        # rather than use min() or list.sort(), do this so that we are assured\n        # that in the case of two equal badnesses, the one first considered is\n        # returned.\n        # NOTE: list.sort() is stable.But leave as it is for now. -JJL\n        minCandidate = candidates[0]\n        for candidate in candidates:\n            if candidate[0] < minCandidate[0]:\n                minCandidate = candidate\n\n        ox, oy = minCandidate[1]\n\n        return ox, oy"
        }
      ]
    },
    {
      "pr_number": 22935,
      "pr_title": "Backport PR #22002: Fix TkAgg memory leaks and test for memory growth regressions",
      "pr_body": "FIX: TkAgg memory leaks and test for memory growth regressions (#22002)\n\ntkinter variables get cleaned up with normal `destroy` and `gc` semantics but tkinter's implementation of trace is effectively global and keeps the callback object alive until the trace is removed.\n\nAdditionally extend and clean up the tests.\n\nCloses #20490\n\nCo-authored-by: Elliott Sales de Andrade <quantum.analyst@gmail.com>\n(cherry picked from commit 1a016f0395c3a8d87b632a47d813db2491863164)\n",
      "issue_id": 20490,
      "issue_title": "Memory leaks on matplotlib 3.4.2 (and 3.4.0)",
      "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\r\n**Bug summary**\r\n\r\nMultiple different memory related errors when running as a part of script creating and saving lots of figures.\r\n\r\n\r\n**Code for reproduction**\r\n\r\nSorry, could not create a minimal example that reproduces this. Just wanted to leave a note. \r\n\r\n\r\n\r\n**Actual outcome**\r\n\r\nThere are multiple different errors that happen. Running the same script will produce different results randomly.\r\n\r\n<!--The output produced by the above code, which may be a screenshot, console output, etc.-->\r\n\r\n**Example 1**\r\nJust prints this. No Exceptions, nothing. Script stops running.\r\n```\r\nunable to alloc 4320000 bytes\r\n```\r\n\r\n**Example 2** \r\n\r\n```python\r\n  File \"c:\\myapp\\calculations\\myscript.py\", line 422, in create_figure\r\n    fig, ax = plt.subplots(nrows=1, figsize=(16, 9))\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\_api\\deprecation.py\", line 471, in wrapper\r\n    return func(*args, **kwargs)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\pyplot.py\", line 1439, in subplots\r\n    fig = figure(**fig_kw)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\pyplot.py\", line 797, in figure\r\n    manager = new_figure_manager(\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\pyplot.py\", line 316, in new_figure_manager\r\n    return _backend_mod.new_figure_manager(*args, **kwargs)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backend_bases.py\", line 3545, in new_figure_manager\r\n    return cls.new_figure_manager_given_figure(num, fig)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backends\\_backend_tk.py\", line 899, in new_figure_manager_given_figure\r\n    canvas = cls.FigureCanvas(figure, master=window)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\_api\\deprecation.py\", line 431, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backends\\_backend_tk.py\", line 174, in __init__\r\n    self._tkphoto = tk.PhotoImage(\r\n  File \"C:\\Python\\Python 3.8.6-32\\lib\\tkinter\\__init__.py\", line 4061, in __init__\r\n    Image.__init__(self, 'photo', name, cnf, master, **kw)\r\n  File \"C:\\Python\\Python 3.8.6-32\\lib\\tkinter\\__init__.py\", line 4006, in __init__\r\n    self.tk.call(('image', 'create', imgtype, name,) + options)\r\n_tkinter.TclError: not enough free memory for image buffer\r\n```\r\n\r\n**Example 3**\r\n\r\n```python\r\n  File \"c:\\myapp\\calculations\\myscript.py\", line 467, in create_some_figures\r\n    plt.savefig(self.folder / \"myfig.png\")\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\pyplot.py\", line 966, in savefig\r\n    res = fig.savefig(*args, **kwargs)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\figure.py\", line 3005, in savefig\r\n    self.canvas.print_figure(fname, **kwargs)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backend_bases.py\", line 2255, in print_figure\r\n    result = print_method(\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backend_bases.py\", line 1669, in wrapper\r\n    return func(*args, **kwargs)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backends\\backend_agg.py\", line 508, in print_png\r\n    FigureCanvasAgg.draw(self)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backends\\backend_agg.py\", line 401, in draw\r\n    self.renderer = self.get_renderer(cleared=True)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backends\\backend_agg.py\", line 417, in get_renderer\r\n    self.renderer = RendererAgg(w, h, self.figure.dpi)\r\n  File \"C:\\Python\\venvs\\adiapp\\lib\\site-packages\\matplotlib\\backends\\backend_agg.py\", line 91, in __init__\r\n    self._renderer = _RendererAgg(int(width), int(height), dpi)\r\nMemoryError: In RendererAgg: Out of memory\r\n```\r\n\r\n**Expected outcome**\r\n\r\n<!--A description of the expected outcome from the code snippet-->\r\n<!--If this used to work in an earlier version of Matplotlib, please note the version it used to work on-->\r\nNo errors. \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:  Windows 10\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): 3.4.2\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg \r\n  * Python version: 32-bit 3.8.6 \r\n\r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n\r\n## Workaround\r\n\r\nAs a side note, tried on **version 3.3.4** and everything works. Does not work on 3.4.0. I'm calling \r\n\r\n```python\r\n        plt.close(\"all\")\r\n        plt.close()\r\n        gc.collect()\r\n```\r\nafter every `plt.savefig()` just to be sure.\r\n",
      "issue_closed_at": "2022-04-28T23:48:45Z",
      "base_commit": "a841f5f51473cb035e5c29fa7665da59bc4ed4d9",
      "changes": [
        {
          "file": "lib/matplotlib/_pylab_helpers.py",
          "type": "function",
          "name": "destroy",
          "class_name": "Gcf",
          "code": "def destroy(cls, num):\n        \"\"\"\n        Destroy manager *num* -- either a manager instance or a manager number.\n\n        In the interactive backends, this is bound to the window \"destroy\" and\n        \"delete\" events.\n\n        It is recommended to pass a manager instance, to avoid confusion when\n        two managers share the same number.\n        \"\"\"\n        if all(hasattr(num, attr) for attr in [\"num\", \"destroy\"]):\n            manager = num\n            if cls.figs.get(manager.num) is manager:\n                cls.figs.pop(manager.num)\n        else:\n            try:\n                manager = cls.figs.pop(num)\n            except KeyError:\n                return\n        if hasattr(manager, \"_cidgcf\"):\n            manager.canvas.mpl_disconnect(manager._cidgcf)\n        manager.destroy()\n        gc.collect(1)"
        },
        {
          "file": "lib/matplotlib/backends/_backend_tk.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ConfigureSubplotsTk",
          "code": "def __init__(self, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.window = None"
        },
        {
          "file": "lib/matplotlib/backends/_backend_tk.py",
          "type": "function",
          "name": "destroy",
          "class_name": "ConfigureSubplotsTk",
          "code": "def destroy(self, *args, **kwargs):\n        if self.window is not None:\n            self.window.destroy()\n            self.window = None"
        }
      ]
    }
  ]
}