{
  "instance_id": "matplotlib__matplotlib-26020",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-05-31T21:36:23Z",
  "problem_statement": "Error creating AxisGrid with non-default axis class\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### Bug report\r\n\r\n**Bug summary**\r\n\r\nCreating `AxesGrid` using cartopy `GeoAxes` as `axis_class` raises `TypeError: 'method' object is not subscriptable`. Seems to be due to different behaviour of `axis` attr. for `mpl_toolkits.axes_grid1.mpl_axes.Axes` and other axes instances (like `GeoAxes`) where `axis` is only a callable. The error is raised in method `mpl_toolkits.axes_grid1.axes_grid._tick_only` when trying to access keys from `axis` attr.\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\nimport matplotlib.pyplot as plt\r\nfrom cartopy.crs import PlateCarree\r\nfrom cartopy.mpl.geoaxes import GeoAxes\r\nfrom mpl_toolkits.axes_grid1 import AxesGrid\r\n\r\nfig = plt.figure()\r\naxes_class = (GeoAxes, dict(map_projection=PlateCarree()))\r\ngr = AxesGrid(fig, 111, nrows_ncols=(1,1),\r\n              axes_class=axes_class)\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```\r\nTraceback (most recent call last):\r\n\r\n  File \"/home/jonasg/stuff/bugreport_mpl_toolkits_AxesGrid.py\", line 16, in <module>\r\n    axes_class=axes_class)\r\n\r\n  File \"/home/jonasg/miniconda3/envs/pya/lib/python3.7/site-packages/mpl_toolkits/axes_grid1/axes_grid.py\", line 618, in __init__\r\n    self.set_label_mode(label_mode)\r\n\r\n  File \"/home/jonasg/miniconda3/envs/pya/lib/python3.7/site-packages/mpl_toolkits/axes_grid1/axes_grid.py\", line 389, in set_label_mode\r\n    _tick_only(ax, bottom_on=False, left_on=False)\r\n\r\n  File \"/home/jonasg/miniconda3/envs/pya/lib/python3.7/site-packages/mpl_toolkits/axes_grid1/axes_grid.py\", line 27, in _tick_only\r\n    ax.axis[\"bottom\"].toggle(ticklabels=bottom_off, label=bottom_off)\r\n\r\nTypeError: 'method' object is not subscriptable\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\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 18.04.4 LTS\r\n  * Matplotlib version: 3.1.2 (conda-forge)\r\n  * Matplotlib backend: Qt5Agg \r\n  * Python version: 3.7.6\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \r\n\r\n```\r\n# Name                    Version                   Build  Channel\r\n_libgcc_mutex             0.1                 conda_forge    conda-forge\r\n_openmp_mutex             4.5                       0_gnu    conda-forge\r\nalabaster                 0.7.12                   py37_0  \r\nantlr-python-runtime      4.7.2                 py37_1001    conda-forge\r\nargh                      0.26.2                   py37_0  \r\nastroid                   2.3.3                    py37_0  \r\natomicwrites              1.3.0                    py37_1  \r\nattrs                     19.3.0                     py_0    conda-forge\r\nautopep8                  1.4.4                      py_0  \r\nbabel                     2.8.0                      py_0  \r\nbackcall                  0.1.0                    py37_0  \r\nbasemap                   1.2.1            py37hd759880_1    conda-forge\r\nbleach                    3.1.0                    py37_0  \r\nbokeh                     1.4.0                    py37_0    conda-forge\r\nbzip2                     1.0.8                h516909a_2    conda-forge\r\nca-certificates           2019.11.28           hecc5488_0    conda-forge\r\ncartopy                   0.17.0          py37hd759880_1006    conda-forge\r\ncertifi                   2019.11.28               py37_0    conda-forge\r\ncf-units                  2.1.3            py37hc1659b7_0    conda-forge\r\ncf_units                  2.0.1           py37h3010b51_1002    conda-forge\r\ncffi                      1.13.2           py37h8022711_0    conda-forge\r\ncftime                    1.0.4.2          py37hc1659b7_0    conda-forge\r\nchardet                   3.0.4                 py37_1003    conda-forge\r\nclick                     7.0                        py_0    conda-forge\r\ncloudpickle               1.2.2                      py_1    conda-forge\r\ncryptography              2.8              py37h72c5cf5_1    conda-forge\r\ncurl                      7.65.3               hf8cf82a_0    conda-forge\r\ncycler                    0.10.0                     py_2    conda-forge\r\ncytoolz                   0.10.1           py37h516909a_0    conda-forge\r\ndask                      2.9.2                      py_0    conda-forge\r\ndask-core                 2.9.2                      py_0    conda-forge\r\ndbus                      1.13.6               he372182_0    conda-forge\r\ndecorator                 4.4.1                      py_0  \r\ndefusedxml                0.6.0                      py_0  \r\ndiff-match-patch          20181111                   py_0  \r\ndistributed               2.9.3                      py_0    conda-forge\r\ndocutils                  0.16                     py37_0  \r\nentrypoints               0.3                      py37_0  \r\nexpat                     2.2.5             he1b5a44_1004    conda-forge\r\nflake8                    3.7.9                    py37_0  \r\nfontconfig                2.13.1            h86ecdb6_1001    conda-forge\r\nfreetype                  2.10.0               he983fc9_1    conda-forge\r\nfsspec                    0.6.2                      py_0    conda-forge\r\nfuture                    0.18.2                   py37_0  \r\ngeonum                    1.4.4                      py_0    conda-forge\r\ngeos                      3.7.2                he1b5a44_2    conda-forge\r\ngettext                   0.19.8.1          hc5be6a0_1002    conda-forge\r\nglib                      2.58.3          py37h6f030ca_1002    conda-forge\r\ngmp                       6.1.2                h6c8ec71_1  \r\ngpxpy                     1.4.0                      py_0    conda-forge\r\ngst-plugins-base          1.14.5               h0935bb2_0    conda-forge\r\ngstreamer                 1.14.5               h36ae1b5_0    conda-forge\r\nhdf4                      4.2.13            hf30be14_1003    conda-forge\r\nhdf5                      1.10.5          nompi_h3c11f04_1104    conda-forge\r\nheapdict                  1.0.1                      py_0    conda-forge\r\nicu                       64.2                 he1b5a44_1    conda-forge\r\nidna                      2.8                   py37_1000    conda-forge\r\nimagesize                 1.2.0                      py_0  \r\nimportlib_metadata        1.4.0                    py37_0    conda-forge\r\nintervaltree              3.0.2                      py_0  \r\nipykernel                 5.1.4            py37h39e3cac_0  \r\nipython                   7.11.1           py37h39e3cac_0  \r\nipython_genutils          0.2.0                    py37_0  \r\niris                      2.2.0                 py37_1003    conda-forge\r\nisort                     4.3.21                   py37_0  \r\njedi                      0.14.1                   py37_0  \r\njeepney                   0.4.2                      py_0  \r\njinja2                    2.10.3                     py_0    conda-forge\r\njpeg                      9c                h14c3975_1001    conda-forge\r\njson5                     0.8.5                      py_0  \r\njsonschema                3.2.0                    py37_0  \r\njupyter_client            5.3.4                    py37_0  \r\njupyter_core              4.6.1                    py37_0  \r\njupyterlab                1.2.5              pyhf63ae98_0  \r\njupyterlab_server         1.0.6                      py_0  \r\nkeyring                   21.1.0                   py37_0  \r\nkiwisolver                1.1.0            py37hc9558a2_0    conda-forge\r\nkrb5                      1.16.4               h2fd8d38_0    conda-forge\r\nlatlon23                  1.0.7                      py_0    conda-forge\r\nlazy-object-proxy         1.4.3            py37h7b6447c_0  \r\nld_impl_linux-64          2.33.1               h53a641e_7    conda-forge\r\nlibblas                   3.8.0               14_openblas    conda-forge\r\nlibcblas                  3.8.0               14_openblas    conda-forge\r\nlibclang                  9.0.1           default_hde54327_0    conda-forge\r\nlibcurl                   7.65.3               hda55be3_0    conda-forge\r\nlibedit                   3.1.20170329      hf8c457e_1001    conda-forge\r\nlibffi                    3.2.1             he1b5a44_1006    conda-forge\r\nlibgcc-ng                 9.2.0                h24d8f2e_2    conda-forge\r\nlibgfortran-ng            7.3.0                hdf63c60_4    conda-forge\r\nlibgomp                   9.2.0                h24d8f2e_2    conda-forge\r\nlibiconv                  1.15              h516909a_1005    conda-forge\r\nliblapack                 3.8.0               14_openblas    conda-forge\r\nlibllvm9                  9.0.1                hc9558a2_0    conda-forge\r\nlibnetcdf                 4.7.3           nompi_h94020b1_100    conda-forge\r\nlibopenblas               0.3.7                h5ec1e0e_6    conda-forge\r\nlibpng                    1.6.37               hed695b0_0    conda-forge\r\nlibsodium                 1.0.16               h1bed415_0  \r\nlibspatialindex           1.9.3                he6710b0_0  \r\nlibssh2                   1.8.2                h22169c7_2    conda-forge\r\nlibstdcxx-ng              9.2.0                hdf63c60_2    conda-forge\r\nlibtiff                   4.1.0                hc3755c2_3    conda-forge\r\nlibuuid                   2.32.1            h14c3975_1000    conda-forge\r\nlibxcb                    1.13              h14c3975_1002    conda-forge\r\nlibxkbcommon              0.9.1                hebb1f50_0    conda-forge\r\nlibxml2                   2.9.10               hee79883_0    conda-forge\r\nlocket                    0.2.0                      py_2    conda-forge\r\nlz4-c                     1.8.3             he1b5a44_1001    conda-forge\r\nmarkupsafe                1.1.1            py37h516909a_0    conda-forge\r\nmatplotlib                3.1.2                    py37_1    conda-forge\r\nmatplotlib-base           3.1.2            py37h250f245_1    conda-forge\r\nmccabe                    0.6.1                    py37_1  \r\nmistune                   0.8.4            py37h7b6447c_0  \r\nmore-itertools            8.1.0                      py_0    conda-forge\r\nmsgpack-python            0.6.2            py37hc9558a2_0    conda-forge\r\nnbconvert                 5.6.1                    py37_0  \r\nnbformat                  5.0.4                      py_0  \r\nnbsphinx                  0.5.1                      py_0    conda-forge\r\nncurses                   6.1               hf484d3e_1002    conda-forge\r\nnetcdf4                   1.5.3           nompi_py37hd35fb8e_102    conda-forge\r\nnotebook                  6.0.3                    py37_0  \r\nnspr                      4.24                 he1b5a44_0    conda-forge\r\nnss                       3.47                 he751ad9_0    conda-forge\r\nnumpy                     1.17.5           py37h95a1406_0    conda-forge\r\nnumpydoc                  0.9.2                      py_0  \r\nolefile                   0.46                       py_0    conda-forge\r\nopenssl                   1.1.1d               h516909a_0    conda-forge\r\nowslib                    0.19.0                     py_2    conda-forge\r\npackaging                 20.0                       py_0    conda-forge\r\npandas                    0.25.3           py37hb3f55d8_0    conda-forge\r\npandoc                    2.2.3.2                       0  \r\npandocfilters             1.4.2                    py37_1  \r\nparso                     0.6.0                      py_0  \r\npartd                     1.1.0                      py_0    conda-forge\r\npathtools                 0.1.2                      py_1  \r\npatsy                     0.5.1                      py_0    conda-forge\r\npcre                      8.43                 he1b5a44_0    conda-forge\r\npexpect                   4.8.0                    py37_0  \r\npickleshare               0.7.5                    py37_0  \r\npillow                    7.0.0            py37hefe7db6_0    conda-forge\r\npip                       20.0.1                   py37_0    conda-forge\r\npluggy                    0.13.0                   py37_0    conda-forge\r\nproj4                     5.2.0             he1b5a44_1006    conda-forge\r\nprometheus_client         0.7.1                      py_0  \r\nprompt_toolkit            3.0.3                      py_0  \r\npsutil                    5.6.7            py37h516909a_0    conda-forge\r\npthread-stubs             0.4               h14c3975_1001    conda-forge\r\nptyprocess                0.6.0                    py37_0  \r\npy                        1.8.1                      py_0    conda-forge\r\npyaerocom                 0.9.0.dev5                dev_0    <develop>\r\npycodestyle               2.5.0                    py37_0  \r\npycparser                 2.19                     py37_1    conda-forge\r\npydocstyle                4.0.1                      py_0  \r\npyepsg                    0.4.0                      py_0    conda-forge\r\npyflakes                  2.1.1                    py37_0  \r\npygments                  2.5.2                      py_0  \r\npyinstrument              3.1.2                    pypi_0    pypi\r\npyinstrument-cext         0.2.2                    pypi_0    pypi\r\npykdtree                  1.3.1           py37hc1659b7_1002    conda-forge\r\npyke                      1.1.1                 py37_1001    conda-forge\r\npylint                    2.4.4                    py37_0  \r\npyopenssl                 19.1.0                   py37_0    conda-forge\r\npyparsing                 2.4.6                      py_0    conda-forge\r\npyproj                    1.9.6           py37h516909a_1002    conda-forge\r\npyqt                      5.12.3           py37hcca6a23_1    conda-forge\r\npyqt5-sip                 4.19.18                  pypi_0    pypi\r\npyqtwebengine             5.12.1                   pypi_0    pypi\r\npyrsistent                0.15.7           py37h7b6447c_0  \r\npyshp                     2.1.0                      py_0    conda-forge\r\npysocks                   1.7.1                    py37_0    conda-forge\r\npytest                    5.3.4                    py37_0    conda-forge\r\npython                    3.7.6                h357f687_2    conda-forge\r\npython-dateutil           2.8.1                      py_0    conda-forge\r\npython-jsonrpc-server     0.3.4                      py_0  \r\npython-language-server    0.31.7                   py37_0  \r\npytz                      2019.3                     py_0    conda-forge\r\npyxdg                     0.26                       py_0  \r\npyyaml                    5.3              py37h516909a_0    conda-forge\r\npyzmq                     18.1.0           py37he6710b0_0  \r\nqdarkstyle                2.8                        py_0  \r\nqt                        5.12.5               hd8c4c69_1    conda-forge\r\nqtawesome                 0.6.1                      py_0  \r\nqtconsole                 4.6.0                      py_1  \r\nqtpy                      1.9.0                      py_0  \r\nreadline                  8.0                  hf8c457e_0    conda-forge\r\nrequests                  2.22.0                   py37_1    conda-forge\r\nrope                      0.16.0                     py_0  \r\nrtree                     0.9.3                    py37_0  \r\nscipy                     1.4.1            py37h921218d_0    conda-forge\r\nseaborn                   0.9.0                      py_2    conda-forge\r\nsecretstorage             3.1.2                    py37_0  \r\nsend2trash                1.5.0                    py37_0  \r\nsetuptools                45.1.0                   py37_0    conda-forge\r\nshapely                   1.6.4           py37hec07ddf_1006    conda-forge\r\nsimplejson                3.17.0           py37h516909a_0    conda-forge\r\nsix                       1.14.0                   py37_0    conda-forge\r\nsnowballstemmer           2.0.0                      py_0  \r\nsortedcontainers          2.1.0                      py_0    conda-forge\r\nsphinx                    2.3.1                      py_0  \r\nsphinx-rtd-theme          0.4.3                    pypi_0    pypi\r\nsphinxcontrib-applehelp   1.0.1                      py_0  \r\nsphinxcontrib-devhelp     1.0.1                      py_0  \r\nsphinxcontrib-htmlhelp    1.0.2                      py_0  \r\nsphinxcontrib-jsmath      1.0.1                      py_0  \r\nsphinxcontrib-qthelp      1.0.2                      py_0  \r\nsphinxcontrib-serializinghtml 1.1.3                      py_0  \r\nspyder                    4.0.1                    py37_0  \r\nspyder-kernels            1.8.1                    py37_0  \r\nsqlite                    3.30.1               hcee41ef_0    conda-forge\r\nsrtm.py                   0.3.4                      py_0    conda-forge\r\nstatsmodels               0.11.0           py37h516909a_0    conda-forge\r\ntblib                     1.6.0                      py_0    conda-forge\r\nterminado                 0.8.3                    py37_0  \r\ntestpath                  0.4.4                      py_0  \r\ntk                        8.6.10               hed695b0_0    conda-forge\r\ntoolz                     0.10.0                     py_0    conda-forge\r\ntornado                   6.0.3            py37h516909a_0    conda-forge\r\ntqdm                      4.43.0                   pypi_0    pypi\r\ntraitlets                 4.3.3                    py37_0  \r\nudunits2                  2.2.27.6          h4e0c4b3_1001    conda-forge\r\nujson                     1.35             py37h14c3975_0  \r\nurllib3                   1.25.7                   py37_0    conda-forge\r\nwatchdog                  0.9.0                    py37_1  \r\nwcwidth                   0.1.8                      py_0    conda-forge\r\nwebencodings              0.5.1                    py37_1  \r\nwheel                     0.33.6                   py37_0    conda-forge\r\nwrapt                     1.11.2           py37h7b6447c_0  \r\nwurlitzer                 2.0.0                    py37_0  \r\nxarray                    0.14.1                     py_1    conda-forge\r\nxorg-libxau               1.0.9                h14c3975_0    conda-forge\r\nxorg-libxdmcp             1.1.3                h516909a_0    conda-forge\r\nxz                        5.2.4             h14c3975_1001    conda-forge\r\nyaml                      0.2.2                h516909a_1    conda-forge\r\nyapf                      0.28.0                     py_0  \r\nzeromq                    4.3.1                he6710b0_3  \r\nzict                      1.0.0                      py_0    conda-forge\r\nzipp                      2.0.0                      py_2    conda-forge\r\nzlib                      1.2.11            h516909a_1006    conda-forge\r\nzstd                      1.4.4                h3b9ef0a_1    conda-forge\r\n```\r\n\n",
  "patch": "diff --git a/lib/mpl_toolkits/axes_grid1/axes_grid.py b/lib/mpl_toolkits/axes_grid1/axes_grid.py\n--- a/lib/mpl_toolkits/axes_grid1/axes_grid.py\n+++ b/lib/mpl_toolkits/axes_grid1/axes_grid.py\n@@ -1,5 +1,6 @@\n from numbers import Number\n import functools\n+from types import MethodType\n \n import numpy as np\n \n@@ -7,14 +8,20 @@\n from matplotlib.gridspec import SubplotSpec\n \n from .axes_divider import Size, SubplotDivider, Divider\n-from .mpl_axes import Axes\n+from .mpl_axes import Axes, SimpleAxisArtist\n \n \n def _tick_only(ax, bottom_on, left_on):\n     bottom_off = not bottom_on\n     left_off = not left_on\n-    ax.axis[\"bottom\"].toggle(ticklabels=bottom_off, label=bottom_off)\n-    ax.axis[\"left\"].toggle(ticklabels=left_off, label=left_off)\n+    if isinstance(ax.axis, MethodType):\n+        bottom = SimpleAxisArtist(ax.xaxis, 1, ax.spines[\"bottom\"])\n+        left = SimpleAxisArtist(ax.yaxis, 1, ax.spines[\"left\"])\n+    else:\n+        bottom = ax.axis[\"bottom\"]\n+        left = ax.axis[\"left\"]\n+    bottom.toggle(ticklabels=bottom_off, label=bottom_off)\n+    left.toggle(ticklabels=left_off, label=left_off)\n \n \n class CbarAxesBase:\n",
  "similar_bug_items": [
    {
      "pr_number": 20518,
      "pr_title": " Support sketch_params in pgf backend",
      "pr_body": "## PR Summary\r\n\r\nFixes matplotlib#20516\r\n\r\nPGF's `random steps` decoration seems to be the most similar,\r\nbut does not exactly match the behaviour described in matplotlib's docs.\r\nTherefore I repurposed the `randomness` argument as a seed to give\r\ncontrol on how the line looks afterwards:\r\n\r\n### before \r\n![current](https://user-images.githubusercontent.com/37397269/123407735-907af300-d5ac-11eb-8fba-6e54d171de45.png)\r\n\r\n### after \r\nwith same seed for both lines: \r\n\r\n![wanted](https://user-images.githubusercontent.com/37397269/123407761-9a9cf180-d5ac-11eb-9bd0-432bac4de25c.png)\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\r\n- [x] Has pytest style unit tests (and `pytest` passes).\r\n- [x] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [x] New features are documented, with examples if plot related.\r\n- [x] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [x] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [N/A] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [N/A] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\r\n\r\nI only tested this with '3.0.2', though.",
      "issue_id": 20516,
      "issue_title": "Sketch params ignored when using PGF backend",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\nCalls to `set_sketch_params()` are ignored by the PGF backend and do not have any influence in the resulting pgf or pdf file.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\n#!/usr/bin/env python3\r\nimport matplotlib as mpl\r\nimport matplotlib.pyplot as plt\r\nmpl.use(\"pgf\")\r\n\r\nmpl.rcParams.update({\r\n    'font.family': 'serif',\r\n    'text.usetex': True,\r\n    'pgf.rcfonts': False,\r\n    'pgf.preamble': [ \r\n            # enable this when using PGF backend with pdf output:\r\n            #r\"\\usepackage{pgf}\",\r\n            #r\"\\usepgfmodule{decorations}\",\r\n            #r\"\\usepgflibrary{decorations.pathmorphing}\",\r\n        ],\r\n})\r\n\r\nfig = plt.figure()\r\nax = fig.add_subplot(111)\r\nax.plot(range(10))\r\n\r\nax.spines[\"bottom\"].set_sketch_params(scale=5, length=10, randomness=42)\r\n\r\nfig.savefig(f\"foo.pgf\")\r\n#fig.savefig(f\"foo.pdf\")\r\n```\r\n\r\n**Actual outcome**\r\n![grafik](https://user-images.githubusercontent.com/37397269/123399005-6ae9eb80-d5a4-11eb-9da8-c05b9d0efa96.png)\r\n\r\n**Expected outcome**\r\n![grafik](https://user-images.githubusercontent.com/37397269/123399084-7e955200-d5a4-11eb-976e-03ae3d5b2275.png)\r\n\r\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Matplotlib version: '3.0.2'\r\n\r\nI am working on a fix for this.",
      "issue_closed_at": "2021-09-08T19:33:23Z",
      "base_commit": "1d129737743d7a1c000367c363cbb890183450bc",
      "changes": [
        {
          "file": "lib/matplotlib/artist.py",
          "type": "function",
          "name": "set_sketch_params",
          "class_name": "Artist",
          "code": "def set_sketch_params(self, scale=None, length=None, randomness=None):\n        \"\"\"\n        Set the sketch parameters.\n\n        Parameters\n        ----------\n        scale : float, optional\n            The amplitude of the wiggle perpendicular to the source\n            line, in pixels.  If scale is `None`, or not provided, no\n            sketch filter will be provided.\n        length : float, optional\n             The length of the wiggle along the line, in pixels\n             (default 128.0)\n        randomness : float, optional\n            The scale factor by which the length is shrunken or\n            expanded (default 16.0)\n\n            .. ACCEPTS: (scale: float, length: float, randomness: float)\n        \"\"\"\n        if scale is None:\n            self._sketch = None\n        else:\n            self._sketch = (scale, length or 128.0, randomness or 16.0)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "_print_pgf_path",
          "class_name": "RendererPgf",
          "code": "def _print_pgf_path(self, gc, path, transform, rgbFace=None):\n        f = 1. / self.dpi\n        # check for clip box / ignore clip for filled paths\n        bbox = gc.get_clip_rectangle() if gc else None\n        maxcoord = 16383 / 72.27 * self.dpi  # Max dimensions in LaTeX.\n        if bbox and (rgbFace is None):\n            p1, p2 = bbox.get_points()\n            clip = (max(p1[0], -maxcoord), max(p1[1], -maxcoord),\n                    min(p2[0], maxcoord), min(p2[1], maxcoord))\n        else:\n            clip = (-maxcoord, -maxcoord, maxcoord, maxcoord)\n        # build path\n        for points, code in path.iter_segments(transform, clip=clip):\n            if code == Path.MOVETO:\n                x, y = tuple(points)\n                writeln(self.fh,\n                        r\"\\pgfpathmoveto{\\pgfqpoint{%fin}{%fin}}\" %\n                        (f * x, f * y))\n            elif code == Path.CLOSEPOLY:\n                writeln(self.fh, r\"\\pgfpathclose\")\n            elif code == Path.LINETO:\n                x, y = tuple(points)\n                writeln(self.fh,\n                        r\"\\pgfpathlineto{\\pgfqpoint{%fin}{%fin}}\" %\n                        (f * x, f * y))\n            elif code == Path.CURVE3:\n                cx, cy, px, py = tuple(points)\n                coords = cx * f, cy * f, px * f, py * f\n                writeln(self.fh,\n                        r\"\\pgfpathquadraticcurveto\"\n                        r\"{\\pgfqpoint{%fin}{%fin}}{\\pgfqpoint{%fin}{%fin}}\"\n                        % coords)\n            elif code == Path.CURVE4:\n                c1x, c1y, c2x, c2y, px, py = tuple(points)\n                coords = c1x * f, c1y * f, c2x * f, c2y * f, px * f, py * f\n                writeln(self.fh,\n                        r\"\\pgfpathcurveto\"\n                        r\"{\\pgfqpoint{%fin}{%fin}}\"\n                        r\"{\\pgfqpoint{%fin}{%fin}}\"\n                        r\"{\\pgfqpoint{%fin}{%fin}}\"\n                        % coords)"
        }
      ]
    },
    {
      "pr_number": 23638,
      "pr_title": "FIX: correctly handle generic font families in svg text-as-text mode",
      "pr_body": "## PR Summary\r\n\r\nThis:\r\n\r\n- expands the generic fonts families to the user configured specific fonts\r\n  in the svg output\r\n- ensures that each font family only appears once per text element in the svg output\r\n- ensures that generic families are unquoted and specific families are\r\n\r\ncloses #22528\r\ncloses #23492\r\n\r\nI still need to write a test for this, but interactively this gives things like\r\n\r\n```svg\r\n <text style=\"font: 10px 'DejaVu Sans', 'WenQuanYi Zen Hei', 'DejaVu Sans', 'Bitstream Vera Sans', 'Computer Modern Sans Serif', 'Lucida Grande', 'Verdana', 'Geneva', 'Lucid', 'Arial', 'Helvetica', 'Avant Garde', sans-serif; text-anchor: start\" x=\"236.16\" y=\"174.528\" transform=\"rotate(-0 236.16 174.528)\">There are \u51e0\u4e2a\u6c49\u5b57 in between!</text>\r\n```\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- [x] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n",
      "issue_id": 23492,
      "issue_title": "[Bug]: svg backend does not use configured generic family lists",
      "issue_body": "### Bug summary\n\nThe FontProperties family parameter can be a generic family such as sans-serif and the actual font should be looked up in rcParams according to the documentation.\n\n### Code for reproduction\n\n```python\n>>> import matplotlib as mpl\r\n>>> import matplotlib.font_manager as fm\r\n>>> mpl.rcParams[\"font.family\"]\r\n['sans-serif']\r\n>>> mpl.rcParams[\"font.sans-serif\"]\r\n['DejaVu Sans',\r\n 'Bitstream Vera Sans',\r\n 'Computer Modern Sans Serif',\r\n 'Lucida Grande',\r\n 'Verdana',\r\n 'Geneva',\r\n 'Lucid',\r\n 'Arial',\r\n 'Helvetica',\r\n 'Avant Garde',\r\n 'sans-serif']\r\n>>> fp = fm.FontProperties()\r\n<matplotlib.font_manager.FontProperties at ...>\r\n>>> fp.get_family()\r\n['sans-serif']\n```\n\n\n### Actual outcome\n\n```python\r\n>>> fp.get_family()\r\n['sans-serif']\r\n```\n\n### Expected outcome\n\n```python\r\n>>> fp.get_family()\r\n['DejaVu Sans',\r\n 'Bitstream Vera Sans',\r\n 'Computer Modern Sans Serif',\r\n 'Lucida Grande',\r\n 'Verdana',\r\n 'Geneva',\r\n 'Lucid',\r\n 'Arial',\r\n 'Helvetica',\r\n 'Avant Garde',\r\n 'sans-serif']\r\n```\n\n### Additional information\n\n_No response_\n\n### Operating system\n\nWindows\n\n### Matplotlib Version\n\n3.5.2\n\n### Matplotlib Backend\n\nTkAgg\n\n### Python version\n\n3.9.13\n\n### Jupyter version\n\nN/A\n\n### Installation\n\npip",
      "issue_closed_at": "2022-09-30T22:27:50Z",
      "base_commit": "3feaa5de08f451ee16146731641a5b4b24662844",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_svg.py",
          "type": "function",
          "name": "_draw_text_as_text",
          "class_name": "RendererSVG",
          "code": "def _draw_text_as_text(self, gc, x, y, s, prop, angle, ismath, mtext=None):\n        writer = self.writer\n\n        color = rgb2hex(gc.get_rgb())\n        style = {}\n        if color != '#000000':\n            style['fill'] = color\n\n        alpha = gc.get_alpha() if gc.get_forced_alpha() else gc.get_rgb()[3]\n        if alpha != 1:\n            style['opacity'] = _short_float_fmt(alpha)\n\n        if not ismath:\n            attrib = {}\n\n            font_parts = []\n            if prop.get_style() != 'normal':\n                font_parts.append(prop.get_style())\n            if prop.get_variant() != 'normal':\n                font_parts.append(prop.get_variant())\n            weight = fm.weight_dict[prop.get_weight()]\n            if weight != 400:\n                font_parts.append(f'{weight}')\n            font_parts.extend([\n                f'{_short_float_fmt(prop.get_size())}px',\n                # ensure quoting\n                f'{\", \".join(repr(f) for f in prop.get_family())}',\n            ])\n            style['font'] = ' '.join(font_parts)\n            if prop.get_stretch() != 'normal':\n                style['font-stretch'] = prop.get_stretch()\n            attrib['style'] = _generate_css(style)\n\n            if mtext and (angle == 0 or mtext.get_rotation_mode() == \"anchor\"):\n                # If text anchoring can be supported, get the original\n                # coordinates and add alignment information.\n\n                # Get anchor coordinates.\n                transform = mtext.get_transform()\n                ax, ay = transform.transform(mtext.get_unitless_position())\n                ay = self.height - ay\n\n                # Don't do vertical anchor alignment. Most applications do not\n                # support 'alignment-baseline' yet. Apply the vertical layout\n                # to the anchor point manually for now.\n                angle_rad = np.deg2rad(angle)\n                dir_vert = np.array([np.sin(angle_rad), np.cos(angle_rad)])\n                v_offset = np.dot(dir_vert, [(x - ax), (y - ay)])\n                ax = ax + v_offset * dir_vert[0]\n                ay = ay + v_offset * dir_vert[1]\n\n                ha_mpl_to_svg = {'left': 'start', 'right': 'end',\n                                 'center': 'middle'}\n                style['text-anchor'] = ha_mpl_to_svg[mtext.get_ha()]\n\n                attrib['x'] = _short_float_fmt(ax)\n                attrib['y'] = _short_float_fmt(ay)\n                attrib['style'] = _generate_css(style)\n                attrib['transform'] = _generate_transform([\n                    (\"rotate\", (-angle, ax, ay))])\n\n            else:\n                attrib['transform'] = _generate_transform([\n                    ('translate', (x, y)),\n                    ('rotate', (-angle,))])\n\n            writer.element('text', s, attrib=attrib)\n\n        else:\n            writer.comment(s)\n\n            width, height, descent, glyphs, rects = \\\n                self._text2path.mathtext_parser.parse(s, 72, prop)\n\n            # Apply attributes to 'g', not 'text', because we likely have some\n            # rectangles as well with the same style and transformation.\n            writer.start('g',\n                         style=_generate_css(style),\n                         transform=_generate_transform([\n                             ('translate', (x, y)),\n                             ('rotate', (-angle,))]),\n                         )\n\n            writer.start('text')\n\n            # Sort the characters by font, and output one tspan for each.\n            spans = {}\n            for font, fontsize, thetext, new_x, new_y in glyphs:\n                entry = fm.ttfFontProperty(font)\n                font_parts = []\n                if entry.style != 'normal':\n                    font_parts.append(entry.style)\n                if entry.variant != 'normal':\n                    font_parts.append(entry.variant)\n                if entry.weight != 400:\n                    font_parts.append(f'{entry.weight}')\n                font_parts.extend([\n                    f'{_short_float_fmt(fontsize)}px',\n                    f'{entry.name!r}',  # ensure quoting\n                ])\n                style = {'font': ' '.join(font_parts)}\n                if entry.stretch != 'normal':\n                    style['font-stretch'] = entry.stretch\n                style = _generate_css(style)\n                if thetext == 32:\n                    thetext = 0xa0  # non-breaking space\n                spans.setdefault(style, []).append((new_x, -new_y, thetext))\n\n            for style, chars in spans.items():\n                chars.sort()\n\n                if len({y for x, y, t in chars}) == 1:  # Are all y's the same?\n                    ys = str(chars[0][1])\n                else:\n                    ys = ' '.join(str(c[1]) for c in chars)\n\n                attrib = {\n                    'style': style,\n                    'x': ' '.join(_short_float_fmt(c[0]) for c in chars),\n                    'y': ys\n                    }\n\n                writer.element(\n                    'tspan',\n                    ''.join(chr(c[2]) for c in chars),\n                    attrib=attrib)\n\n            writer.end('text')\n\n            for x, y, width, height in rects:\n                writer.element(\n                    'rect',\n                    x=_short_float_fmt(x),\n                    y=_short_float_fmt(-y-1),\n                    width=_short_float_fmt(width),\n                    height=_short_float_fmt(height)\n                    )\n\n            writer.end('g')"
        },
        {
          "file": "lib/matplotlib/font_manager.py",
          "type": "function",
          "name": "findfont",
          "class_name": "FontManager",
          "code": "def findfont(self, prop, fontext='ttf', directory=None,\n                 fallback_to_default=True, rebuild_if_missing=True):\n        \"\"\"\n        Find a font that most closely matches the given font properties.\n\n        Parameters\n        ----------\n        prop : str or `~matplotlib.font_manager.FontProperties`\n            The font properties to search for. This can be either a\n            `.FontProperties` object or a string defining a\n            `fontconfig patterns`_.\n\n        fontext : {'ttf', 'afm'}, default: 'ttf'\n            The extension of the font file:\n\n            - 'ttf': TrueType and OpenType fonts (.ttf, .ttc, .otf)\n            - 'afm': Adobe Font Metrics (.afm)\n\n        directory : str, optional\n            If given, only search this directory and its subdirectories.\n\n        fallback_to_default : bool\n            If True, will fallback to the default font family (usually\n            \"DejaVu Sans\" or \"Helvetica\") if the first lookup hard-fails.\n\n        rebuild_if_missing : bool\n            Whether to rebuild the font cache and search again if the first\n            match appears to point to a nonexisting font (i.e., the font cache\n            contains outdated entries).\n\n        Returns\n        -------\n        str\n            The filename of the best matching font.\n\n        Notes\n        -----\n        This performs a nearest neighbor search.  Each font is given a\n        similarity score to the target font properties.  The first font with\n        the highest score is returned.  If no matches below a certain\n        threshold are found, the default font (usually DejaVu Sans) is\n        returned.\n\n        The result is cached, so subsequent lookups don't have to\n        perform the O(n) nearest neighbor search.\n\n        See the `W3C Cascading Style Sheet, Level 1\n        <http://www.w3.org/TR/1998/REC-CSS2-19980512/>`_ documentation\n        for a description of the font finding algorithm.\n\n        .. _fontconfig patterns:\n           https://www.freedesktop.org/software/fontconfig/fontconfig-user.html\n        \"\"\"\n        # Pass the relevant rcParams (and the font manager, as `self`) to\n        # _findfont_cached so to prevent using a stale cache entry after an\n        # rcParam was changed.\n        rc_params = tuple(tuple(mpl.rcParams[key]) for key in [\n            \"font.serif\", \"font.sans-serif\", \"font.cursive\", \"font.fantasy\",\n            \"font.monospace\"])\n        return self._findfont_cached(\n            prop, fontext, directory, fallback_to_default, rebuild_if_missing,\n            rc_params)"
        },
        {
          "file": "lib/matplotlib/font_manager.py",
          "type": "function",
          "name": "_findfont_cached",
          "class_name": "FontManager",
          "code": "def _findfont_cached(self, prop, fontext, directory, fallback_to_default,\n                         rebuild_if_missing, rc_params):\n\n        prop = FontProperties._from_any(prop)\n\n        fname = prop.get_file()\n        if fname is not None:\n            return fname\n\n        if fontext == 'afm':\n            fontlist = self.afmlist\n        else:\n            fontlist = self.ttflist\n\n        best_score = 1e64\n        best_font = None\n\n        _log.debug('findfont: Matching %s.', prop)\n        for font in fontlist:\n            if (directory is not None and\n                    Path(directory) not in Path(font.fname).parents):\n                continue\n            # Matching family should have top priority, so multiply it by 10.\n            score = (self.score_family(prop.get_family(), font.name) * 10\n                     + self.score_style(prop.get_style(), font.style)\n                     + self.score_variant(prop.get_variant(), font.variant)\n                     + self.score_weight(prop.get_weight(), font.weight)\n                     + self.score_stretch(prop.get_stretch(), font.stretch)\n                     + self.score_size(prop.get_size(), font.size))\n            _log.debug('findfont: score(%s) = %s', font, score)\n            if score < best_score:\n                best_score = score\n                best_font = font\n            if score == 0:\n                break\n\n        if best_font is None or best_score >= 10.0:\n            if fallback_to_default:\n                _log.warning(\n                    'findfont: Font family %s not found. Falling back to %s.',\n                    prop.get_family(), self.defaultFamily[fontext])\n                for family in map(str.lower, prop.get_family()):\n                    if family in font_family_aliases:\n                        _log.warning(\n                            \"findfont: Generic family %r not found because \"\n                            \"none of the following families were found: %s\",\n                            family, \", \".join(self._expand_aliases(family)))\n                default_prop = prop.copy()\n                default_prop.set_family(self.defaultFamily[fontext])\n                return self.findfont(default_prop, fontext, directory,\n                                     fallback_to_default=False)\n            else:\n                raise ValueError(f\"Failed to find font {prop}, and fallback \"\n                                 f\"to the default font was disabled\")\n        else:\n            _log.debug('findfont: Matching %s to %s (%r) with score of %f.',\n                       prop, best_font.name, best_font.fname, best_score)\n            result = best_font.fname\n\n        if not os.path.isfile(result):\n            if rebuild_if_missing:\n                _log.info(\n                    'findfont: Found a missing font file.  Rebuilding cache.')\n                new_fm = _load_fontmanager(try_read_cache=False)\n                # Replace self by the new fontmanager, because users may have\n                # a reference to this specific instance.\n                # TODO: _load_fontmanager should really be (used by) a method\n                # modifying the instance in place.\n                vars(self).update(vars(new_fm))\n                return self.findfont(\n                    prop, fontext, directory, rebuild_if_missing=False)\n            else:\n                raise ValueError(\"No valid font could be found\")\n\n        return _cached_realpath(result)"
        },
        {
          "file": "lib/matplotlib/font_manager.py",
          "type": "function",
          "name": "_findfont_cached",
          "class_name": "FontManager",
          "code": "def _findfont_cached(self, prop, fontext, directory, fallback_to_default,\n                         rebuild_if_missing, rc_params):\n\n        prop = FontProperties._from_any(prop)\n\n        fname = prop.get_file()\n        if fname is not None:\n            return fname\n\n        if fontext == 'afm':\n            fontlist = self.afmlist\n        else:\n            fontlist = self.ttflist\n\n        best_score = 1e64\n        best_font = None\n\n        _log.debug('findfont: Matching %s.', prop)\n        for font in fontlist:\n            if (directory is not None and\n                    Path(directory) not in Path(font.fname).parents):\n                continue\n            # Matching family should have top priority, so multiply it by 10.\n            score = (self.score_family(prop.get_family(), font.name) * 10\n                     + self.score_style(prop.get_style(), font.style)\n                     + self.score_variant(prop.get_variant(), font.variant)\n                     + self.score_weight(prop.get_weight(), font.weight)\n                     + self.score_stretch(prop.get_stretch(), font.stretch)\n                     + self.score_size(prop.get_size(), font.size))\n            _log.debug('findfont: score(%s) = %s', font, score)\n            if score < best_score:\n                best_score = score\n                best_font = font\n            if score == 0:\n                break\n\n        if best_font is None or best_score >= 10.0:\n            if fallback_to_default:\n                _log.warning(\n                    'findfont: Font family %s not found. Falling back to %s.',\n                    prop.get_family(), self.defaultFamily[fontext])\n                for family in map(str.lower, prop.get_family()):\n                    if family in font_family_aliases:\n                        _log.warning(\n                            \"findfont: Generic family %r not found because \"\n                            \"none of the following families were found: %s\",\n                            family, \", \".join(self._expand_aliases(family)))\n                default_prop = prop.copy()\n                default_prop.set_family(self.defaultFamily[fontext])\n                return self.findfont(default_prop, fontext, directory,\n                                     fallback_to_default=False)\n            else:\n                raise ValueError(f\"Failed to find font {prop}, and fallback \"\n                                 f\"to the default font was disabled\")\n        else:\n            _log.debug('findfont: Matching %s to %s (%r) with score of %f.',\n                       prop, best_font.name, best_font.fname, best_score)\n            result = best_font.fname\n\n        if not os.path.isfile(result):\n            if rebuild_if_missing:\n                _log.info(\n                    'findfont: Found a missing font file.  Rebuilding cache.')\n                new_fm = _load_fontmanager(try_read_cache=False)\n                # Replace self by the new fontmanager, because users may have\n                # a reference to this specific instance.\n                # TODO: _load_fontmanager should really be (used by) a method\n                # modifying the instance in place.\n                vars(self).update(vars(new_fm))\n                return self.findfont(\n                    prop, fontext, directory, rebuild_if_missing=False)\n            else:\n                raise ValueError(\"No valid font could be found\")\n\n        return _cached_realpath(result)"
        }
      ]
    },
    {
      "pr_number": 4843,
      "pr_title": "Rev coord wrapping",
      "pr_body": "Closes #4292\n",
      "issue_id": 4292,
      "issue_title": "Annotation with negative axes fraction coordinate placed incorrectly with v1.4.3",
      "issue_body": "I am finding that annotations placed outside of axes get placed incorrectly in v1.4.3.  Here is an example:\n\n```\nimport matplotlib.pyplot as plt\nimport matplotlib as mpl\n\nfig, ax = plt.subplots()\n\nax.annotate('+ pts', \n    xytext = [40, 20], textcoords = 'axes points', \\\n    xy = [40, 20], xycoords = 'axes points', fontsize = 32)\nax.annotate('- pts', \n    xytext = [40, -20], textcoords = 'axes points', \\\n    xy = [40, -20], xycoords = 'axes points', fontsize = 32)\nax.annotate('+ frac', \n    xytext = [0.5, 0.1], textcoords = 'axes fraction', \\\n    xy = [0.5, 0.1], xycoords = 'axes fraction', fontsize = 32)\nax.annotate('- frac', \n    xytext = [0.5, -0.1], textcoords = 'axes fraction', \\\n    xy = [0.5, -0.1], xycoords = 'axes fraction', fontsize = 32)\nplt.savefig('test' + mpl.__version__ + '.png')\n```\n\nIn matplotlib v1.4.2 the plot looks like this:\n![test1 4 2](https://cloud.githubusercontent.com/assets/4099759/6883289/4627552e-d56c-11e4-8518-543d4dee233e.png)\n\nIn matplotlib v1.4.3 the plot looks like this:\n![test1 4 3](https://cloud.githubusercontent.com/assets/4099759/6883290/4d13ae3c-d56c-11e4-85f1-dd4b195dc9d6.png)\n\nApparently v1.4.3 takes a negative axes fraction (or points) and wraps it around to the top of the axes.\n",
      "issue_closed_at": "2015-08-06T18:48:29Z",
      "base_commit": "263c06ac864fe8425caa629f6f1789646b8538b1",
      "changes": [
        {
          "file": "lib/matplotlib/text.py",
          "type": "function",
          "name": "_get_xy",
          "class_name": "_AnnotationBase",
          "code": "def _get_xy(self, renderer, x, y, s):\n        if isinstance(s, tuple):\n            s1, s2 = s\n        else:\n            s1, s2 = s, s\n\n        if s1 == 'data':\n            x = float(self.convert_xunits(x))\n        if s2 == 'data':\n            y = float(self.convert_yunits(y))\n\n        tr = self._get_xy_transform(renderer, (x, y), s)\n        x1, y1 = tr.transform_point((x, y))\n        return x1, y1"
        },
        {
          "file": "lib/matplotlib/text.py",
          "type": "function",
          "name": "_get_xy_transform",
          "class_name": "_AnnotationBase",
          "code": "def _get_xy_transform(self, renderer, xy, s):\n\n        if isinstance(s, tuple):\n            s1, s2 = s\n            from matplotlib.transforms import blended_transform_factory\n            tr1 = self._get_xy_transform(renderer, xy, s1)\n            tr2 = self._get_xy_transform(renderer, xy, s2)\n            tr = blended_transform_factory(tr1, tr2)\n            return tr\n\n        if six.callable(s):\n            tr = s(renderer)\n            if isinstance(tr, BboxBase):\n                return BboxTransformTo(tr)\n            elif isinstance(tr, Transform):\n                return tr\n            else:\n                raise RuntimeError(\"unknown return type ...\")\n        if isinstance(s, Artist):\n            bbox = s.get_window_extent(renderer)\n            return BboxTransformTo(bbox)\n        elif isinstance(s, BboxBase):\n            return BboxTransformTo(s)\n        elif isinstance(s, Transform):\n            return s\n        elif not is_string_like(s):\n            raise RuntimeError(\"unknown coordinate type : %s\" % (s,))\n\n        if s == 'data':\n            return self.axes.transData\n        elif s == 'polar':\n            from matplotlib.projections import PolarAxes\n            tr = PolarAxes.PolarTransform()\n            trans = tr + self.axes.transData\n            return trans\n\n        s_ = s.split()\n        if len(s_) != 2:\n            raise ValueError(\"%s is not a recognized coordinate\" % s)\n\n        bbox0, xy0 = None, None\n\n        bbox_name, unit = s_\n        # if unit is offset-like\n        if bbox_name == \"figure\":\n            bbox0 = self.figure.bbox\n        elif bbox_name == \"axes\":\n            bbox0 = self.axes.bbox\n        # elif bbox_name == \"bbox\":\n        #     if bbox is None:\n        #         raise RuntimeError(\"bbox is specified as a coordinate but \"\n        #                            \"never set\")\n        #     bbox0 = self._get_bbox(renderer, bbox)\n\n        if bbox0 is not None:\n            x, y = xy\n            bounds = bbox0.extents\n            if x < 0:\n                x0 = bounds[2]\n            else:\n                x0 = bounds[0]\n            if y < 0:\n                y0 = bounds[3]\n            else:\n                y0 = bounds[1]\n            xy0 = (x0, y0)\n        elif bbox_name == \"offset\":\n            xy0 = self._get_ref_xy(renderer)\n\n        if xy0 is not None:\n            # reference x, y in display coordinate\n            ref_x, ref_y = xy0\n            from matplotlib.transforms import Affine2D\n            if unit == \"points\":\n                # dots per points\n                dpp = self.figure.get_dpi() / 72.\n                tr = Affine2D().scale(dpp, dpp)\n            elif unit == \"pixels\":\n                tr = Affine2D()\n            elif unit == \"fontsize\":\n                fontsize = self.get_size()\n                dpp = fontsize * self.figure.get_dpi() / 72.\n                tr = Affine2D().scale(dpp,\n                                      dpp)\n            elif unit == \"fraction\":\n                w, h = bbox0.bounds[2:]\n                tr = Affine2D().scale(w, h)\n            else:\n                raise ValueError(\"%s is not a recognized coordinate\" % s)\n\n            return tr.translate(ref_x, ref_y)\n\n        else:\n            raise ValueError(\"%s is not a recognized coordinate\" % s)"
        },
        {
          "file": "lib/matplotlib/text.py",
          "type": "function",
          "name": "_update_position_xytext",
          "class_name": "Annotation",
          "code": "def _update_position_xytext(self, renderer, xy_pixel):\n        \"\"\"Update the pixel positions of the annotation text and the arrow\n        patch.\n        \"\"\"\n        # generate transformation,\n        self.set_transform(self._get_xy_transform(\n            renderer, self.xy, self.anncoords))\n\n        ox0, oy0 = self._get_xy_display()\n        ox1, oy1 = xy_pixel\n\n        if self.arrowprops:\n            x0, y0 = xy_pixel\n            l, b, w, h = Text.get_window_extent(self, renderer).bounds\n            r = l + w\n            t = b + h\n            xc = 0.5 * (l + r)\n            yc = 0.5 * (b + t)\n\n            d = self.arrowprops.copy()\n            ms = d.pop(\"mutation_scale\", self.get_size())\n            ms = renderer.points_to_pixels(ms)\n            self.arrow_patch.set_mutation_scale(ms)\n\n            if \"arrowstyle\" not in d:\n                # Approximately simulate the YAArrow.\n                # Pop its kwargs:\n                shrink = d.pop('shrink', 0.0)\n                width = d.pop('width', 4)\n                headwidth = d.pop('headwidth', 12)\n                # Ignore frac--it is useless.\n                frac = d.pop('frac', None)\n                if frac is not None:\n                    warnings.warn(\n                        \"'frac' option in 'arrowstyle' is no longer supported;\"\n                        \" use 'headlength' to set the head length in points.\")\n                headlength = d.pop('headlength', 12)\n\n                to_style = self.figure.dpi / (72 * ms)\n\n                stylekw = dict(head_length=headlength * to_style,\n                               head_width=headwidth * to_style,\n                               tail_width=width * to_style)\n\n                self.arrow_patch.set_arrowstyle('simple', **stylekw)\n\n                # using YAArrow style:\n                # pick the x,y corner of the text bbox closest to point\n                # annotated\n                xpos = ((l, 0), (xc, 0.5), (r, 1))\n                ypos = ((b, 0), (yc, 0.5), (t, 1))\n\n                dsu = [(abs(val[0] - x0), val) for val in xpos]\n                dsu.sort()\n                _, (x, relposx) = dsu[0]\n\n                dsu = [(abs(val[0] - y0), val) for val in ypos]\n                dsu.sort()\n                _, (y, relposy) = dsu[0]\n\n                self._arrow_relpos = (relposx, relposy)\n\n                r = np.hypot((y - y0), (x - x0))\n                shrink_pts = shrink * r / renderer.points_to_pixels(1)\n                self.arrow_patch.shrinkA = shrink_pts\n                self.arrow_patch.shrinkB = shrink_pts\n\n            # adjust the starting point of the arrow relative to\n            # the textbox.\n            # TODO : Rotation needs to be accounted.\n            relpos = self._arrow_relpos\n            bbox = Text.get_window_extent(self, renderer)\n            ox0 = bbox.x0 + bbox.width * relpos[0]\n            oy0 = bbox.y0 + bbox.height * relpos[1]\n\n            # The arrow will be drawn from (ox0, oy0) to (ox1,\n            # oy1). It will be first clipped by patchA and patchB.\n            # Then it will be shrunk by shirnkA and shrinkB\n            # (in points). If patch A is not set, self.bbox_patch\n            # is used.\n\n            self.arrow_patch.set_positions((ox0, oy0), (ox1, oy1))\n\n            if \"patchA\" in d:\n                self.arrow_patch.set_patchA(d.pop(\"patchA\"))\n            else:\n                if self._bbox_patch:\n                    self.arrow_patch.set_patchA(self._bbox_patch)\n                else:\n                    pad = renderer.points_to_pixels(4)\n                    if self.get_text().strip() == \"\":\n                        self.arrow_patch.set_patchA(None)\n                        return\n\n                    bbox = Text.get_window_extent(self, renderer)\n                    l, b, w, h = bbox.bounds\n                    l -= pad / 2.\n                    b -= pad / 2.\n                    w += pad\n                    h += pad\n                    r = Rectangle(xy=(l, b),\n                                  width=w,\n                                  height=h,\n                                  )\n                    r.set_transform(mtransforms.IdentityTransform())\n                    r.set_clip_on(False)\n\n                    self.arrow_patch.set_patchA(r)"
        }
      ]
    },
    {
      "pr_number": 17635,
      "pr_title": "Fix autoscaling with tiny sticky values.",
      "pr_body": "## PR Summary\r\n\r\nCloses #17633.  I guess may be nice to have in 3.3, but heh.\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": 17633,
      "issue_title": "Stackplot fails for small numbers",
      "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\nIt appears that `stackplot` fails for small numbers in version 3.2.1, or at least something goes wrong in my clean installation, see figures below.\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\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\nx =  [0, 1]\r\ny1 = 1e-8*np.ones_like(x)\r\ny2 = 1e-9*np.ones_like(x)\r\n\r\nfig, axs = plt.subplots(2, 1)\r\naxs[0].stackplot(x, y1, y2)  # This plot is fine\r\naxs[1].stackplot(x, y1/10, y2/10)  # This plot is wrong\r\nplt.show()\r\n```\r\n\r\n**Actual outcome**\r\n![image](https://user-images.githubusercontent.com/3267619/84637648-4d1d8b80-af39-11ea-8f33-624ef07089e6.png)\r\nSomething went wrong in the second subplot.\r\n<!--The output produced by the above code, which may be a screenshot, console output, etc.-->\r\n\r\n**Expected outcome**\r\nExpected this, as generated with `matplotlib` version 3.1.2\r\n![mpl312](https://user-images.githubusercontent.com/3267619/84637520-219aa100-af39-11ea-8319-e9166e6d9f64.png)\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\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Operating system: macOS 10.12.6 (Sierra)\r\n  * Matplotlib version: 3.2.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): MacOSX\r\n  * Python version: 3.8.3\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\nCreated a conda environment with only the`numpy` package, then installed matplotlib with `pip`.",
      "issue_closed_at": "2020-06-15T19:26:17Z",
      "base_commit": "c3bfeb9c3c42fadd6a11d04fc9d68e2943242a64",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_base.py",
          "type": "function",
          "name": "handle_single_axis",
          "class_name": "_AxesBase",
          "code": "def handle_single_axis(scale, autoscaleon, shared_axes, interval,\n                               minpos, axis, margin, stickies, set_bound):\n\n            if not (scale and autoscaleon):\n                return  # nothing to do...\n\n            shared = shared_axes.get_siblings(self)\n            dl = [ax.dataLim for ax in shared]\n            # ignore non-finite data limits if good limits exist\n            finite_dl = [d for d in dl if np.isfinite(d).all()]\n            if len(finite_dl):\n                # if finite limits exist for at least one axis (and the\n                # other is infinite), restore the finite limits\n                x_finite = [d for d in dl\n                            if (np.isfinite(d.intervalx).all() and\n                                (d not in finite_dl))]\n                y_finite = [d for d in dl\n                            if (np.isfinite(d.intervaly).all() and\n                                (d not in finite_dl))]\n\n                dl = finite_dl\n                dl.extend(x_finite)\n                dl.extend(y_finite)\n\n            bb = mtransforms.BboxBase.union(dl)\n            x0, x1 = getattr(bb, interval)\n            # If x0 and x1 are non finite, use the locator to figure out\n            # default limits.\n            locator = axis.get_major_locator()\n            x0, x1 = locator.nonsingular(x0, x1)\n\n            # Prevent margin addition from crossing a sticky value.  Small\n            # tolerances (whose values come from isclose()) must be used due to\n            # floating point issues with streamplot.\n            def tol(x): return 1e-5 * abs(x) + 1e-8\n            # Index of largest element < x0 + tol, if any.\n            i0 = stickies.searchsorted(x0 + tol(x0)) - 1\n            x0bound = stickies[i0] if i0 != -1 else None\n            # Index of smallest element > x1 - tol, if any.\n            i1 = stickies.searchsorted(x1 - tol(x1))\n            x1bound = stickies[i1] if i1 != len(stickies) else None\n\n            # Add the margin in figure space and then transform back, to handle\n            # non-linear scales.\n            minpos = getattr(bb, minpos)\n            transform = axis.get_transform()\n            inverse_trans = transform.inverted()\n            x0, x1 = axis._scale.limit_range_for_scale(x0, x1, minpos)\n            x0t, x1t = transform.transform([x0, x1])\n            delta = (x1t - x0t) * margin\n            if not np.isfinite(delta):\n                delta = 0  # If a bound isn't finite, set margin to zero.\n            x0, x1 = inverse_trans.transform([x0t - delta, x1t + delta])\n\n            # Apply sticky bounds.\n            if x0bound is not None:\n                x0 = max(x0, x0bound)\n            if x1bound is not None:\n                x1 = min(x1, x1bound)\n\n            if not self._tight:\n                x0, x1 = locator.view_limits(x0, x1)\n            set_bound(x0, x1)"
        }
      ]
    },
    {
      "pr_number": 19888,
      "pr_title": "Fix errorbar drawstyle",
      "pr_body": "## PR Summary\r\nclose #19879 \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\r\n- [x] Has pytest style unit tests (and `pytest` passes).\r\n- [x] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [N/A] New features are documented, with examples if plot related.\r\n- [x] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [x] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [N/A] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [N/A] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\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": 19879,
      "issue_title": "Using \"drawstyle\" raises AttributeError in errorbar, when yerr is specified.",
      "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\nHello maintainers! \r\nStarting from 3.4.1, I'm having failures in CI related to a failure in `errorbar` (e.g. https://github.com/StingraySoftware/HENDRICS/runs/2267849177?check_suite_focus=true). Until at least 3.3.4, this code line\r\n\r\n```\r\nplt.errorbar(x, profile, yerr=profile_err, drawstyle=\"steps-mid\")\r\n```\r\n\r\nused to work, now it doesn't. After a little debugging, I realized that `yerr` and `drawstyle` seem incompatible. I write a MWE below\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\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\nIn [1]: import numpy as np\r\n\r\nIn [2]: import matplotlib.pyplot as plt\r\n\r\nIn [3]: x = np.arange(10)\r\n\r\nIn [4]: y = np.random.poisson(100, 10)\r\n\r\nIn [5]: plt.errorbar(x, y, drawstyle=\"steps-mid\")\r\nOut[5]: <ErrorbarContainer object of 3 artists>\r\n\r\nIn [6]: plt.show()  # It works\r\n\r\nIn [7]: yerr = np.zeros_like(y) + 10\r\n\r\nIn [8]: plt.errorbar(x, y, yerr=yerr, drawstyle=\"steps-mid\")\r\n---------------------------------------------------------------------------\r\nAttributeError                            Traceback (most recent call last)\r\n<ipython-input-8-a70dfad4683c> in <module>\r\n----> 1 plt.errorbar(x, y, yerr=yerr, drawstyle=\"steps-mid\")\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/pyplot.py in errorbar(x, y, yerr, xerr, fmt, ecolor, elinewidth, capsize, barsabove, lolims, uplims, xlolims, xuplims, errorevery, capthick, data, **kwargs)\r\n   2739         uplims=False, xlolims=False, xuplims=False, errorevery=1,\r\n   2740         capthick=None, *, data=None, **kwargs):\r\n-> 2741     return gca().errorbar(\r\n   2742         x, y, yerr=yerr, xerr=xerr, fmt=fmt, ecolor=ecolor,\r\n   2743         elinewidth=elinewidth, capsize=capsize, barsabove=barsabove,\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/__init__.py in inner(ax, data, *args, **kwargs)\r\n   1350     def inner(ax, *args, data=None, **kwargs):\r\n   1351         if data is None:\r\n-> 1352             return func(ax, *map(sanitize_sequence, args), **kwargs)\r\n   1353 \r\n   1354         bound = new_sig.bind(ax, *args, **kwargs)\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/axes/_axes.py in errorbar(self, x, y, yerr, xerr, fmt, ecolor, elinewidth, capsize, barsabove, lolims, uplims, xlolims, xuplims, errorevery, capthick, **kwargs)\r\n   3496         if yerr is not None:\r\n   3497             lower, upper = extract_err('y', yerr, y, lolims, uplims)\r\n-> 3498             barcols.append(self.vlines(\r\n   3499                 *apply_mask([x, lower, upper], everymask), **eb_lines_style))\r\n   3500             # select points without upper/lower limits in y and\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/__init__.py in inner(ax, data, *args, **kwargs)\r\n   1350     def inner(ax, *args, data=None, **kwargs):\r\n   1351         if data is None:\r\n-> 1352             return func(ax, *map(sanitize_sequence, args), **kwargs)\r\n   1353 \r\n   1354         bound = new_sig.bind(ax, *args, **kwargs)\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/axes/_axes.py in vlines(self, x, ymin, ymax, colors, linestyles, label, **kwargs)\r\n   1118                                      linestyles=linestyles, label=label)\r\n   1119         self.add_collection(lines, autolim=False)\r\n-> 1120         lines.update(kwargs)\r\n   1121 \r\n   1122         if len(x) > 0:\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/artist.py in update(self, props)\r\n   1060                     func = getattr(self, f\"set_{k}\", None)\r\n   1061                     if not callable(func):\r\n-> 1062                         raise AttributeError(f\"{type(self).__name__!r} object \"\r\n   1063                                              f\"has no property {k!r}\")\r\n   1064                     ret.append(func(v))\r\n\r\nAttributeError: 'LineCollection' object has no property 'drawstyle'\r\n\r\n```\r\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Operating system: Mac OS, Linux, Windows (all tested on CI)\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): 3.4.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): various\r\n  * Python version: 3.8\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\nI installed matplotlib from pip (on CI) and conda (my laptop)\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
      "issue_closed_at": "2021-04-09T00:06:34Z",
      "base_commit": "0c29bc0dd8169870dd93d656f33b62df0e6e96a6",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_axes.py",
          "type": "function",
          "name": "errorbar",
          "class_name": "Axes",
          "code": "def errorbar(self, x, y, yerr=None, xerr=None,\n                 fmt='', ecolor=None, elinewidth=None, capsize=None,\n                 barsabove=False, lolims=False, uplims=False,\n                 xlolims=False, xuplims=False, errorevery=1, capthick=None,\n                 **kwargs):\n        \"\"\"\n        Plot y versus x as lines and/or markers with attached errorbars.\n\n        *x*, *y* define the data locations, *xerr*, *yerr* define the errorbar\n        sizes. By default, this draws the data markers/lines as well the\n        errorbars. Use fmt='none' to draw errorbars without any data markers.\n\n        Parameters\n        ----------\n        x, y : float or array-like\n            The data positions.\n\n        xerr, yerr : float or array-like, shape(N,) or shape(2, N), optional\n            The errorbar sizes:\n\n            - scalar: Symmetric +/- values for all data points.\n            - shape(N,): Symmetric +/-values for each data point.\n            - shape(2, N): Separate - and + values for each bar. First row\n              contains the lower errors, the second row contains the upper\n              errors.\n            - *None*: No errorbar.\n\n            Note that all error arrays should have *positive* values.\n\n            See :doc:`/gallery/statistics/errorbar_features`\n            for an example on the usage of ``xerr`` and ``yerr``.\n\n        fmt : str, default: ''\n            The format for the data points / data lines. See `.plot` for\n            details.\n\n            Use 'none' (case insensitive) to plot errorbars without any data\n            markers.\n\n        ecolor : color, default: None\n            The color of the errorbar lines.  If None, use the color of the\n            line connecting the markers.\n\n        elinewidth : float, default: None\n            The linewidth of the errorbar lines. If None, the linewidth of\n            the current style is used.\n\n        capsize : float, default: :rc:`errorbar.capsize`\n            The length of the error bar caps in points.\n\n        capthick : float, default: None\n            An alias to the keyword argument *markeredgewidth* (a.k.a. *mew*).\n            This setting is a more sensible name for the property that\n            controls the thickness of the error bar cap in points. For\n            backwards compatibility, if *mew* or *markeredgewidth* are given,\n            then they will over-ride *capthick*. This may change in future\n            releases.\n\n        barsabove : bool, default: False\n            If True, will plot the errorbars above the plot\n            symbols. Default is below.\n\n        lolims, uplims, xlolims, xuplims : bool, default: False\n            These arguments can be used to indicate that a value gives only\n            upper/lower limits.  In that case a caret symbol is used to\n            indicate this. *lims*-arguments may be scalars, or array-likes of\n            the same length as *xerr* and *yerr*.  To use limits with inverted\n            axes, `~.Axes.set_xlim` or `~.Axes.set_ylim` must be called before\n            :meth:`errorbar`.  Note the tricky parameter names: setting e.g.\n            *lolims* to True means that the y-value is a *lower* limit of the\n            True value, so, only an *upward*-pointing arrow will be drawn!\n\n        errorevery : int or (int, int), default: 1\n            draws error bars on a subset of the data. *errorevery* =N draws\n            error bars on the points (x[::N], y[::N]).\n            *errorevery* =(start, N) draws error bars on the points\n            (x[start::N], y[start::N]). e.g. errorevery=(6, 3)\n            adds error bars to the data at (x[6], x[9], x[12], x[15], ...).\n            Used to avoid overlapping error bars when two series share x-axis\n            values.\n\n        Returns\n        -------\n        `.ErrorbarContainer`\n            The container contains:\n\n            - plotline: `.Line2D` instance of x, y plot markers and/or line.\n            - caplines: A tuple of `.Line2D` instances of the error bar caps.\n            - barlinecols: A tuple of `.LineCollection` with the horizontal and\n              vertical error ranges.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            All other keyword arguments are passed on to the `~.Axes.plot` call\n            drawing the markers. For example, this code makes big red squares\n            with thick green edges::\n\n                x, y, yerr = rand(3, 10)\n                errorbar(x, y, yerr, marker='s', mfc='red',\n                         mec='green', ms=20, mew=4)\n\n            where *mfc*, *mec*, *ms* and *mew* are aliases for the longer\n            property names, *markerfacecolor*, *markeredgecolor*, *markersize*\n            and *markeredgewidth*.\n\n            Valid kwargs for the marker properties are `.Line2D` properties:\n\n            %(Line2D_kwdoc)s\n        \"\"\"\n        kwargs = cbook.normalize_kwargs(kwargs, mlines.Line2D)\n        # anything that comes in as 'None', drop so the default thing\n        # happens down stream\n        kwargs = {k: v for k, v in kwargs.items() if v is not None}\n        kwargs.setdefault('zorder', 2)\n\n        # Casting to object arrays preserves units.\n        if not isinstance(x, np.ndarray):\n            x = np.asarray(x, dtype=object)\n        if not isinstance(y, np.ndarray):\n            y = np.asarray(y, dtype=object)\n        if xerr is not None and not isinstance(xerr, np.ndarray):\n            xerr = np.asarray(xerr, dtype=object)\n        if yerr is not None and not isinstance(yerr, np.ndarray):\n            yerr = np.asarray(yerr, dtype=object)\n        x, y = np.atleast_1d(x, y)  # Make sure all the args are iterable.\n        if len(x) != len(y):\n            raise ValueError(\"'x' and 'y' must have the same size\")\n\n        if isinstance(errorevery, Integral):\n            errorevery = (0, errorevery)\n        if isinstance(errorevery, tuple):\n            if (len(errorevery) == 2 and\n                    isinstance(errorevery[0], Integral) and\n                    isinstance(errorevery[1], Integral)):\n                errorevery = slice(errorevery[0], None, errorevery[1])\n            else:\n                raise ValueError(\n                    f'errorevery={errorevery!r} is a not a tuple of two '\n                    f'integers')\n        elif isinstance(errorevery, slice):\n            pass\n        elif not isinstance(errorevery, str) and np.iterable(errorevery):\n            # fancy indexing\n            try:\n                x[errorevery]\n            except (ValueError, IndexError) as err:\n                raise ValueError(\n                    f\"errorevery={errorevery!r} is iterable but not a valid \"\n                    f\"NumPy fancy index to match 'xerr'/'yerr'\") from err\n        else:\n            raise ValueError(\n                f\"errorevery={errorevery!r} is not a recognized value\")\n        everymask = np.zeros(len(x), bool)\n        everymask[errorevery] = True\n\n        label = kwargs.pop(\"label\", None)\n        kwargs['label'] = '_nolegend_'\n\n        # Create the main line and determine overall kwargs for child artists.\n        # We avoid calling self.plot() directly, or self._get_lines(), because\n        # that would call self._process_unit_info again, and do other indirect\n        # data processing.\n        (data_line, base_style), = self._get_lines._plot_args(\n            (x, y) if fmt == '' else (x, y, fmt), kwargs, return_kwargs=True)\n\n        # Do this after creating `data_line` to avoid modifying `base_style`.\n        if barsabove:\n            data_line.set_zorder(kwargs['zorder'] - .1)\n        else:\n            data_line.set_zorder(kwargs['zorder'] + .1)\n\n        # Add line to plot, or throw it away and use it to determine kwargs.\n        if fmt.lower() != 'none':\n            self.add_line(data_line)\n        else:\n            data_line = None\n            # Remove alpha=0 color that _get_lines._plot_args returns for\n            # 'none' format, and replace it with user-specified color, if\n            # supplied.\n            base_style.pop('color')\n            if 'color' in kwargs:\n                base_style['color'] = kwargs.pop('color')\n\n        if 'color' not in base_style:\n            base_style['color'] = 'C0'\n        if ecolor is None:\n            ecolor = base_style['color']\n\n        # Eject any marker information from line format string, as it's not\n        # needed for bars or caps.\n        base_style.pop('marker', None)\n        base_style.pop('markersize', None)\n        base_style.pop('markerfacecolor', None)\n        base_style.pop('markeredgewidth', None)\n        base_style.pop('markeredgecolor', None)\n        base_style.pop('markevery', None)\n        base_style.pop('linestyle', None)\n        base_style.pop('fillstyle', None)\n\n        # Make the style dict for the line collections (the bars).\n        eb_lines_style = {**base_style, 'color': ecolor}\n\n        if elinewidth:\n            eb_lines_style['linewidth'] = elinewidth\n        elif 'linewidth' in kwargs:\n            eb_lines_style['linewidth'] = kwargs['linewidth']\n\n        for key in ('transform', 'alpha', 'zorder', 'rasterized'):\n            if key in kwargs:\n                eb_lines_style[key] = kwargs[key]\n\n        # Make the style dict for caps (the \"hats\").\n        eb_cap_style = {**base_style, 'linestyle': 'none'}\n        if capsize is None:\n            capsize = rcParams[\"errorbar.capsize\"]\n        if capsize > 0:\n            eb_cap_style['markersize'] = 2. * capsize\n        if capthick is not None:\n            eb_cap_style['markeredgewidth'] = capthick\n\n        # For backwards-compat, allow explicit setting of\n        # 'markeredgewidth' to over-ride capthick.\n        for key in ('markeredgewidth', 'transform', 'alpha',\n                    'zorder', 'rasterized'):\n            if key in kwargs:\n                eb_cap_style[key] = kwargs[key]\n        eb_cap_style['color'] = ecolor\n\n        barcols = []\n        caplines = []\n\n        # arrays fine here, they are booleans and hence not units\n        lolims = np.broadcast_to(lolims, len(x)).astype(bool)\n        uplims = np.broadcast_to(uplims, len(x)).astype(bool)\n        xlolims = np.broadcast_to(xlolims, len(x)).astype(bool)\n        xuplims = np.broadcast_to(xuplims, len(x)).astype(bool)\n\n        # Vectorized fancy-indexer.\n        def apply_mask(arrays, mask): return [array[mask] for array in arrays]\n\n        def extract_err(name, err, data, lolims, uplims):\n            \"\"\"\n            Private function to compute error bars.\n\n            Parameters\n            ----------\n            name : {'x', 'y'}\n                Name used in the error message.\n            err : array-like\n                xerr or yerr from errorbar().\n            data : array-like\n                x or y from errorbar().\n            lolims : array-like\n                Error is only applied on **upper** side when this is True.  See\n                the note in the main docstring about this parameter's name.\n            uplims : array-like\n                Error is only applied on **lower** side when this is True.  See\n                the note in the main docstring about this parameter's name.\n            \"\"\"\n            try:\n                np.broadcast_to(err, (2, len(data)))\n            except ValueError:\n                raise ValueError(\n                    f\"'{name}err' (shape: {np.shape(err)}) must be a scalar \"\n                    f\"or a 1D or (2, n) array-like whose shape matches \"\n                    f\"'{name}' (shape: {np.shape(data)})\") from None\n            # This is like\n            #     low, high = np.broadcast_to(...)\n            #     return data - low * ~lolims, data + high * ~uplims\n            # except that broadcast_to would strip units.\n            return data + np.row_stack([-(1 - lolims), 1 - uplims]) * err\n\n        if xerr is not None:\n            left, right = extract_err('x', xerr, x, xlolims, xuplims)\n            barcols.append(self.hlines(\n                *apply_mask([y, left, right], everymask), **eb_lines_style))\n            # select points without upper/lower limits in x and\n            # draw normal errorbars for these points\n            noxlims = ~(xlolims | xuplims)\n            if noxlims.any() and capsize > 0:\n                yo, lo, ro = apply_mask([y, left, right], noxlims & everymask)\n                caplines.extend([\n                    mlines.Line2D(lo, yo, marker='|', **eb_cap_style),\n                    mlines.Line2D(ro, yo, marker='|', **eb_cap_style)])\n            if xlolims.any():\n                xo, yo, ro = apply_mask([x, y, right], xlolims & everymask)\n                if self.xaxis_inverted():\n                    marker = mlines.CARETLEFTBASE\n                else:\n                    marker = mlines.CARETRIGHTBASE\n                caplines.append(mlines.Line2D(\n                    ro, yo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='|', **eb_cap_style))\n            if xuplims.any():\n                xo, yo, lo = apply_mask([x, y, left], xuplims & everymask)\n                if self.xaxis_inverted():\n                    marker = mlines.CARETRIGHTBASE\n                else:\n                    marker = mlines.CARETLEFTBASE\n                caplines.append(mlines.Line2D(\n                    lo, yo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='|', **eb_cap_style))\n\n        if yerr is not None:\n            lower, upper = extract_err('y', yerr, y, lolims, uplims)\n            barcols.append(self.vlines(\n                *apply_mask([x, lower, upper], everymask), **eb_lines_style))\n            # select points without upper/lower limits in y and\n            # draw normal errorbars for these points\n            noylims = ~(lolims | uplims)\n            if noylims.any() and capsize > 0:\n                xo, lo, uo = apply_mask([x, lower, upper], noylims & everymask)\n                caplines.extend([\n                    mlines.Line2D(xo, lo, marker='_', **eb_cap_style),\n                    mlines.Line2D(xo, uo, marker='_', **eb_cap_style)])\n            if lolims.any():\n                xo, yo, uo = apply_mask([x, y, upper], lolims & everymask)\n                if self.yaxis_inverted():\n                    marker = mlines.CARETDOWNBASE\n                else:\n                    marker = mlines.CARETUPBASE\n                caplines.append(mlines.Line2D(\n                    xo, uo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='_', **eb_cap_style))\n            if uplims.any():\n                xo, yo, lo = apply_mask([x, y, lower], uplims & everymask)\n                if self.yaxis_inverted():\n                    marker = mlines.CARETUPBASE\n                else:\n                    marker = mlines.CARETDOWNBASE\n                caplines.append(mlines.Line2D(\n                    xo, lo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='_', **eb_cap_style))\n\n        for l in caplines:\n            self.add_line(l)\n\n        self._request_autoscale_view()\n        errorbar_container = ErrorbarContainer(\n            (data_line, tuple(caplines), tuple(barcols)),\n            has_xerr=(xerr is not None), has_yerr=(yerr is not None),\n            label=label)\n        self.containers.append(errorbar_container)\n\n        return errorbar_container"
        }
      ]
    }
  ]
}