{
  "Selected_candidate": {
    "pr_number": 23332,
    "pr_title": "Validate Text linespacing on input.",
    "pr_body": "... and also tweak test_two_2line_texts to not set up an axes (just a\r\nfigure), as it doesn't need that.\r\n\r\nCloses #19223 (transform_rotates_text is no longer a problem since #19575).\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n**Tests and Styling**\r\n- [ ] Has pytest style unit tests (and `pytest` passes).\r\n- [ ] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n\r\n**Documentation**\r\n- [ ] New features are documented, with examples if plot related.\r\n- [ ] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [ ] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\r\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\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 main, 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": 19223,
    "issue_title": "Certain non-hashable parameters to text() give cryptic error messages",
    "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\nPer the title.  See https://discourse.matplotlib.org/t/cryptic-exception-when-axes-parameters-are-not-as-expected/ as well.\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug.\r\nPlease make sure to minimize the number of dependencies required, and provide\r\nany necessary plotted data.\r\nAvoid using threads, as Matplotlib is (explicitly) not thread-safe.-->\r\n\r\n```python\r\nfigtext(.5, .5, \"foo\", rotation=[90])\r\n# or\r\nfigtext(.5, .5, \"foo\", transform_rotates_text=[0])\r\n# or\r\nfigtext(.5, .5, \"foo\", linespacing=[0])\r\n```\r\nall fail with\r\n```\r\nTypeError: unhashable type: 'list'\r\n```\r\n\r\n**Actual outcome**\r\n\r\nsee above\r\n\r\n**Expected outcome**\r\n\r\nError out in the setter instead.\r\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Operating system: linux\r\n  * Matplotlib version: (`import matplotlib; print(matplotlib.__version__)`): master\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): any\r\n  * Python version: 38\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
    "issue_closed_at": "2022-06-24T22:25:33Z",
    "base_commit": "6e5a5415ddc513606eac5c116fde492370a3f7f6",
    "changes": [
      {
        "file": "lib/matplotlib/text.py",
        "type": "line",
        "name": "line 5",
        "code": "import functools\nimport logging\nimport math\nimport numbers\nimport weakref\n\nimport numpy as np"
      },
      {
        "file": "lib/matplotlib/text.py",
        "type": "function",
        "name": "__init__",
        "class_name": "Annotation",
        "code": "def __init__(self, text, xy,\n                 xytext=None,\n                 xycoords='data',\n                 textcoords=None,\n                 arrowprops=None,\n                 annotation_clip=None,\n                 **kwargs):\n        \"\"\"\n        Annotate the point *xy* with text *text*.\n\n        In the simplest form, the text is placed at *xy*.\n\n        Optionally, the text can be displayed in another position *xytext*.\n        An arrow pointing from the text to the annotated point *xy* can then\n        be added by defining *arrowprops*.\n\n        Parameters\n        ----------\n        text : str\n            The text of the annotation.\n\n        xy : (float, float)\n            The point *(x, y)* to annotate. The coordinate system is determined\n            by *xycoords*.\n\n        xytext : (float, float), default: *xy*\n            The position *(x, y)* to place the text at. The coordinate system\n            is determined by *textcoords*.\n\n        xycoords : str or `.Artist` or `.Transform` or callable or \\\n(float, float), default: 'data'\n\n            The coordinate system that *xy* is given in. The following types\n            of values are supported:\n\n            - One of the following strings:\n\n              ==================== ============================================\n              Value                Description\n              ==================== ============================================\n              'figure points'      Points from the lower left of the figure\n              'figure pixels'      Pixels from the lower left of the figure\n              'figure fraction'    Fraction of figure from lower left\n              'subfigure points'   Points from the lower left of the subfigure\n              'subfigure pixels'   Pixels from the lower left of the subfigure\n              'subfigure fraction' Fraction of subfigure from lower left\n              'axes points'        Points from lower left corner of axes\n              'axes pixels'        Pixels from lower left corner of axes\n              'axes fraction'      Fraction of axes from lower left\n              'data'               Use the coordinate system of the object\n                                   being annotated (default)\n              'polar'              *(theta, r)* if not native 'data'\n                                   coordinates\n              ==================== ============================================\n\n              Note that 'subfigure pixels' and 'figure pixels' are the same\n              for the parent figure, so users who want code that is usable in\n              a subfigure can use 'subfigure pixels'.\n\n            - An `.Artist`: *xy* is interpreted as a fraction of the artist's\n              `~matplotlib.transforms.Bbox`. E.g. *(0, 0)* would be the lower\n              left corner of the bounding box and *(0.5, 1)* would be the\n              center top of the bounding box.\n\n            - A `.Transform` to transform *xy* to screen coordinates.\n\n            - A function with one of the following signatures::\n\n                def transform(renderer) -> Bbox\n                def transform(renderer) -> Transform\n\n              where *renderer* is a `.RendererBase` subclass.\n\n              The result of the function is interpreted like the `.Artist` and\n              `.Transform` cases above.\n\n            - A tuple *(xcoords, ycoords)* specifying separate coordinate\n              systems for *x* and *y*. *xcoords* and *ycoords* must each be\n              of one of the above described types.\n\n            See :ref:`plotting-guide-annotation` for more details.\n\n        textcoords : str or `.Artist` or `.Transform` or callable or \\\n(float, float), default: value of *xycoords*\n            The coordinate system that *xytext* is given in.\n\n            All *xycoords* values are valid as well as the following\n            strings:\n\n            =================   =========================================\n            Value               Description\n            =================   =========================================\n            'offset points'     Offset (in points) from the *xy* value\n            'offset pixels'     Offset (in pixels) from the *xy* value\n            =================   =========================================\n\n        arrowprops : dict, optional\n            The properties used to draw a `.FancyArrowPatch` arrow between the\n            positions *xy* and *xytext*.  Defaults to None, i.e. no arrow is\n            drawn.\n\n            For historical reasons there are two different ways to specify\n            arrows, \"simple\" and \"fancy\":\n\n            **Simple arrow:**\n\n            If *arrowprops* does not contain the key 'arrowstyle' the\n            allowed keys are:\n\n            ==========   ======================================================\n            Key          Description\n            ==========   ======================================================\n            width        The width of the arrow in points\n            headwidth    The width of the base of the arrow head in points\n            headlength   The length of the arrow head in points\n            shrink       Fraction of total length to shrink from both ends\n            ?            Any key to :class:`matplotlib.patches.FancyArrowPatch`\n            ==========   ======================================================\n\n            The arrow is attached to the edge of the text box, the exact\n            position (corners or centers) depending on where it's pointing to.\n\n            **Fancy arrow:**\n\n            This is used if 'arrowstyle' is provided in the *arrowprops*.\n\n            Valid keys are the following `~matplotlib.patches.FancyArrowPatch`\n            parameters:\n\n            ===============  ==================================================\n            Key              Description\n            ===============  ==================================================\n            arrowstyle       the arrow style\n            connectionstyle  the connection style\n            relpos           see below; default is (0.5, 0.5)\n            patchA           default is bounding box of the text\n            patchB           default is None\n            shrinkA          default is 2 points\n            shrinkB          default is 2 points\n            mutation_scale   default is text size (in points)\n            mutation_aspect  default is 1.\n            ?                any key for :class:`matplotlib.patches.PathPatch`\n            ===============  ==================================================\n\n            The exact starting point position of the arrow is defined by\n            *relpos*. It's a tuple of relative coordinates of the text box,\n            where (0, 0) is the lower left corner and (1, 1) is the upper\n            right corner. Values <0 and >1 are supported and specify points\n            outside the text box. By default (0.5, 0.5) the starting point is\n            centered in the text box.\n\n        annotation_clip : bool or None, default: None\n            Whether to clip (i.e. not draw) the annotation when the annotation\n            point *xy* is outside the axes area.\n\n            - If *True*, the annotation will be clipped when *xy* is outside\n              the axes.\n            - If *False*, the annotation will always be drawn.\n            - If *None*, the annotation will be clipped when *xy* is outside\n              the axes and *xycoords* is 'data'.\n\n        **kwargs\n            Additional kwargs are passed to `~matplotlib.text.Text`.\n\n        Returns\n        -------\n        `.Annotation`\n\n        See Also\n        --------\n        :ref:`plotting-guide-annotation`\n\n        \"\"\"\n        _AnnotationBase.__init__(self,\n                                 xy,\n                                 xycoords=xycoords,\n                                 annotation_clip=annotation_clip)\n        # warn about wonky input data\n        if (xytext is None and\n                textcoords is not None and\n                textcoords != xycoords):\n            _api.warn_external(\"You have used the `textcoords` kwarg, but \"\n                               \"not the `xytext` kwarg.  This can lead to \"\n                               \"surprising results.\")\n\n        # clean up textcoords and assign default\n        if textcoords is None:\n            textcoords = self.xycoords\n        self._textcoords = textcoords\n\n        # cleanup xytext defaults\n        if xytext is None:\n            xytext = self.xy\n        x, y = xytext\n\n        self.arrowprops = arrowprops\n        if arrowprops is not None:\n            arrowprops = arrowprops.copy()\n            if \"arrowstyle\" in arrowprops:\n                self._arrow_relpos = arrowprops.pop(\"relpos\", (0.5, 0.5))\n            else:\n                # modified YAArrow API to be used with FancyArrowPatch\n                for key in [\n                        'width', 'headwidth', 'headlength', 'shrink', 'frac']:\n                    arrowprops.pop(key, None)\n            self.arrow_patch = FancyArrowPatch((0, 0), (1, 1), **arrowprops)\n        else:\n            self.arrow_patch = None\n\n        # Must come last, as some kwargs may be propagated to arrow_patch.\n        Text.__init__(self, x, y, text, **kwargs)"
      },
      {
        "file": "lib/matplotlib/text.py",
        "type": "function",
        "name": "set_linespacing",
        "class_name": "Text",
        "code": "def set_linespacing(self, spacing):\n        \"\"\"\n        Set the line spacing as a multiple of the font size.\n\n        The default line spacing is 1.2.\n\n        Parameters\n        ----------\n        spacing : float (multiple of font size)\n        \"\"\"\n        self._linespacing = spacing\n        self.stale = True"
      },
      {
        "file": "lib/matplotlib/text.py",
        "type": "function",
        "name": "set_rotation",
        "class_name": "Text",
        "code": "def set_rotation(self, s):\n        \"\"\"\n        Set the rotation of the text.\n\n        Parameters\n        ----------\n        s : float or {'vertical', 'horizontal'}\n            The rotation angle in degrees in mathematically positive direction\n            (counterclockwise). 'horizontal' equals 0, 'vertical' equals 90.\n        \"\"\"\n        if isinstance(s, numbers.Real):\n            self._rotation = float(s) % 360\n        elif cbook._str_equal(s, 'horizontal') or s is None:\n            self._rotation = 0.\n        elif cbook._str_equal(s, 'vertical'):\n            self._rotation = 90.\n        else:\n            raise ValueError(\"rotation must be 'vertical', 'horizontal' or \"\n                             f\"a number, not {s}\")\n        self.stale = True"
      }
    ]
  },
  "Justification": "Candidate A exhibits the most relevant structural similarity to the CURRENT bug, as it involves error messages directly originating from matplotlib function calls. Both the CURRENT bug and Candidate A's bug reports involve incorrect handling of arguments leading to exceptions in matplotlib's rendering engine. Additionally, while the types of errors are different, the context of performance and interaction with matplotlib's behavior during plotting is aligned, offering insights into how types and properties may inadvertently lead to similar AttributeErrors, thus potentially guiding the debugging process for the CURRENT bug.",
  "instance_id": "matplotlib__matplotlib-25442",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-03-12T21:58:08Z",
  "problem_statement": "[Bug]: Attribute Error combining matplotlib 3.7.1 and mplcursor on data selection\n### Bug summary\r\n\r\nIf you combine mplcursor and matplotlib 3.7.1, you'll get an `AttributeError: 'NoneType' object has no attribute 'canvas'` after clicking a few data points. Henceforth, selecting a new data point will trigger the same traceback. Otherwise, it works fine. \r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\nimport mplcursors as mpl\r\n\r\nx = np.arange(1, 11)    \r\ny1 = x\r\n\r\nplt.scatter(x,y1)\r\n\r\nmpl.cursor()\r\nplt.show()\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n```\r\nTraceback (most recent call last):\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\cbook\\__init__.py\", line 304, in process\r\n    func(*args, **kwargs)\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1550, in on_release\r\n    if self._check_still_parented() and self.got_artist:\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1560, in _check_still_parented\r\n    self.disconnect()\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1568, in disconnect\r\n    self.canvas.mpl_disconnect(cid)\r\n  File \"C:\\Users\\MrAni\\Python\\miniconda3\\lib\\site-packages\\matplotlib\\offsetbox.py\", line 1517, in <lambda>\r\n    canvas = property(lambda self: self.ref_artist.figure.canvas)\r\nAttributeError: 'NoneType' object has no attribute 'canvas'\r\n```\r\n\r\n### Expected outcome\r\n\r\nNo terminal output\r\n\r\n### Additional information\r\n\r\nUsing matplotlib 3.7.0 or lower works fine. Using a conda install or pip install doesn't affect the output. \r\n\r\n### Operating system\r\n\r\nWindows 11 and Windwos 10 \r\n\r\n### Matplotlib Version\r\n\r\n3.7.1\r\n\r\n### Matplotlib Backend\r\n\r\nQtAgg\r\n\r\n### Python version\r\n\r\n3.9.16\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\nconda\n",
  "patch": "diff --git a/lib/matplotlib/offsetbox.py b/lib/matplotlib/offsetbox.py\n--- a/lib/matplotlib/offsetbox.py\n+++ b/lib/matplotlib/offsetbox.py\n@@ -1500,16 +1500,23 @@ def __init__(self, ref_artist, use_blit=False):\n             ref_artist.set_picker(True)\n         self.got_artist = False\n         self._use_blit = use_blit and self.canvas.supports_blit\n-        self.cids = [\n-            self.canvas.callbacks._connect_picklable(\n-                'pick_event', self.on_pick),\n-            self.canvas.callbacks._connect_picklable(\n-                'button_release_event', self.on_release),\n+        callbacks = ref_artist.figure._canvas_callbacks\n+        self._disconnectors = [\n+            functools.partial(\n+                callbacks.disconnect, callbacks._connect_picklable(name, func))\n+            for name, func in [\n+                (\"pick_event\", self.on_pick),\n+                (\"button_release_event\", self.on_release),\n+                (\"motion_notify_event\", self.on_motion),\n+            ]\n         ]\n \n     # A property, not an attribute, to maintain picklability.\n     canvas = property(lambda self: self.ref_artist.figure.canvas)\n \n+    cids = property(lambda self: [\n+        disconnect.args[0] for disconnect in self._disconnectors[:2]])\n+\n     def on_motion(self, evt):\n         if self._check_still_parented() and self.got_artist:\n             dx = evt.x - self.mouse_x\n@@ -1536,16 +1543,12 @@ def on_pick(self, evt):\n                 self.ref_artist.draw(\n                     self.ref_artist.figure._get_renderer())\n                 self.canvas.blit()\n-            self._c1 = self.canvas.callbacks._connect_picklable(\n-                \"motion_notify_event\", self.on_motion)\n             self.save_offset()\n \n     def on_release(self, event):\n         if self._check_still_parented() and self.got_artist:\n             self.finalize_offset()\n             self.got_artist = False\n-            self.canvas.mpl_disconnect(self._c1)\n-\n             if self._use_blit:\n                 self.ref_artist.set_animated(False)\n \n@@ -1558,14 +1561,8 @@ def _check_still_parented(self):\n \n     def disconnect(self):\n         \"\"\"Disconnect the callbacks.\"\"\"\n-        for cid in self.cids:\n-            self.canvas.mpl_disconnect(cid)\n-        try:\n-            c1 = self._c1\n-        except AttributeError:\n-            pass\n-        else:\n-            self.canvas.mpl_disconnect(c1)\n+        for disconnector in self._disconnectors:\n+            disconnector()\n \n     def save_offset(self):\n         pass\n"
}