{
  "instance_id": "matplotlib__matplotlib-23563",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-08-05T15:52:55Z",
  "problem_statement": "[Bug]: 'Line3D' object has no attribute '_verts3d'\n### Bug summary\n\nI use matplotlib 3D to visualize some lines in 3D. When I first run the following code, the code can run right. But, if I give `x_s_0[n]` a numpy array, it will report the error 'input operand has more dimensions than allowed by the axis remapping'. The point is when next I give  `x_s_0[n]` and other variables an int number, the AttributeError: 'Line3D' object has no attribute '_verts3d' will appear and can not be fixed whatever I change the variables or delete them. The error can be only fixed when I restart the kernel of ipython console. I don't know why it happens, so I come here for help.\n\n### Code for reproduction\n\n```python\nx_s_0 = np.array(['my int number list'])\r\nx_e_0 = np.array(['my int number list'])\r\ny_s_0 = np.array(['my int number list'])\r\ny_e_0 = np.array(['my int number list'])\r\nz_s_0 = np.array(['my int number list'])\r\nz_e_0 = np.array(['my int number list'])\r\n\r\nfig = plt.figure()\r\n        ax = fig.gca(projection='3d')\r\n        ax.view_init(elev=90, azim=0)\r\n        ax.set_zlim3d(-10, 10)\r\n        clr_list = 'r-'\r\n\r\n        for n in range(np.size(z_s_0, axis=0)):\r\n            ax.plot([int(x_s_0[n]), int(x_e_0[n])],\r\n                    [int(y_s_0[n]), int(y_e_0[n])],\r\n                    [int(z_s_0[n]), int(z_e_0[n])], clr_list)\r\n\r\n        plt.xlabel('x')\r\n        plt.ylabel('y')\r\n        # ax.zlabel('z')\r\n        plt.title('90-0')\r\n        plt.show()\n```\n\n\n### Actual outcome\n\nTraceback (most recent call last):\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/IPython/core/interactiveshell.py\", line 3444, in run_code\r\n    exec(code_obj, self.user_global_ns, self.user_ns)\r\n  File \"<ipython-input-80-e04907066a16>\", line 20, in <module>\r\n    plt.show()\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/pyplot.py\", line 368, in show\r\n    return _backend_mod.show(*args, **kwargs)\r\n  File \"/home/hanyaning/.pycharm_helpers/pycharm_matplotlib_backend/backend_interagg.py\", line 29, in __call__\r\n    manager.show(**kwargs)\r\n  File \"/home/hanyaning/.pycharm_helpers/pycharm_matplotlib_backend/backend_interagg.py\", line 112, in show\r\n    self.canvas.show()\r\n  File \"/home/hanyaning/.pycharm_helpers/pycharm_matplotlib_backend/backend_interagg.py\", line 68, in show\r\n    FigureCanvasAgg.draw(self)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/backends/backend_agg.py\", line 436, in draw\r\n    self.figure.draw(self.renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/artist.py\", line 73, in draw_wrapper\r\n    result = draw(artist, renderer, *args, **kwargs)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/artist.py\", line 50, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/figure.py\", line 2803, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/artist.py\", line 50, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/mpl_toolkits/mplot3d/axes3d.py\", line 469, in draw\r\n    super().draw(renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/artist.py\", line 50, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/axes/_base.py\", line 3082, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/matplotlib/artist.py\", line 50, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/hanyaning/anaconda3/envs/SBeA/lib/python3.8/site-packages/mpl_toolkits/mplot3d/art3d.py\", line 215, in draw\r\n    xs3d, ys3d, zs3d = self._verts3d\r\nAttributeError: 'Line3D' object has no attribute '_verts3d'\n\n### Expected outcome\n\nSome 3D lines\n\n### Additional information\n\n_No response_\n\n### Operating system\n\nLocal: windows + pycharm, Remote: Ubuntu 20.04\n\n### Matplotlib Version\n\n3.5.0\n\n### Matplotlib Backend\n\nmodule://backend_interagg\n\n### Python version\n\n3.8.12\n\n### Jupyter version\n\n_No response_\n\n### Installation\n\npip\n",
  "patch": "diff --git a/lib/mpl_toolkits/mplot3d/art3d.py b/lib/mpl_toolkits/mplot3d/art3d.py\n--- a/lib/mpl_toolkits/mplot3d/art3d.py\n+++ b/lib/mpl_toolkits/mplot3d/art3d.py\n@@ -171,6 +171,7 @@ def __init__(self, xs, ys, zs, *args, **kwargs):\n     def set_3d_properties(self, zs=0, zdir='z'):\n         xs = self.get_xdata()\n         ys = self.get_ydata()\n+        zs = cbook._to_unmasked_float_array(zs).ravel()\n         zs = np.broadcast_to(zs, len(xs))\n         self._verts3d = juggle_axes(xs, ys, zs, zdir)\n         self.stale = True\n",
  "similar_bug_items": [
    {
      "pr_number": 9534,
      "pr_title": "Fix webagg",
      "pr_body": "Adjust where the various js files are stored.\r\n\r\nFixes #9380.",
      "issue_id": 9380,
      "issue_title": "Cannot import pyplot. NameError:  'FigureManagerWebAgg' is not defined",
      "issue_body": "\r\n### Bug report\r\n\r\nWhen using the webagg backend on 2.1.0, importing pyplot fails with a NameError.\r\n\r\n**Code for reproduction**\r\n```python\r\nimport matplotlib\r\nmatplotlib.use('WebAgg')\r\nfrom matplotlib import pyplot\r\n```\r\n\r\n**Actual outcome**\r\n\r\nI get a name error. traceback excerpt:\r\n```\r\n.../python3.6/site-packages/matplotlib/backends/__init__.py in pylab_setup(name)\r\n     58     # imports. 0 means only perform absolute imports.\r\n     59     backend_mod = __import__(backend_name, globals(), locals(),\r\n---> 60                              [backend_name], 0)\r\n     61 \r\n     62     # Things we pull in from all backends\r\n\r\n.../python3.6/site-packages/matplotlib/backends/backend_webagg.py in <module>()\r\n    325 \r\n    326 @_Backend.export\r\n--> 327 class _BackendWebAgg(_Backend):\r\n    328     FigureCanvas = FigureCanvasWebAgg\r\n    329     FigureManager = FigureManagerWebAgg\r\n\r\n.../python3.6/site-packages/matplotlib/backends/backend_webagg.py in _BackendWebAgg()\r\n    327 class _BackendWebAgg(_Backend):\r\n    328     FigureCanvas = FigureCanvasWebAgg\r\n--> 329     FigureManager = FigureManagerWebAgg\r\n    330 \r\n    331     @staticmethod\r\n\r\nNameError: name 'FigureManagerWebAgg' is not defined\r\n```\r\n\r\n\r\n**Matplotlib version**\r\n  * Operating system:  Mac OS 10.12.6\r\n  * Matplotlib version: 2.1.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): WebAgg\r\n  * Python version: Python 3.6.2 \r\n  * Jupyter version (if applicable): -\r\n  * Other libraries: \r\n\r\n\r\nI installed matplotlib via pip.\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": "2017-11-01T18:25:59Z",
      "base_commit": "473409e2ddea7ccacb4823ff4cfbf19759cf987a",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_nbagg.py",
          "type": "function",
          "name": "get_javascript",
          "class_name": "FigureManagerNbAgg",
          "code": "def get_javascript(cls, stream=None):\n        if stream is None:\n            output = io.StringIO()\n        else:\n            output = stream\n        super(FigureManagerNbAgg, cls).get_javascript(stream=output)\n        with io.open(os.path.join(\n                os.path.dirname(__file__),\n                \"web_backend\",\n                \"nbagg_mpl.js\"), encoding='utf8') as fd:\n            output.write(fd.read())\n        if stream is None:\n            return output.getvalue()"
        },
        {
          "file": "lib/matplotlib/backends/backend_webagg_core.py",
          "type": "function",
          "name": "get_javascript",
          "class_name": "FigureManagerWebAgg",
          "code": "def get_javascript(cls, stream=None):\n        if stream is None:\n            output = io.StringIO()\n        else:\n            output = stream\n\n        with io.open(os.path.join(\n                os.path.dirname(__file__),\n                \"web_backend\",\n                \"mpl.js\"), encoding='utf8') as fd:\n            output.write(fd.read())\n\n        toolitems = []\n        for name, tooltip, image, method in cls.ToolbarCls.toolitems:\n            if name is None:\n                toolitems.append(['', '', '', ''])\n            else:\n                toolitems.append([name, tooltip, image, method])\n        output.write(\"mpl.toolbar_items = {0};\\n\\n\".format(\n            json.dumps(toolitems)))\n\n        extensions = []\n        for filetype, ext in sorted(FigureCanvasWebAggCore.\n                                    get_supported_filetypes_grouped().\n                                    items()):\n            if not ext[0] == 'pgf':  # pgf does not support BytesIO\n                extensions.append(ext[0])\n        output.write(\"mpl.extensions = {0};\\n\\n\".format(\n            json.dumps(extensions)))\n\n        output.write(\"mpl.default_extension = {0};\".format(\n            json.dumps(FigureCanvasWebAggCore.get_default_filetype())))\n\n        if stream is None:\n            return output.getvalue()"
        }
      ]
    },
    {
      "pr_number": 12461,
      "pr_title": "FIX: make add_lines work with new colorbar",
      "pr_body": "## PR Summary\r\n\r\nCloses #12458\r\n\r\nSee below for a less pathological example...\r\n\r\n`cbar.add_lines` was not operational after new colorbar changes in 3.0.0.  \r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport matplotlib as mpl\r\nimport numpy as np\r\n\r\nfig = plt.figure()\r\nax = fig.add_axes([0.05, 0.8, 0.9, 0.15])\r\ncmap = mpl.cm.cool\r\nnorm = mpl.colors.Normalize(vmin=-4, vmax=4)\r\nlevels = (-1.0, 1.0, 2.0, 3.0)\r\n\r\ncb = mpl.colorbar.ColorbarBase(ax, cmap=cmap, norm=norm)\r\ncolors_bg = np.tile(list((1.0, 1.0, 1.0, 1.0)), (len(levels), 1))\r\ncb.add_lines(levels=levels, colors=colors_bg, linewidths=1)\r\n\r\n```\r\n\r\n## Before\r\n![cbarold](https://user-images.githubusercontent.com/1562854/46681303-ce871d00-cb9f-11e8-8acd-e5defedfe90c.png)\r\n\r\n\r\n## After\r\n\r\n![cbarnew](https://user-images.githubusercontent.com/1562854/46681265-b3b4a880-cb9f-11e8-8efd-87b24d565751.png)\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\nfig, ax = plt.subplots()\r\nX = np.random.rand(10, 10)*10000\r\npcm = ax.pcolormesh(X)\r\n# add 1000 to make colors visible...\r\ncont = ax.contour(X+1000)\r\ncb = fig.colorbar(pcm)\r\ncb.add_lines(cont)\r\nplt.show()\r\n```\r\n\r\n## Before\r\n\r\n![before](https://user-images.githubusercontent.com/1562854/46701766-b4683180-cbd5-11e8-8c18-d994ba47e396.png)\r\n\r\n## After\r\n![new](https://user-images.githubusercontent.com/1562854/46701741-9d294400-cbd5-11e8-9006-708b7ef51fdc.png)\r\n\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) 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": 12458,
      "issue_title": "add_lines misses lines for matplotlib.colorbar.ColorbarBase",
      "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\nFor matplotlib.colorbar.ColorbarBase(...) usage of add_lines(...) may yield missing lines/poorly adjusted lines.\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.pyplot as plt\r\nimport matplotlib as mpl\r\nimport numpy as np\r\n\r\nfig = plt.figure()\r\nax = fig.add_axes([0.05, 0.8, 0.9, 0.15])\r\ncmap = mpl.cm.cool\r\nnorm = mpl.colors.Normalize(vmin=-4, vmax=4)\r\nlevels = (-1.0, 1.0, 2.0, 3.0)\r\n\r\ncb = mpl.colorbar.ColorbarBase(ax, cmap=cmap, norm=norm)\r\ncolors_bg = np.tile(list((1.0, 1.0, 1.0, 1.0)), (len(levels), 1))\r\ncb.add_lines(levels=levels, colors=colors_bg, linewidths=7)\r\n\r\nplt.show()\r\n\r\n\r\n```\r\n\r\n**Expected outcome**\r\n\r\n4 white lines, crossing the colorbar.\r\nUsed to work on matplotlib 2.2.3!\r\n\r\n**Matplotlib version**\r\n  * Operating system: Ubuntu 18.04 LTS\r\n  * Matplotlib version: 3.0.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.0\r\n\r\nMiniconda install, fully updated.",
      "issue_closed_at": "2018-10-14T19:38:43Z",
      "base_commit": "f93222a555160bf967f02dae42cce39a03384774",
      "changes": [
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "add_lines",
          "class_name": "Colorbar",
          "code": "def add_lines(self, CS, erase=True):\n        '''\n        Add the lines from a non-filled\n        :class:`~matplotlib.contour.ContourSet` to the colorbar.\n\n        Set *erase* to False if these lines should be added to\n        any pre-existing lines.\n        '''\n        if not isinstance(CS, contour.ContourSet) or CS.filled:\n            raise ValueError('add_lines is only for a ContourSet of lines')\n        tcolors = [c[0] for c in CS.tcolors]\n        tlinewidths = [t[0] for t in CS.tlinewidths]\n        # The following was an attempt to get the colorbar lines\n        # to follow subsequent changes in the contour lines,\n        # but more work is needed: specifically, a careful\n        # look at event sequences, and at how\n        # to make one object track another automatically.\n        #tcolors = [col.get_colors()[0] for col in CS.collections]\n        #tlinewidths = [col.get_linewidth()[0] for lw in CS.collections]\n        ColorbarBase.add_lines(self, CS.levels, tcolors, tlinewidths,\n                               erase=erase)"
        }
      ]
    },
    {
      "pr_number": 15798,
      "pr_title": "Better default behavior for boxplots when rcParams['lines.marker'] is set",
      "pr_body": "## PR Summary\r\ncloses #15730\r\navoid using rcParams['lines.marker'] for boxplots.\r\n\r\n**Code snippet for reproducing the issue:**\r\n```\r\nplt.rcParams['lines.marker'] = 's'\r\nplt.boxplot(range(100))\r\n```\r\n**Before this PR:**\r\n![test](https://user-images.githubusercontent.com/15175620/69918581-ee337180-1441-11ea-9937-dfe8c5005946.png)\r\n\r\n**After this PR:**\r\n![test1](https://user-images.githubusercontent.com/15175620/69918585-f4295280-1441-11ea-9e35-bfc4b07c07d8.png)\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) 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-->\r\n",
      "issue_id": 15730,
      "issue_title": "Setting lines.marker = s in matplotlibrc also sets markers in boxplots",
      "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\nSetting this line\r\n    lines.marker      : s    ## the default marker\r\nin the matplotlibrc leads to an unexpected result:\r\n\r\n```\r\n%pylab inline\r\nboxplot(range(100))\r\n\r\n```\r\n\r\n![fig](https://user-images.githubusercontent.com/3391614/69190942-b0982580-0ade-11ea-903e-8f59e3f5e70d.jpg)\r\n",
      "issue_closed_at": "2019-12-30T18:06:35Z",
      "base_commit": "22ac46006fa8ea9dfb5b37941ffa7980f0447d8a",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_axes.py",
          "type": "function",
          "name": "bxp",
          "class_name": "Axes",
          "code": "def bxp(self, bxpstats, positions=None, widths=None, vert=True,\n            patch_artist=False, shownotches=False, showmeans=False,\n            showcaps=True, showbox=True, showfliers=True,\n            boxprops=None, whiskerprops=None, flierprops=None,\n            medianprops=None, capprops=None, meanprops=None,\n            meanline=False, manage_ticks=True, zorder=None):\n        \"\"\"\n        Drawing function for box and whisker plots.\n\n        Make a box and whisker plot for each column of *x* or each\n        vector in sequence *x*.  The box extends from the lower to\n        upper quartile values of the data, with a line at the median.\n        The whiskers extend from the box to show the range of the\n        data.  Flier points are those past the end of the whiskers.\n\n        Parameters\n        ----------\n        bxpstats : list of dicts\n          A list of dictionaries containing stats for each boxplot.\n          Required keys are:\n\n          - ``med``: The median (scalar float).\n\n          - ``q1``: The first quartile (25th percentile) (scalar\n            float).\n\n          - ``q3``: The third quartile (75th percentile) (scalar\n            float).\n\n          - ``whislo``: Lower bound of the lower whisker (scalar\n            float).\n\n          - ``whishi``: Upper bound of the upper whisker (scalar\n            float).\n\n          Optional keys are:\n\n          - ``mean``: The mean (scalar float). Needed if\n            ``showmeans=True``.\n\n          - ``fliers``: Data beyond the whiskers (sequence of floats).\n            Needed if ``showfliers=True``.\n\n          - ``cilo`` & ``cihi``: Lower and upper confidence intervals\n            about the median. Needed if ``shownotches=True``.\n\n          - ``label``: Name of the dataset (string). If available,\n            this will be used a tick label for the boxplot\n\n        positions : array-like, default: [1, 2, ..., n]\n          Sets the positions of the boxes. The ticks and limits\n          are automatically set to match the positions.\n\n        widths : array-like, default: None\n          Either a scalar or a vector and sets the width of each\n          box. The default is ``0.15*(distance between extreme\n          positions)``, clipped to no less than 0.15 and no more than\n          0.5.\n\n        vert : bool, default: True\n          If `True` (default), makes the boxes vertical.  If `False`,\n          makes horizontal boxes.\n\n        patch_artist : bool, default: False\n          If `False` produces boxes with the `.Line2D` artist.\n          If `True` produces boxes with the `~matplotlib.patches.Patch` artist.\n\n        shownotches : bool, default: False\n          If `False` (default), produces a rectangular box plot.\n          If `True`, will produce a notched box plot\n\n        showmeans : bool, default: False\n          If `True`, will toggle on the rendering of the means\n\n        showcaps  : bool, default: True\n          If `True`, will toggle on the rendering of the caps\n\n        showbox  : bool, default: True\n          If `True`, will toggle on the rendering of the box\n\n        showfliers : bool, default: True\n          If `True`, will toggle on the rendering of the fliers\n\n        boxprops : dict or None (default)\n          If provided, will set the plotting style of the boxes\n\n        whiskerprops : dict or None (default)\n          If provided, will set the plotting style of the whiskers\n\n        capprops : dict or None (default)\n          If provided, will set the plotting style of the caps\n\n        flierprops : dict or None (default)\n          If provided will set the plotting style of the fliers\n\n        medianprops : dict or None (default)\n          If provided, will set the plotting style of the medians\n\n        meanprops : dict or None (default)\n          If provided, will set the plotting style of the means\n\n        meanline : bool, default: False\n          If `True` (and *showmeans* is `True`), will try to render the mean\n          as a line spanning the full width of the box according to\n          *meanprops*. Not recommended if *shownotches* is also True.\n          Otherwise, means will be shown as points.\n\n        manage_ticks : bool, default: True\n          If True, the tick locations and labels will be adjusted to match the\n          boxplot positions.\n\n        zorder : scalar, default: None\n          The zorder of the resulting boxplot.\n\n        Returns\n        -------\n        result : dict\n          A dictionary mapping each component of the boxplot to a list\n          of the `.Line2D` instances created. That dictionary has the\n          following keys (assuming vertical boxplots):\n\n          - ``boxes``: the main body of the boxplot showing the\n            quartiles and the median's confidence intervals if\n            enabled.\n\n          - ``medians``: horizontal lines at the median of each box.\n\n          - ``whiskers``: the vertical lines extending to the most\n            extreme, non-outlier data points.\n\n          - ``caps``: the horizontal lines at the ends of the\n            whiskers.\n\n          - ``fliers``: points representing data that extend beyond\n            the whiskers (fliers).\n\n          - ``means``: points or lines representing the means.\n\n        Examples\n        --------\n        .. plot:: gallery/statistics/bxp.py\n\n        \"\"\"\n        # lists of artists to be output\n        whiskers = []\n        caps = []\n        boxes = []\n        medians = []\n        means = []\n        fliers = []\n\n        # empty list of xticklabels\n        datalabels = []\n\n        # Use default zorder if none specified\n        if zorder is None:\n            zorder = mlines.Line2D.zorder\n\n        zdelta = 0.1\n\n        def line_props_with_rcdefaults(subkey, explicit, zdelta=0):\n            d = {k.split('.')[-1]: v for k, v in rcParams.items()\n                 if k.startswith(f'boxplot.{subkey}')}\n            d['zorder'] = zorder + zdelta\n            if explicit is not None:\n                d.update(\n                    cbook.normalize_kwargs(explicit, mlines.Line2D._alias_map))\n            return d\n\n        # box properties\n        if patch_artist:\n            final_boxprops = dict(\n                linestyle=rcParams['boxplot.boxprops.linestyle'],\n                linewidth=rcParams['boxplot.boxprops.linewidth'],\n                edgecolor=rcParams['boxplot.boxprops.color'],\n                facecolor=('white' if rcParams['_internal.classic_mode'] else\n                           rcParams['patch.facecolor']),\n                zorder=zorder,\n            )\n            if boxprops is not None:\n                final_boxprops.update(\n                    cbook.normalize_kwargs(\n                        boxprops, mpatches.PathPatch._alias_map))\n        else:\n            final_boxprops = line_props_with_rcdefaults('boxprops', boxprops)\n        final_whiskerprops = line_props_with_rcdefaults(\n            'whiskerprops', whiskerprops)\n        final_capprops = line_props_with_rcdefaults(\n            'capprops', capprops)\n        final_flierprops = line_props_with_rcdefaults(\n            'flierprops', flierprops)\n        final_medianprops = line_props_with_rcdefaults(\n            'medianprops', medianprops, zdelta)\n        final_meanprops = line_props_with_rcdefaults(\n            'meanprops', meanprops, zdelta)\n        removed_prop = 'marker' if meanline else 'linestyle'\n        # Only remove the property if it's not set explicitly as a parameter.\n        if meanprops is None or removed_prop not in meanprops:\n            final_meanprops[removed_prop] = ''\n\n        def to_vc(xs, ys):\n            # convert arguments to verts and codes, append (0, 0) (ignored).\n            verts = np.append(np.column_stack([xs, ys]), [(0, 0)], 0)\n            codes = ([mpath.Path.MOVETO]\n                     + [mpath.Path.LINETO] * (len(verts) - 2)\n                     + [mpath.Path.CLOSEPOLY])\n            return verts, codes\n\n        def patch_list(xs, ys, **kwargs):\n            verts, codes = to_vc(xs, ys)\n            path = mpath.Path(verts, codes)\n            patch = mpatches.PathPatch(path, **kwargs)\n            self.add_artist(patch)\n            return [patch]\n\n        # vertical or horizontal plot?\n        if vert:\n            def doplot(*args, **kwargs):\n                return self.plot(*args, **kwargs)\n\n            def dopatch(xs, ys, **kwargs):\n                return patch_list(xs, ys, **kwargs)\n\n        else:\n            def doplot(*args, **kwargs):\n                shuffled = []\n                for i in range(0, len(args), 2):\n                    shuffled.extend([args[i + 1], args[i]])\n                return self.plot(*shuffled, **kwargs)\n\n            def dopatch(xs, ys, **kwargs):\n                xs, ys = ys, xs  # flip X, Y\n                return patch_list(xs, ys, **kwargs)\n\n        # input validation\n        N = len(bxpstats)\n        datashape_message = (\"List of boxplot statistics and `{0}` \"\n                             \"values must have same the length\")\n        # check position\n        if positions is None:\n            positions = list(range(1, N + 1))\n        elif len(positions) != N:\n            raise ValueError(datashape_message.format(\"positions\"))\n\n        positions = np.array(positions)\n        if len(positions) > 0 and not isinstance(positions[0], Number):\n            raise TypeError(\"positions should be an iterable of numbers\")\n\n        # width\n        if widths is None:\n            widths = [np.clip(0.15 * np.ptp(positions), 0.15, 0.5)] * N\n        elif np.isscalar(widths):\n            widths = [widths] * N\n        elif len(widths) != N:\n            raise ValueError(datashape_message.format(\"widths\"))\n\n        for pos, width, stats in zip(positions, widths, bxpstats):\n            # try to find a new label\n            datalabels.append(stats.get('label', pos))\n\n            # whisker coords\n            whisker_x = np.ones(2) * pos\n            whiskerlo_y = np.array([stats['q1'], stats['whislo']])\n            whiskerhi_y = np.array([stats['q3'], stats['whishi']])\n\n            # cap coords\n            cap_left = pos - width * 0.25\n            cap_right = pos + width * 0.25\n            cap_x = np.array([cap_left, cap_right])\n            cap_lo = np.ones(2) * stats['whislo']\n            cap_hi = np.ones(2) * stats['whishi']\n\n            # box and median coords\n            box_left = pos - width * 0.5\n            box_right = pos + width * 0.5\n            med_y = [stats['med'], stats['med']]\n\n            # notched boxes\n            if shownotches:\n                box_x = [box_left, box_right, box_right, cap_right, box_right,\n                         box_right, box_left, box_left, cap_left, box_left,\n                         box_left]\n                box_y = [stats['q1'], stats['q1'], stats['cilo'],\n                         stats['med'], stats['cihi'], stats['q3'],\n                         stats['q3'], stats['cihi'], stats['med'],\n                         stats['cilo'], stats['q1']]\n                med_x = cap_x\n\n            # plain boxes\n            else:\n                box_x = [box_left, box_right, box_right, box_left, box_left]\n                box_y = [stats['q1'], stats['q1'], stats['q3'], stats['q3'],\n                         stats['q1']]\n                med_x = [box_left, box_right]\n\n            # maybe draw the box:\n            if showbox:\n                if patch_artist:\n                    boxes.extend(dopatch(box_x, box_y, **final_boxprops))\n                else:\n                    boxes.extend(doplot(box_x, box_y, **final_boxprops))\n\n            # draw the whiskers\n            whiskers.extend(doplot(\n                whisker_x, whiskerlo_y, **final_whiskerprops\n            ))\n            whiskers.extend(doplot(\n                whisker_x, whiskerhi_y, **final_whiskerprops\n            ))\n\n            # maybe draw the caps:\n            if showcaps:\n                caps.extend(doplot(cap_x, cap_lo, **final_capprops))\n                caps.extend(doplot(cap_x, cap_hi, **final_capprops))\n\n            # draw the medians\n            medians.extend(doplot(med_x, med_y, **final_medianprops))\n\n            # maybe draw the means\n            if showmeans:\n                if meanline:\n                    means.extend(doplot(\n                        [box_left, box_right], [stats['mean'], stats['mean']],\n                        **final_meanprops\n                    ))\n                else:\n                    means.extend(doplot(\n                        [pos], [stats['mean']], **final_meanprops\n                    ))\n\n            # maybe draw the fliers\n            if showfliers:\n                # fliers coords\n                flier_x = np.full(len(stats['fliers']), pos, dtype=np.float64)\n                flier_y = stats['fliers']\n\n                fliers.extend(doplot(\n                    flier_x, flier_y, **final_flierprops\n                ))\n\n        if manage_ticks:\n            axis_name = \"x\" if vert else \"y\"\n            interval = getattr(self.dataLim, f\"interval{axis_name}\")\n            axis = getattr(self, f\"{axis_name}axis\")\n            positions = axis.convert_units(positions)\n            # The 0.5 additional padding ensures reasonable-looking boxes\n            # even when drawing a single box.  We set the sticky edge to\n            # prevent margins expansion, in order to match old behavior (back\n            # when separate calls to boxplot() would completely reset the axis\n            # limits regardless of what was drawn before).  The sticky edges\n            # are attached to the median lines, as they are always present.\n            interval[:] = (min(interval[0], min(positions) - .5),\n                           max(interval[1], max(positions) + .5))\n            for median, position in zip(medians, positions):\n                getattr(median.sticky_edges, axis_name).extend(\n                    [position - .5, position + .5])\n            # Modified from Axis.set_ticks and Axis.set_ticklabels.\n            locator = axis.get_major_locator()\n            if not isinstance(axis.get_major_locator(),\n                              mticker.FixedLocator):\n                locator = mticker.FixedLocator([])\n                axis.set_major_locator(locator)\n            locator.locs = np.array([*locator.locs, *positions])\n            formatter = axis.get_major_formatter()\n            if not isinstance(axis.get_major_formatter(),\n                              mticker.FixedFormatter):\n                formatter = mticker.FixedFormatter([])\n                axis.set_major_formatter(formatter)\n            formatter.seq = [*formatter.seq, *datalabels]\n\n            self._request_autoscale_view(\n                scalex=self._autoscaleXon, scaley=self._autoscaleYon)\n\n        return dict(whiskers=whiskers, caps=caps, boxes=boxes,\n                    medians=medians, fliers=fliers, means=means)"
        },
        {
          "file": "lib/matplotlib/axes/_axes.py",
          "type": "function",
          "name": "line_props_with_rcdefaults",
          "class_name": "Axes",
          "code": "def line_props_with_rcdefaults(subkey, explicit, zdelta=0):\n            d = {k.split('.')[-1]: v for k, v in rcParams.items()\n                 if k.startswith(f'boxplot.{subkey}')}\n            d['zorder'] = zorder + zdelta\n            if explicit is not None:\n                d.update(\n                    cbook.normalize_kwargs(explicit, mlines.Line2D._alias_map))\n            return d"
        }
      ]
    },
    {
      "pr_number": 17163,
      "pr_title": "Fix clipping of markers in PDF backend.",
      "pr_body": "The bbox only contains the points of the marker, but the line will extend outside by half the line width. This was handled before, except when the line is at an angle to the edge, as then the line outside is wider than half a line width. The maximum is at 45\u00b0, or \u221a2.\r\n\r\nThis fixes corners on 'v', '^', '<', '>', 'p', 'h', 'H', 'D', 'd', 'X'.\r\n\r\nFixes #9829.\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n- [x] Has Pytest style unit tests\r\n- [x] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) compliant\r\n- [n/a] New features are documented, with examples if plot related\r\n- [n/a] Documentation is sphinx and numpydoc compliant\r\n- [n/a] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [n/a] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way",
      "issue_id": 9829,
      "issue_title": "Vertices clipped for certain markers when plotting more than two points and saving as pdf",
      "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\nThe vertices on certain markers ('^', 'D', 'v') are clipped when using `plot()` to plot data with more than two points and saving the plot as a PDF.\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.pyplot as plt\r\nsize = 256, 16\r\nfig,ax = plt.subplots()\r\nnum = 3\r\n\r\nfig,ax = plt.subplots()\r\nax.plot([i+1 for i in range(num)],[1 for i in range(num)], '^', markersize=3, markeredgewidth=0.7, color='green', markeredgecolor='black')\r\nax.set_xlim(0, num +1)\r\nax.set_xticks(())\r\nax.set_yticks(())\r\nfig.savefig('test.pdf') # This doesn't work properly.\r\nfig.savefig('test.svg')\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\n\r\n![pdfmarker](https://user-images.githubusercontent.com/8723338/33104959-e7cacfd6-cef8-11e7-880a-0e8aa8643454.png)\r\n![svgmarker](https://user-images.githubusercontent.com/8723338/33104963-ea6f906e-cef8-11e7-8742-ce5b2421c337.png)\r\n\r\nThe top image is of the marker from the .pdf file. The bottom image is of the marker from the .svg file.\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\nThe marker in the .pdf file should look like the marker in the .svg file.\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 17.04\r\n  * Matplotlib version: 2.1.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): TkAgg\r\n  * Python version: Python 3.5.3\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \r\n\r\nI installed matplotlib via pip.\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": "2020-06-10T20:38:57Z",
      "base_commit": "f42b24f534ce5d9eb89d95387695bb2cee017829",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_pdf.py",
          "type": "function",
          "name": "markerObject",
          "class_name": "PdfFile",
          "code": "def markerObject(self, path, trans, fill, stroke, lw, joinstyle,\n                     capstyle):\n        \"\"\"Return name of a marker XObject representing the given path.\"\"\"\n        # self.markers used by markerObject, writeMarkers, close:\n        # mapping from (path operations, fill?, stroke?) to\n        #   [name, object reference, bounding box, linewidth]\n        # This enables different draw_markers calls to share the XObject\n        # if the gc is sufficiently similar: colors etc can vary, but\n        # the choices of whether to fill and whether to stroke cannot.\n        # We need a bounding box enclosing all of the XObject path,\n        # but since line width may vary, we store the maximum of all\n        # occurring line widths in self.markers.\n        # close() is somewhat tightly coupled in that it expects the\n        # first two components of each value in self.markers to be the\n        # name and object reference.\n        pathops = self.pathOperations(path, trans, simplify=False)\n        key = (tuple(pathops), bool(fill), bool(stroke), joinstyle, capstyle)\n        result = self.markers.get(key)\n        if result is None:\n            name = Name('M%d' % len(self.markers))\n            ob = self.reserveObject('marker %d' % len(self.markers))\n            bbox = path.get_extents(trans)\n            self.markers[key] = [name, ob, bbox, lw]\n        else:\n            if result[-1] < lw:\n                result[-1] = lw\n            name = result[0]\n        return name"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}