{
  "instance_id": "matplotlib__matplotlib-25498",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-03-18T17:01:11Z",
  "problem_statement": "Update colorbar after changing mappable.norm\nHow can I update a colorbar, after I changed the norm instance of the colorbar?\n\n`colorbar.update_normal(mappable)` has now effect and `colorbar.update_bruteforce(mappable)` throws a `ZeroDivsionError`-Exception.\n\nConsider this example:\n\n``` python\nimport matplotlib.pyplot as plt\nfrom matplotlib.colors import LogNorm\nimport numpy as np\n\nimg = 10**np.random.normal(1, 1, size=(50, 50))\n\nfig, ax = plt.subplots(1, 1)\nplot = ax.imshow(img, cmap='gray')\ncb = fig.colorbar(plot, ax=ax)\nplot.norm = LogNorm()\ncb.update_normal(plot)  # no effect\ncb.update_bruteforce(plot)  # throws ZeroDivisionError\nplt.show()\n```\n\nOutput for `cb.update_bruteforce(plot)`:\n\n```\nTraceback (most recent call last):\n  File \"test_norm.py\", line 12, in <module>\n    cb.update_bruteforce(plot)\n  File \"/home/maxnoe/.local/anaconda3/lib/python3.4/site-packages/matplotlib/colorbar.py\", line 967, in update_bruteforce\n    self.draw_all()\n  File \"/home/maxnoe/.local/anaconda3/lib/python3.4/site-packages/matplotlib/colorbar.py\", line 342, in draw_all\n    self._process_values()\n  File \"/home/maxnoe/.local/anaconda3/lib/python3.4/site-packages/matplotlib/colorbar.py\", line 664, in _process_values\n    b = self.norm.inverse(self._uniform_y(self.cmap.N + 1))\n  File \"/home/maxnoe/.local/anaconda3/lib/python3.4/site-packages/matplotlib/colors.py\", line 1011, in inverse\n    return vmin * ma.power((vmax / vmin), val)\nZeroDivisionError: division by zero\n```\n\n",
  "patch": "diff --git a/lib/matplotlib/colorbar.py b/lib/matplotlib/colorbar.py\n--- a/lib/matplotlib/colorbar.py\n+++ b/lib/matplotlib/colorbar.py\n@@ -301,11 +301,6 @@ def __init__(self, ax, mappable=None, *, cmap=None,\n         if mappable is None:\n             mappable = cm.ScalarMappable(norm=norm, cmap=cmap)\n \n-        # Ensure the given mappable's norm has appropriate vmin and vmax\n-        # set even if mappable.draw has not yet been called.\n-        if mappable.get_array() is not None:\n-            mappable.autoscale_None()\n-\n         self.mappable = mappable\n         cmap = mappable.cmap\n         norm = mappable.norm\n@@ -1101,7 +1096,10 @@ def _process_values(self):\n             b = np.hstack((b, b[-1] + 1))\n \n         # transform from 0-1 to vmin-vmax:\n+        if self.mappable.get_array() is not None:\n+            self.mappable.autoscale_None()\n         if not self.norm.scaled():\n+            # If we still aren't scaled after autoscaling, use 0, 1 as default\n             self.norm.vmin = 0\n             self.norm.vmax = 1\n         self.norm.vmin, self.norm.vmax = mtransforms.nonsingular(\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": 4308,
      "pr_title": "fixes #2885, #3935, #3693, for hatched fill",
      "pr_body": "The last patch only removed clipping for the colour fill, hatches in fill_between were still clipped. Passing through rgbFace fixes this.\n",
      "issue_id": 2885,
      "issue_title": "PGF backend messes up fill",
      "issue_body": "I have a figure that I need to compile using PGF backend because the \\Game symbol is incorrect (another issue will be opened for that problem).\n\nBut when I do, the curves bounding the fill seem to get cropped to the visible window, which breaks the fill, as seen here:\n\n![ddlabeled](https://f.cloud.github.com/assets/1859947/2372461/caffe8b6-a83a-11e3-96a1-6fbc26fbdefb.jpg)\n\nWhen using the other backends, this behavior is not seen, for instance, \n![ddlabeled](https://f.cloud.github.com/assets/1859947/2372475/e6b08f70-a83a-11e3-9eaf-8f13e9886077.png)\n",
      "issue_closed_at": "2015-03-19T21:09:05Z",
      "base_commit": "a488cbc3926d8409270a7c79b7923268fdc93dce",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_pgf.py",
          "type": "function",
          "name": "draw_path",
          "class_name": "RendererPgf",
          "code": "def draw_path(self, gc, path, transform, rgbFace=None):\n        writeln(self.fh, r\"\\begin{pgfscope}\")\n        # draw the path\n        self._print_pgf_clip(gc)\n        self._print_pgf_path_styles(gc, rgbFace)\n        self._print_pgf_path(gc, path, transform, rgbFace)\n        self._pgf_path_draw(stroke=gc.get_linewidth() != 0.0,\n                            fill=rgbFace is not None)\n        writeln(self.fh, r\"\\end{pgfscope}\")\n\n        # if present, draw pattern on top\n        if gc.get_hatch():\n            writeln(self.fh, r\"\\begin{pgfscope}\")\n            self._print_pgf_path_styles(gc, rgbFace)\n\n            # combine clip and path for clipping\n            self._print_pgf_clip(gc)\n            self._print_pgf_path(gc, path, transform)\n            writeln(self.fh, r\"\\pgfusepath{clip}\")\n\n            # build pattern definition\n            writeln(self.fh, r\"\\pgfsys@defobject{currentpattern}{\\pgfqpoint{0in}{0in}}{\\pgfqpoint{1in}{1in}}{\")\n            writeln(self.fh, r\"\\begin{pgfscope}\")\n            writeln(self.fh, r\"\\pgfpathrectangle{\\pgfqpoint{0in}{0in}}{\\pgfqpoint{1in}{1in}}\")\n            writeln(self.fh, r\"\\pgfusepath{clip}\")\n            scale = mpl.transforms.Affine2D().scale(self.dpi)\n            self._print_pgf_path(None, gc.get_hatch_path(), scale)\n            self._pgf_path_draw(stroke=True)\n            writeln(self.fh, r\"\\end{pgfscope}\")\n            writeln(self.fh, r\"}\")\n            # repeat pattern, filling the bounding rect of the path\n            f = 1. / self.dpi\n            (xmin, ymin), (xmax, ymax) = path.get_extents(transform).get_points()\n            xmin, xmax = f * xmin, f * xmax\n            ymin, ymax = f * ymin, f * ymax\n            repx, repy = int(math.ceil(xmax-xmin)), int(math.ceil(ymax-ymin))\n            writeln(self.fh, r\"\\pgfsys@transformshift{%fin}{%fin}\" % (xmin, ymin))\n            for iy in range(repy):\n                for ix in range(repx):\n                    writeln(self.fh, r\"\\pgfsys@useobject{currentpattern}{}\")\n                    writeln(self.fh, r\"\\pgfsys@transformshift{1in}{0in}\")\n                writeln(self.fh, r\"\\pgfsys@transformshift{-%din}{0in}\" % repx)\n                writeln(self.fh, r\"\\pgfsys@transformshift{0in}{1in}\")\n\n            writeln(self.fh, r\"\\end{pgfscope}\")"
        }
      ]
    },
    {
      "pr_number": 3110,
      "pr_title": "BUG: Add Figure.delcolorbar() to fully delete a colorbar",
      "pr_body": "Closes #2688.\n\nThis creates a new Figure.delcolorbar() function, named to mimic Figure.delaxes().  The new function will delete a colorbar.  If that colorbar was creating with `use_gridspec=True`, the function will also reset the gridspec.  I added two tests to confirm that the function works as intended for colorbars creating with `use_gridspec` set to either `True` or `False`.\n",
      "issue_id": 2688,
      "issue_title": "Deleting axis in matplotlib > v1.2.1 does not work similar to v1.1.1",
      "issue_body": "This issue is described in this Stackoverflow question:\n\nhttp://stackoverflow.com/questions/17052380/deleting-axis-in-matplotlib-v1-2-1-does-not-work-similar-to-v1-1-1\n\nI am using matplotlib (python win32 v2.7.5) to plot contour plots with color bars that are animated or the contour gets updated. In order to update the plot, I delete the color bar axis while keeping the original plot axis untouched. In version 1.1.1 of matplotlib, the program was working okay, however, when I upgraded to 1.2.1 (and higher) of matplotlib I started to notice my plot is getting squashed to the left.\n\nThe following code demos the issues.  Run it in version 1.1.1 and then in 1.2.1 to see the differences. \n\n``` python\nimport matplotlib.pyplot as plt\nimport numpy as np\nfrom matplotlib.mlab import bivariate_normal\nfrom matplotlib.colors import LogNorm\n\ndelta = 0.5\nx = np.arange(-3.0, 4.001, delta)\ny = np.arange(-4.0, 3.001, delta)\nX, Y = np.meshgrid(x, y)\nZ = bivariate_normal(X, Y, 1.0, 1.0, 0.0, 0.0)\nfig  = plt.figure()\nfig.subplots_adjust(left=0.1, bottom=0.1, right=0.97, top=0.92)\nax   = fig.add_subplot(1,1,1)\naxim = ax.imshow(Z,norm = LogNorm())\ncb   = fig.colorbar(axim)\n\n##Note:  These are not replicated, I put them here to show how a refresh of the \n##          contour plot multiple times will look like\nfig.delaxes(fig.axes[1])\nfig.subplots_adjust(left=0.1, bottom=0.1, right=0.97, top=0.92)\ncb   = fig.colorbar(axim)\n\nfig.delaxes(fig.axes[1])\nfig.subplots_adjust(left=0.1, bottom=0.1, right=0.97, top=0.92)\ncb   = fig.colorbar(axim)\n\nplt.show()\n```\n\nIf i delete ALL axis and re-create them each time, then it looks ok even in new version, but I am thinking that this is not efficient process with animation. \n",
      "issue_closed_at": "2014-06-11T16:58:27Z",
      "base_commit": "0b481b579e31c630813667321e1f022190f7a200",
      "changes": [
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "_locate",
          "class_name": "ColorbarBase",
          "code": "def _locate(self, x):\n        '''\n        Given a set of color data values, return their\n        corresponding colorbar data coordinates.\n        '''\n        if isinstance(self.norm, (colors.NoNorm, colors.BoundaryNorm)):\n            b = self._boundaries\n            xn = x\n        else:\n            # Do calculations using normalized coordinates so\n            # as to make the interpolation more accurate.\n            b = self.norm(self._boundaries, clip=False).filled()\n            xn = self.norm(x, clip=False).filled()\n\n        # The rest is linear interpolation with extrapolation at ends.\n        ii = np.searchsorted(b, xn)\n        i0 = ii - 1\n        itop = (ii == len(b))\n        ibot = (ii == 0)\n        i0[itop] -= 1\n        ii[itop] -= 1\n        i0[ibot] += 1\n        ii[ibot] += 1\n\n        db = np.take(b, ii) - np.take(b, i0)\n        y = self._y\n        dy = np.take(y, ii) - np.take(y, i0)\n        z = np.take(y, i0) + (xn - np.take(b, i0)) * dy / db\n        return z"
        },
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "update_bruteforce",
          "class_name": "Colorbar",
          "code": "def update_bruteforce(self, mappable):\n        '''\n        Destroy and rebuild the colorbar.  This is\n        intended to become obsolete, and will probably be\n        deprecated and then removed.  It is not called when\n        the pyplot.colorbar function or the Figure.colorbar\n        method are used to create the colorbar.\n\n        '''\n        # We are using an ugly brute-force method: clearing and\n        # redrawing the whole thing.  The problem is that if any\n        # properties have been changed by methods other than the\n        # colorbar methods, those changes will be lost.\n        self.ax.cla()\n        # clearing the axes will delete outline, patch, solids, and lines:\n        self.outline = None\n        self.patch = None\n        self.solids = None\n        self.lines = list()\n        self.dividers = None\n        self.set_alpha(mappable.get_alpha())\n        self.cmap = mappable.cmap\n        self.norm = mappable.norm\n        self.config_axis()\n        self.draw_all()\n        if isinstance(self.mappable, contour.ContourSet):\n            CS = self.mappable\n            if not CS.filled:\n                self.add_lines(CS)"
        },
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "colorbar_factory",
          "class_name": null,
          "code": "def colorbar_factory(cax, mappable, **kwargs):\n    \"\"\"\n    Creates a colorbar on the given axes for the given mappable.\n\n    Typically, for automatic colorbar placement given only a mappable use\n    :meth:`~matplotlib.figure.Figure.colorbar`.\n\n    \"\"\"\n    # if the given mappable is a contourset with any hatching, use\n    # ColorbarPatch else use Colorbar\n    if (isinstance(mappable, contour.ContourSet)\n            and any([hatch is not None for hatch in mappable.hatches])):\n        cb = ColorbarPatch(cax, mappable, **kwargs)\n    else:\n        cb = Colorbar(cax, mappable, **kwargs)\n\n    mappable.callbacksSM.connect('changed', cb.on_mappable_changed)\n    mappable.colorbar = cb\n\n    return cb"
        }
      ]
    },
    {
      "pr_number": 12461,
      "pr_title": "FIX: make add_lines work with new colorbar",
      "pr_body": "## PR Summary\r\n\r\nCloses #12458\r\n\r\nSee below for a less pathological example...\r\n\r\n`cbar.add_lines` was not operational after new colorbar changes in 3.0.0.  \r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport matplotlib as mpl\r\nimport numpy as np\r\n\r\nfig = plt.figure()\r\nax = fig.add_axes([0.05, 0.8, 0.9, 0.15])\r\ncmap = mpl.cm.cool\r\nnorm = mpl.colors.Normalize(vmin=-4, vmax=4)\r\nlevels = (-1.0, 1.0, 2.0, 3.0)\r\n\r\ncb = mpl.colorbar.ColorbarBase(ax, cmap=cmap, norm=norm)\r\ncolors_bg = np.tile(list((1.0, 1.0, 1.0, 1.0)), (len(levels), 1))\r\ncb.add_lines(levels=levels, colors=colors_bg, linewidths=1)\r\n\r\n```\r\n\r\n## Before\r\n![cbarold](https://user-images.githubusercontent.com/1562854/46681303-ce871d00-cb9f-11e8-8acd-e5defedfe90c.png)\r\n\r\n\r\n## After\r\n\r\n![cbarnew](https://user-images.githubusercontent.com/1562854/46681265-b3b4a880-cb9f-11e8-8efd-87b24d565751.png)\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\nfig, ax = plt.subplots()\r\nX = np.random.rand(10, 10)*10000\r\npcm = ax.pcolormesh(X)\r\n# add 1000 to make colors visible...\r\ncont = ax.contour(X+1000)\r\ncb = fig.colorbar(pcm)\r\ncb.add_lines(cont)\r\nplt.show()\r\n```\r\n\r\n## Before\r\n\r\n![before](https://user-images.githubusercontent.com/1562854/46701766-b4683180-cbd5-11e8-8c18-d994ba47e396.png)\r\n\r\n## After\r\n![new](https://user-images.githubusercontent.com/1562854/46701741-9d294400-cbd5-11e8-9006-708b7ef51fdc.png)\r\n\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) compliant\r\n- [ ] New features are documented, with examples if plot related\r\n- [ ] Documentation is sphinx and numpydoc compliant\r\n- [ ] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [ ] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->",
      "issue_id": 12458,
      "issue_title": "add_lines misses lines for matplotlib.colorbar.ColorbarBase",
      "issue_body": "<!--To help us understand and resolve your issue, please fill out the form to the best of your ability.-->\r\n<!--You can feel free to delete the sections that do not apply.-->\r\n\r\n### Bug report\r\n\r\n**Bug summary**\r\n\r\nFor matplotlib.colorbar.ColorbarBase(...) usage of add_lines(...) may yield missing lines/poorly adjusted lines.\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug, also minimizing the number of dependencies required-->\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport matplotlib as mpl\r\nimport numpy as np\r\n\r\nfig = plt.figure()\r\nax = fig.add_axes([0.05, 0.8, 0.9, 0.15])\r\ncmap = mpl.cm.cool\r\nnorm = mpl.colors.Normalize(vmin=-4, vmax=4)\r\nlevels = (-1.0, 1.0, 2.0, 3.0)\r\n\r\ncb = mpl.colorbar.ColorbarBase(ax, cmap=cmap, norm=norm)\r\ncolors_bg = np.tile(list((1.0, 1.0, 1.0, 1.0)), (len(levels), 1))\r\ncb.add_lines(levels=levels, colors=colors_bg, linewidths=7)\r\n\r\nplt.show()\r\n\r\n\r\n```\r\n\r\n**Expected outcome**\r\n\r\n4 white lines, crossing the colorbar.\r\nUsed to work on matplotlib 2.2.3!\r\n\r\n**Matplotlib version**\r\n  * Operating system: Ubuntu 18.04 LTS\r\n  * Matplotlib version: 3.0.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.7.0\r\n\r\nMiniconda install, fully updated.",
      "issue_closed_at": "2018-10-14T19:38:43Z",
      "base_commit": "f93222a555160bf967f02dae42cce39a03384774",
      "changes": [
        {
          "file": "lib/matplotlib/colorbar.py",
          "type": "function",
          "name": "add_lines",
          "class_name": "Colorbar",
          "code": "def add_lines(self, CS, erase=True):\n        '''\n        Add the lines from a non-filled\n        :class:`~matplotlib.contour.ContourSet` to the colorbar.\n\n        Set *erase* to False if these lines should be added to\n        any pre-existing lines.\n        '''\n        if not isinstance(CS, contour.ContourSet) or CS.filled:\n            raise ValueError('add_lines is only for a ContourSet of lines')\n        tcolors = [c[0] for c in CS.tcolors]\n        tlinewidths = [t[0] for t in CS.tlinewidths]\n        # The following was an attempt to get the colorbar lines\n        # to follow subsequent changes in the contour lines,\n        # but more work is needed: specifically, a careful\n        # look at event sequences, and at how\n        # to make one object track another automatically.\n        #tcolors = [col.get_colors()[0] for col in CS.collections]\n        #tlinewidths = [col.get_linewidth()[0] for lw in CS.collections]\n        ColorbarBase.add_lines(self, CS.levels, tcolors, tlinewidths,\n                               erase=erase)"
        }
      ]
    },
    {
      "pr_number": 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"
        }
      ]
    }
  ]
}