{
  "instance_id": "matplotlib__matplotlib-23964",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-09-20T13:49:19Z",
  "problem_statement": "[Bug]: Text label with empty line causes a \"TypeError: cannot unpack non-iterable NoneType object\" in PostScript backend\n### Bug summary\n\nWhen saving a figure with the PostScript backend, a\r\n> TypeError: cannot unpack non-iterable NoneType object\r\n\r\nhappens if the figure contains a multi-line text label with an empty line (see example).\n\n### Code for reproduction\n\n```python\nfrom matplotlib.figure import Figure\r\n\r\nfigure = Figure()\r\nax = figure.add_subplot(111)\r\n# ax.set_title('\\nLower title')  # this would cause an error as well\r\nax.annotate(text='\\nLower label', xy=(0, 0))\r\nfigure.savefig('figure.eps')\n```\n\n\n### Actual outcome\n\n$ ./venv/Scripts/python save_ps.py\r\nTraceback (most recent call last):\r\n  File \"C:\\temp\\matplotlib_save_ps\\save_ps.py\", line 7, in <module>\r\n    figure.savefig('figure.eps')\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\figure.py\", line 3272, in savefig\r\n    self.canvas.print_figure(fname, **kwargs)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\backend_bases.py\", line 2338, in print_figure\r\n    result = print_method(\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\backend_bases.py\", line 2204, in <lambda>\r\n    print_method = functools.wraps(meth)(lambda *args, **kwargs: meth(\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\_api\\deprecation.py\", line 410, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\backends\\backend_ps.py\", line 869, in _print_ps\r\n    printer(fmt, outfile, dpi=dpi, dsc_comments=dsc_comments,\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\backends\\backend_ps.py\", line 927, in _print_figure\r\n    self.figure.draw(renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\artist.py\", line 74, in draw_wrapper\r\n    result = draw(artist, renderer, *args, **kwargs)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\figure.py\", line 3069, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\image.py\", line 131, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\axes\\_base.py\", line 3106, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\image.py\", line 131, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\text.py\", line 1995, in draw\r\n    Text.draw(self, renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\text.py\", line 736, in draw\r\n    textrenderer.draw_text(gc, x, y, clean_line,\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\backends\\backend_ps.py\", line 248, in wrapper\r\n    return meth(self, *args, **kwargs)\r\n  File \"C:\\temp\\matplotlib_save_ps\\venv\\lib\\site-packages\\matplotlib\\backends\\backend_ps.py\", line 673, in draw_text\r\n    for ps_name, xs_names in stream:\r\nTypeError: cannot unpack non-iterable NoneType object\r\n\n\n### Expected outcome\n\nThe figure can be saved as `figure.eps` without error.\n\n### Additional information\n\n- seems to happen if a text label or title contains a linebreak with an empty line\r\n- works without error for other backends such as PNG, PDF, SVG, Qt\r\n- works with matplotlib<=3.5.3\r\n- adding `if curr_stream:` before line 669 of `backend_ps.py` seems to fix the bug \n\n### Operating system\n\nWindows\n\n### Matplotlib Version\n\n3.6.0\n\n### Matplotlib Backend\n\n_No response_\n\n### Python version\n\n3.9.13\n\n### Jupyter version\n\n_No response_\n\n### Installation\n\npip\n",
  "patch": "diff --git a/lib/matplotlib/backends/backend_ps.py b/lib/matplotlib/backends/backend_ps.py\n--- a/lib/matplotlib/backends/backend_ps.py\n+++ b/lib/matplotlib/backends/backend_ps.py\n@@ -665,8 +665,9 @@ def draw_text(self, gc, x, y, s, prop, angle, ismath=False, mtext=None):\n                 curr_stream[1].append(\n                     (item.x, item.ft_object.get_glyph_name(item.glyph_idx))\n                 )\n-            # append the last entry\n-            stream.append(curr_stream)\n+            # append the last entry if exists\n+            if curr_stream:\n+                stream.append(curr_stream)\n \n         self.set_color(*gc.get_rgb())\n \n",
  "similar_bug_items": [
    {
      "pr_number": 9661,
      "pr_title": "Fix arcs with very large width/height.",
      "pr_body": "## PR Summary\r\n\r\nLarge width/height triggers a different code path that didn't correctly\r\ndraw limited arcs instead of the whole ellipse.\r\n\r\nFixes #9659.\r\n\r\n## PR Checklist\r\n\r\n- [x] Has Pytest style unit tests\r\n- [x] Code is PEP 8 compliant\r\n- [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": 9659,
      "issue_title": "patches.Arc objects randomly drawing the full ellipse",
      "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\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n\r\nProviding theta1 and theta2 to the Arc constructor sometimes causes the arc to be drawn as the full ellipse.\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\nfrom matplotlib.patches import Arc\r\nimport matplotlib.pyplot as plt\r\n\r\npatches = [\r\n    Arc((143954140, 374), 476, 476, theta1=0, theta2=180),\r\n    Arc((143954902, 374), 2001, 749, theta1=0, theta2=180),\r\n    Arc((143955137, 374), 2471, 749, theta1=0, theta2=180),\r\n    Arc((143955276, 374), 2748, 749, theta1=0, theta2=180),\r\n    Arc((143955362, 374), 2022, 749, theta1=0, theta2=180),\r\n    Arc((143955500, 374), 2299, 749, theta1=0, theta2=180),\r\n    Arc((143955641, 374), 488, 488, theta1=0, theta2=180),\r\n    Arc((143955650, 374), 506, 506, theta1=0, theta2=180),\r\n    Arc((143955885, 374), 976, 749, theta1=0, theta2=180),\r\n    Arc((143956128, 374), 489, 489, theta1=0, theta2=180),\r\n    Arc((143956135, 374), 467, 467, theta1=0, theta2=180),\r\n    Arc((143956137, 374), 471, 471, theta1=0, theta2=180),\r\n    Arc((143956276, 374), 748, 748, theta1=0, theta2=180),\r\n    Arc((143956515, 374), 1226, 749, theta1=0, theta2=180),\r\n    Arc((143956167, 374), 412, 412, theta1=0, theta2=180),\r\n    Arc((143956170, 374), 402, 402, theta1=0, theta2=180),\r\n    Arc((143956172, 374), 401, 401, theta1=0, theta2=180),\r\n    Arc((143956183, 374), 380, 380, theta1=0, theta2=180),\r\n    Arc((143956321, 374), 657, 657, theta1=0, theta2=180),\r\n    Arc((143956610, 374), 80, 80, theta1=0, theta2=180),\r\n    Arc((143956849, 374), 558, 558, theta1=0, theta2=180),\r\n    Arc((143956928, 374), 400, 400, theta1=0, theta2=180),\r\n    Arc((143956960, 374), 335, 335, theta1=0, theta2=180),\r\n    Arc((143956962, 374), 331, 331, theta1=0, theta2=180),\r\n    Arc((143957055, 374), 146, 146, theta1=0, theta2=180),\r\n    Arc((143957346, 374), 105, 105, theta1=0, theta2=180),\r\n    Arc((143957397, 374), 207, 207, theta1=0, theta2=180),\r\n    Arc((143957448, 374), 308, 308, theta1=0, theta2=180),\r\n    Arc((143957459, 374), 331, 331, theta1=0, theta2=180),\r\n    Arc((143957475, 374), 363, 363, theta1=0, theta2=180),\r\n    Arc((143957696, 374), 804, 749, theta1=0, theta2=180),\r\n    Arc((143958871, 374), 3154, 749, theta1=0, theta2=180),\r\n    Arc((143957954, 374), 287, 287, theta1=0, theta2=180),\r\n    Arc((143957957, 374), 292, 292, theta1=0, theta2=180),\r\n    Arc((143958125, 374), 628, 628, theta1=0, theta2=180),\r\n    Arc((143958305, 374), 268, 268, theta1=0, theta2=180),\r\n    Arc((143958313, 374), 251, 251, theta1=0, theta2=180),\r\n    Arc((143958368, 374), 142, 142, theta1=0, theta2=180),\r\n    Arc((143958369, 374), 136, 136, theta1=0, theta2=180),\r\n    Arc((143958370, 374), 138, 138, theta1=0, theta2=180),\r\n    Arc((143958905, 374), 535, 535, theta1=0, theta2=180),\r\n    Arc((143959528, 374), 1780, 749, theta1=0, theta2=180),\r\n    Arc((143959543, 374), 1810, 749, theta1=0, theta2=180),\r\n    Arc((143959544, 374), 1813, 749, theta1=0, theta2=180),\r\n    Arc((143959849, 374), 1198, 749, theta1=0, theta2=180),\r\n    Arc((143960119, 374), 658, 658, theta1=0, theta2=180),\r\n    Arc((143960678, 374), 151, 151, theta1=0, theta2=180),\r\n    Arc((143960682, 374), 158, 158, theta1=0, theta2=180),\r\n    Arc((143960682, 374), 159, 159, theta1=0, theta2=180),\r\n    Arc((143960723, 374), 241, 241, theta1=0, theta2=180),\r\n    Arc((143960797, 374), 388, 388, theta1=0, theta2=180),\r\n    Arc((143960845, 374), 484, 484, theta1=0, theta2=180),\r\n    Arc((143960809, 374), 363, 363, theta1=0, theta2=180),\r\n    Arc((143960943, 374), 95, 95, theta1=0, theta2=180),\r\n    Arc((143960945, 374), 92, 92, theta1=0, theta2=180)\r\n]\r\n\r\nplt.figure()\r\nax = plt.gca()\r\nmap(ax.add_patch, patches)\r\nplt.xlim(143953697, 143961337)\r\nplt.ylim(-150, 900)\r\nplt.show()\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![image](https://user-images.githubusercontent.com/17572426/32301746-967dedf2-bf35-11e7-8263-102883c3a99a.png)\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\nAll the arcs should be half-ellipses above the line `y = 0.5`.\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 Sierra\r\n  * Matplotlib version: 2.1.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): MacOSX\r\n  * Python version: 2.7.14 (Homebrew)\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: numpy 13.3.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\npython 2.7.14 from homebrew (`brew install python`); matplotlib from pip (`sudo -H pip install matplotlib`)\r\n",
      "issue_closed_at": "2017-11-02T10:07:32Z",
      "base_commit": "c46599356379b148afe331ad8bc05fb343cf7e85",
      "changes": [
        {
          "file": "lib/matplotlib/patches.py",
          "type": "function",
          "name": "iter_circle_intersect_on_line_seg",
          "class_name": "Arc",
          "code": "def iter_circle_intersect_on_line_seg(x0, y0, x1, y1):\n            epsilon = 1e-9\n            if x1 < x0:\n                x0e, x1e = x1, x0\n            else:\n                x0e, x1e = x0, x1\n            if y1 < y0:\n                y0e, y1e = y1, y0\n            else:\n                y0e, y1e = y0, y1\n            x0e -= epsilon\n            y0e -= epsilon\n            x1e += epsilon\n            y1e += epsilon\n            for x, y in iter_circle_intersect_on_line(x0, y0, x1, y1):\n                if x >= x0e and x <= x1e and y >= y0e and y <= y1e:\n                    yield x, y"
        }
      ]
    },
    {
      "pr_number": 8047,
      "pr_title": "Correct theta values when drawing a non-circular arc",
      "pr_body": "Fixes #8046. I'm not sure if this is the right place to do the correction, so a second opinion would be good. I've checked this works with a full range of angles.",
      "issue_id": 8046,
      "issue_title": "Arc patch with starting and ending angle",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\n- Arc patch does not behave as expected when drawing elliptical arcs between two angles specified by theta1 and theta2.\r\n\r\n**Code for reproduction**\r\n```python\r\nimport matplotlib.pyplot as plt\r\nfrom matplotlib.patches import Arc\r\n\r\n# Ellipse parameters\r\nR = 1.0\r\nx, y = 0.0, 0.0\r\na, b = 2.0*R, R\r\n\r\n# Figure setup\r\nfig_width, fig_height = 3.30, 3.30\r\nfig = plt.figure(figsize=(fig_width, fig_height), frameon=False)\r\nax = fig.add_axes([0.0, 0.0, 1.0, 1.0], aspect='equal')\r\nax.set_axis_off()\r\nax.set_xlim(-R*1.05, R*1.05)\r\nax.set_ylim(-R*1.05, R*1.05)\r\n# Axes\r\nax.axhline(0.0)\r\nax.axvline(0.0)\r\n# 45 degree line\r\nax.plot([0.0, 1.0], [0.0, 1.0], 'k--')\r\n# Arcs\r\nax.add_patch(Arc((x, y), a, b, \r\n                 theta1=0.0, theta2=360.0, edgecolor='k'))\r\nax.add_patch(Arc((x, y), a, b,\r\n                 theta1=0.0, theta2=45.0, edgecolor='r', lw=1.5))\r\nfig.savefig('Arc_patch_bug.png')\r\n```\r\n\r\n**Actual outcome**\r\n\r\n![arc_patch_bug](https://cloud.githubusercontent.com/assets/20580126/22736007/3d8cc2bc-edf4-11e6-85ef-3e042599b626.png)\r\n\r\n**Expected outcome**\r\n\r\n- Red elliptical arc between the x-axis (0 degrees) and the dashed line (45 degrees), following the black ellipse.\r\n\r\n**Matplotlib version**\r\n\r\n- Matplotlib 2.0.0, Python 2.7.13, OSX\r\n- Installed with MacPorts\r\n\r\n",
      "issue_closed_at": "2017-03-10T23:28:10Z",
      "base_commit": "1f173ddfdbe44a978df7126c606588a0fc75fbd9",
      "changes": [
        {
          "file": "lib/matplotlib/patches.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ConnectionPatch",
          "code": "def __init__(self, xyA, xyB, coordsA, coordsB=None,\n                 axesA=None, axesB=None,\n                 arrowstyle=\"-\",\n                 arrow_transmuter=None,\n                 connectionstyle=\"arc3\",\n                 connector=None,\n                 patchA=None,\n                 patchB=None,\n                 shrinkA=0.,\n                 shrinkB=0.,\n                 mutation_scale=10.,\n                 mutation_aspect=None,\n                 clip_on=False,\n                 dpi_cor=1.,\n                 **kwargs):\n        \"\"\"\n        Connect point *xyA* in *coordsA* with point *xyB* in *coordsB*\n\n\n        Valid keys are\n\n\n        ===============  ======================================================\n        Key              Description\n        ===============  ======================================================\n        arrowstyle       the arrow style\n        connectionstyle  the connection style\n        relpos           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\n        *coordsA* and *coordsB* are strings that indicate the\n        coordinates of *xyA* and *xyB*.\n\n        =================   ===================================================\n        Property            Description\n        =================   ===================================================\n        'figure points'     points from the lower left corner of the figure\n        'figure pixels'     pixels from the lower left corner of the figure\n        'figure fraction'   0,0 is lower left of figure and 1,1 is upper, right\n        'axes points'       points from lower left corner of axes\n        'axes pixels'       pixels from lower left corner of axes\n        'axes fraction'     0,1 is lower left of axes and 1,1 is upper right\n        'data'              use the coordinate system of the object being\n                            annotated (default)\n        'offset points'     Specify an offset (in points) from the *xy* value\n\n        'polar'             you can specify *theta*, *r* for the annotation,\n                            even in cartesian plots.  Note that if you\n                            are using a polar axes, you do not need\n                            to specify polar for the coordinate\n                            system since that is the native \"data\" coordinate\n                            system.\n        =================   ===================================================\n\n        \"\"\"\n        if coordsB is None:\n            coordsB = coordsA\n        # we'll draw ourself after the artist we annotate by default\n        self.xy1 = xyA\n        self.xy2 = xyB\n        self.coords1 = coordsA\n        self.coords2 = coordsB\n\n        self.axesA = axesA\n        self.axesB = axesB\n\n        FancyArrowPatch.__init__(self,\n                                 posA=(0, 0), posB=(1, 1),\n                                 arrowstyle=arrowstyle,\n                                 arrow_transmuter=arrow_transmuter,\n                                 connectionstyle=connectionstyle,\n                                 connector=connector,\n                                 patchA=patchA,\n                                 patchB=patchB,\n                                 shrinkA=shrinkA,\n                                 shrinkB=shrinkB,\n                                 mutation_scale=mutation_scale,\n                                 mutation_aspect=mutation_aspect,\n                                 clip_on=clip_on,\n                                 dpi_cor=dpi_cor,\n                                 **kwargs)\n\n        # if True, draw annotation only if self.xy is inside the axes\n        self._annotation_clip = None"
        },
        {
          "file": "lib/matplotlib/patches.py",
          "type": "function",
          "name": "draw",
          "class_name": "ConnectionPatch",
          "code": "def draw(self, renderer):\n        \"\"\"\n        Draw.\n        \"\"\"\n\n        if renderer is not None:\n            self._renderer = renderer\n        if not self.get_visible():\n            return\n\n        if not self._check_xy(renderer):\n            return\n\n        FancyArrowPatch.draw(self, renderer)"
        },
        {
          "file": "lib/matplotlib/patches.py",
          "type": "function",
          "name": "iter_circle_intersect_on_line_seg",
          "class_name": "Arc",
          "code": "def iter_circle_intersect_on_line_seg(x0, y0, x1, y1):\n            epsilon = 1e-9\n            if x1 < x0:\n                x0e, x1e = x1, x0\n            else:\n                x0e, x1e = x0, x1\n            if y1 < y0:\n                y0e, y1e = y1, y0\n            else:\n                y0e, y1e = y0, y1\n            x0e -= epsilon\n            y0e -= epsilon\n            x1e += epsilon\n            y1e += epsilon\n            for x, y in iter_circle_intersect_on_line(x0, y0, x1, y1):\n                if x >= x0e and x <= x1e and y >= y0e and y <= y1e:\n                    yield x, y"
        }
      ]
    },
    {
      "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": 2587,
      "pr_title": "Make backend_pgf more flexible when saving to file-handles or streams (fix #1625).",
      "pr_body": "Since binary data and thus raster images cannot be embedded in pgf code, saving to a stream was prohibited in backend_pgf. This PR allows saving pgf code to streams after deactivating draw_image() and emitting a user warning.\n\nI'll merge this in a few days if there are no objections (fixing issue #1625).\n",
      "issue_id": 1625,
      "issue_title": "saving pgf to a stream is not supported",
      "issue_body": "Hi,\n\nThe pgf output does not support a stream as illustrated by this snippet\n\n``` python\nimport sys\nplt.plot([1,2])\nplt.savefig(sys.stdout, format='pgf')\n```\n\nIt returns:\n\n``` python\nValueError: saving pgf to a stream is not supported, consider using the pdf option of the pgf-backend\n```\n\nOther formats like png, eps, ps, svg support it. I don't see any good reason to do not support streams for pgf and I guess it is a bug :)\n\nThanks.\n",
      "issue_closed_at": "2013-11-12T13:20:52Z",
      "base_commit": "c91589c024aa4c95288173112d3d5c44e1703d27",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "line",
          "name": "line 12",
          "code": "import codecs\nimport atexit\nimport weakref\n\nimport matplotlib as mpl\nfrom matplotlib.backend_bases import RendererBase, GraphicsContextBase,\\"
        },
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FigureManagerPgf",
          "code": "def __init__(self, *args):\n        FigureManagerBase.__init__(self, *args)"
        },
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "print_pgf",
          "class_name": "FigureCanvasPgf",
          "code": "def print_pgf(self, fname_or_fh, *args, **kwargs):\n        \"\"\"\n        Output pgf commands for drawing the figure so it can be included and\n        rendered in latex documents.\n        \"\"\"\n        if kwargs.get(\"dryrun\", False):\n            self._print_pgf_to_fh(None, *args, **kwargs)\n            return\n\n        # figure out where the pgf is to be written to\n        if is_string_like(fname_or_fh):\n            with codecs.open(fname_or_fh, \"w\", encoding=\"utf-8\") as fh:\n                self._print_pgf_to_fh(fh, *args, **kwargs)\n        elif is_writable_file_like(fname_or_fh):\n            raise ValueError(\"saving pgf to a stream is not supported, \" +\n                             \"consider using the pdf option of the pgf-backend\")\n        else:\n            raise ValueError(\"filename must be a path\")"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}