{
  "instance_id": "matplotlib__matplotlib-22711",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-03-27T00:34:37Z",
  "problem_statement": "[Bug]: cannot give init value for RangeSlider widget\n### Bug summary\r\n\r\nI think `xy[4] = .25, val[0]` should be commented in /matplotlib/widgets. py\", line 915, in set_val\r\nas it prevents to initialized value for RangeSlider\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\nfrom matplotlib.widgets import RangeSlider\r\n\r\n# generate a fake image\r\nnp.random.seed(19680801)\r\nN = 128\r\nimg = np.random.randn(N, N)\r\n\r\nfig, axs = plt.subplots(1, 2, figsize=(10, 5))\r\nfig.subplots_adjust(bottom=0.25)\r\n\r\nim = axs[0].imshow(img)\r\naxs[1].hist(img.flatten(), bins='auto')\r\naxs[1].set_title('Histogram of pixel intensities')\r\n\r\n# Create the RangeSlider\r\nslider_ax = fig.add_axes([0.20, 0.1, 0.60, 0.03])\r\nslider = RangeSlider(slider_ax, \"Threshold\", img.min(), img.max(),valinit=[0.0,0.0])\r\n\r\n# Create the Vertical lines on the histogram\r\nlower_limit_line = axs[1].axvline(slider.val[0], color='k')\r\nupper_limit_line = axs[1].axvline(slider.val[1], color='k')\r\n\r\n\r\ndef update(val):\r\n    # The val passed to a callback by the RangeSlider will\r\n    # be a tuple of (min, max)\r\n\r\n    # Update the image's colormap\r\n    im.norm.vmin = val[0]\r\n    im.norm.vmax = val[1]\r\n\r\n    # Update the position of the vertical lines\r\n    lower_limit_line.set_xdata([val[0], val[0]])\r\n    upper_limit_line.set_xdata([val[1], val[1]])\r\n\r\n    # Redraw the figure to ensure it updates\r\n    fig.canvas.draw_idle()\r\n\r\n\r\nslider.on_changed(update)\r\nplt.show()\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n```python\r\n  File \"<ipython-input-52-b704c53e18d4>\", line 19, in <module>\r\n    slider = RangeSlider(slider_ax, \"Threshold\", img.min(), img.max(),valinit=[0.0,0.0])\r\n\r\n  File \"/Users/Vincent/opt/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/widgets.py\", line 778, in __init__\r\n    self.set_val(valinit)\r\n\r\n  File \"/Users/Vincent/opt/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/widgets.py\", line 915, in set_val\r\n    xy[4] = val[0], .25\r\n\r\nIndexError: index 4 is out of bounds for axis 0 with size 4\r\n```\r\n\r\n### Expected outcome\r\n\r\nrange slider with user initial values\r\n\r\n### Additional information\r\n\r\nerror can be removed by commenting this line\r\n```python\r\n\r\n    def set_val(self, val):\r\n        \"\"\"\r\n        Set slider value to *val*.\r\n\r\n        Parameters\r\n        ----------\r\n        val : tuple or array-like of float\r\n        \"\"\"\r\n        val = np.sort(np.asanyarray(val))\r\n        if val.shape != (2,):\r\n            raise ValueError(\r\n                f\"val must have shape (2,) but has shape {val.shape}\"\r\n            )\r\n        val[0] = self._min_in_bounds(val[0])\r\n        val[1] = self._max_in_bounds(val[1])\r\n        xy = self.poly.xy\r\n        if self.orientation == \"vertical\":\r\n            xy[0] = .25, val[0]\r\n            xy[1] = .25, val[1]\r\n            xy[2] = .75, val[1]\r\n            xy[3] = .75, val[0]\r\n            # xy[4] = .25, val[0]\r\n        else:\r\n            xy[0] = val[0], .25\r\n            xy[1] = val[0], .75\r\n            xy[2] = val[1], .75\r\n            xy[3] = val[1], .25\r\n            # xy[4] = val[0], .25\r\n        self.poly.xy = xy\r\n        self.valtext.set_text(self._format(val))\r\n        if self.drawon:\r\n            self.ax.figure.canvas.draw_idle()\r\n        self.val = val\r\n        if self.eventson:\r\n            self._observers.process(\"changed\", val)\r\n\r\n```\r\n\r\n### Operating system\r\n\r\nOSX\r\n\r\n### Matplotlib Version\r\n\r\n3.5.1\r\n\r\n### Matplotlib Backend\r\n\r\n_No response_\r\n\r\n### Python version\r\n\r\n3.8\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\npip\n",
  "patch": "diff --git a/lib/matplotlib/widgets.py b/lib/matplotlib/widgets.py\n--- a/lib/matplotlib/widgets.py\n+++ b/lib/matplotlib/widgets.py\n@@ -813,7 +813,10 @@ def _update_val_from_pos(self, pos):\n             val = self._max_in_bounds(pos)\n             self.set_max(val)\n         if self._active_handle:\n-            self._active_handle.set_xdata([val])\n+            if self.orientation == \"vertical\":\n+                self._active_handle.set_ydata([val])\n+            else:\n+                self._active_handle.set_xdata([val])\n \n     def _update(self, event):\n         \"\"\"Update the slider position.\"\"\"\n@@ -836,11 +839,16 @@ def _update(self, event):\n             return\n \n         # determine which handle was grabbed\n-        handle = self._handles[\n-            np.argmin(\n+        if self.orientation == \"vertical\":\n+            handle_index = np.argmin(\n+                np.abs([h.get_ydata()[0] - event.ydata for h in self._handles])\n+            )\n+        else:\n+            handle_index = np.argmin(\n                 np.abs([h.get_xdata()[0] - event.xdata for h in self._handles])\n             )\n-        ]\n+        handle = self._handles[handle_index]\n+\n         # these checks ensure smooth behavior if the handles swap which one\n         # has a higher value. i.e. if one is dragged over and past the other.\n         if handle is not self._active_handle:\n@@ -904,14 +912,22 @@ def set_val(self, val):\n             xy[2] = .75, val[1]\n             xy[3] = .75, val[0]\n             xy[4] = .25, val[0]\n+\n+            self._handles[0].set_ydata([val[0]])\n+            self._handles[1].set_ydata([val[1]])\n         else:\n             xy[0] = val[0], .25\n             xy[1] = val[0], .75\n             xy[2] = val[1], .75\n             xy[3] = val[1], .25\n             xy[4] = val[0], .25\n+\n+            self._handles[0].set_xdata([val[0]])\n+            self._handles[1].set_xdata([val[1]])\n+\n         self.poly.xy = xy\n         self.valtext.set_text(self._format(val))\n+\n         if self.drawon:\n             self.ax.figure.canvas.draw_idle()\n         self.val = val\n",
  "similar_bug_items": [
    {
      "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": 4587,
      "pr_title": "DOC: clairify auto-level behavior",
      "pr_body": "closes #4537\n",
      "issue_id": 4537,
      "issue_title": "Confusion about the number of contour levels",
      "issue_body": "According to the docstring, `plt.contour(X, Y, Z, N)` should \"contour _N_ automatically-chosen levels\".\n\nThe actual results are somewhat surprising:\n\n``` python\nimport numpy as np\nimport matplotlib.pyplot as plt\n\nnp.random.seed(0)\nxx, yy = np.mgrid[-3:3:, -3:3]\n\nfor _ in xrange(10):\n    zz = np.random.randn(6, 6)\n    cset = plt.contour(xx, yy,  zz, 9)\n    print(len(cset.levels)),\n```\n\nprints `9 9 8 8 9 9 8 8 9 9`.\n\nAdditionally, the docstring for `plt.contourf` has the same prose, but adds some more confusing behavior:\n\n``` python\nfor _ in xrange(10):\n    zz = np.random.randn(6, 6)\n    cset = plt.contourf(xx, yy,  zz, 9)\n    print(len(cset.levels)),\n```\n\nprints `11 10 11 10 10 11 10 11 11 11` (note that the `N` is the same as for the `contour` call above).\n\nAm I missing something, or is this is a bug?\n",
      "issue_closed_at": "2015-07-05T08:05:19Z",
      "base_commit": "aad1619a2ea8d9f2cd41dcee12a059623c744984",
      "changes": [
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_initialize_x_y",
          "class_name": "QuadContourSet",
          "code": "def _initialize_x_y(self, z):\n        \"\"\"\n        Return X, Y arrays such that contour(Z) will match imshow(Z)\n        if origin is not None.\n        The center of pixel Z[i,j] depends on origin:\n        if origin is None, x = j, y = i;\n        if origin is 'lower', x = j + 0.5, y = i + 0.5;\n        if origin is 'upper', x = j + 0.5, y = Nrows - i - 0.5\n        If extent is not None, x and y will be scaled to match,\n        as in imshow.\n        If origin is None and extent is not None, then extent\n        will give the minimum and maximum values of x and y.\n        \"\"\"\n        if z.ndim != 2:\n            raise TypeError(\"Input must be a 2D array.\")\n        else:\n            Ny, Nx = z.shape\n        if self.origin is None:  # Not for image-matching.\n            if self.extent is None:\n                return np.meshgrid(np.arange(Nx), np.arange(Ny))\n            else:\n                x0, x1, y0, y1 = self.extent\n                x = np.linspace(x0, x1, Nx)\n                y = np.linspace(y0, y1, Ny)\n                return np.meshgrid(x, y)\n        # Match image behavior:\n        if self.extent is None:\n            x0, x1, y0, y1 = (0, Nx, 0, Ny)\n        else:\n            x0, x1, y0, y1 = self.extent\n        dx = float(x1 - x0) / Nx\n        dy = float(y1 - y0) / Ny\n        x = x0 + (np.arange(Nx) + 0.5) * dx\n        y = y0 + (np.arange(Ny) + 0.5) * dy\n        if self.origin == 'upper':\n            y = y[::-1]\n        return np.meshgrid(x, y)"
        }
      ]
    },
    {
      "pr_number": 16534,
      "pr_title": "DOC: MaxNLocator and contour/contourf doc update (replaces #16428)",
      "pr_body": "## PR Summary\r\ncloses  #12729 \r\n\r\nI messed my github matplotlib repo settings (and not github doesn't recognize my old branch... it says `pharshalp wants to merge 1 commit into matplotlib:master from **unknown repository**`) so closed the older PR and recreated it here.\r\nSorry for the noise!\r\n**Please see #16428 the reviewers' comments, suggestions and discussion.**\r\n\r\n---------------------------\r\n`MaxNLocator`'s docs were incorrect (please see the details described in #12729).\r\n\r\nHere is a numerical experiment that I ran to determine the difference between the number of tick locations returned by `MaxNLocator` and the argument `nbins`.\r\n\r\n```python\r\nimport matplotlib.ticker as mticker\r\nimport numpy as np\r\n\r\nsteps = np.linspace(1, 10, 37)   # Overkill -- 1.  ,  1.25,  1.5 ,  1.75,  2.... , 10\r\nvmax_arr = np.linspace(0, 1, 1001)[1:]  # Overkill\r\n\r\n# number of ticks - number of bins\r\nnticks_minus_nbins = np.zeros((1001, 40, 3), dtype=np.int64)\r\nfor vmax_ind, vmax in enumerate(vmax_arr):\r\n    for nbins in range(1, 40+1):\r\n        locator = mticker.MaxNLocator(nbins, steps=steps, min_n_ticks=1)\r\n        nticks_minus_nbins[vmax_ind, nbins-1, 0] = len(locator.tick_values(0, vmax))-nbins\r\n        nticks_minus_nbins[vmax_ind, nbins-1, 1] = len(locator.tick_values(-vmax, 0))-nbins\r\n        nticks_minus_nbins[vmax_ind, nbins-1, 2] = len(locator.tick_values(-vmax, vmax))-nbins\r\nprint(np.max(nticks_minus_nbins))  # returns 2\r\n```\r\n\r\nIt turns out (as was shown in the original issue) that `MaxNLocator` returns at most `nbins+2` tick locations (`nbins+1` intervals).\r\n\r\nSince contour/contourf uses MaxNLocator (as pointed out in the original issue) updated their docs as well.\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": 12729,
      "issue_title": "Docs for contour levels argument is incorrect",
      "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\nIn 2.2.3 a sentence made it into the [`contourf` docs](https://matplotlib.org/2.2.3/api/_as_gen/matplotlib.axes.Axes.contourf.html#matplotlib.axes.Axes.contourf) saying for the levels of \r\n`contourf(x,y,z,levels, ....)`\r\n\r\n> levels : int or array-like, optional\r\nDetermines the number and positions of the contour lines / regions.\r\n**If an int n, use n data intervals; i.e. draw n+1 contour lines.** The level heights are automatically chosen.\r\n\r\nThe bold sentence is obviously wrong, as seen from the below image. E.g. the top right plot has `n=4` specified, but it produces 5 intervals and 4 contour lines. When specifying `n=5`, the same 5 intervals and 4 contour lines are produced.\r\n\r\n![image](https://user-images.githubusercontent.com/23121882/47958686-96250400-dfd0-11e8-9478-09f0f7c5bbf6.png)\r\n\r\nOne may look at the generated levels as a function of the input argument,\r\n \r\n![image](https://user-images.githubusercontent.com/23121882/47958664-e2bc0f80-dfcf-11e8-89ca-d4a0fc16c3a4.png)\r\n\r\nThis is because a `MaxNLocator` [is used](https://github.com/matplotlib/matplotlib/blob/3bf907632ea3dd791fb24210a0fefba33bcab070/lib/matplotlib/contour.py#L1191).\r\n\r\n    self.locator = ticker.MaxNLocator(N + 1, min_n_ticks=1)\r\n\r\nThe [`MaxNLocator`](https://matplotlib.org/api/ticker_api.html#matplotlib.ticker.MaxNLocator) docs say\r\n\r\n> **Select no more than N intervals at nice locations.**\r\n\r\nI would deduce, that the `contourf` docstring should rather say\r\n\r\n**If an int n, use m data intervals, where m <= n+1; i.e. draw m-1 contour lines.**\r\n\r\n\r\nHowever, using different data, one may even find\r\n\r\n```\r\nlocator = mticker.MaxNLocator(4+1, min_n_ticks=1)\r\nprint(len(locator.tick_values(-0.11109, 1-0.1111)))\r\n# prints 7\r\n\r\nlocator = mticker.MaxNLocator(9+1, min_n_ticks=1)\r\nprint(len(locator.tick_values(-0.11109, 1-0.1111)))\r\n# prints 12\r\n```\r\n\r\nwhich might suggest that also the `MaxNLocator` docstring is wrong, and should really say\r\n\r\n> **Select no more than N+1 intervals at nice locations.**\r\n\r\nBut maybe other data would even produce `N+x` intervals?\r\n\r\n**ToDo:**\r\n1. Find out how many intervals `MaxNLocator` is capable of producing.\r\n2. Correct `MaxNLocator` docstring\r\n3. Based on that, correct `contourf` docstring.",
      "issue_closed_at": "2020-03-02T19:14:29Z",
      "base_commit": "7d10cb61723faf8cef4422ad959ac9e1e7d1209d",
      "changes": [
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_initialize_x_y",
          "class_name": "QuadContourSet",
          "code": "def _initialize_x_y(self, z):\n        \"\"\"\n        Return X, Y arrays such that contour(Z) will match imshow(Z)\n        if origin is not None.\n        The center of pixel Z[i, j] depends on origin:\n        if origin is None, x = j, y = i;\n        if origin is 'lower', x = j + 0.5, y = i + 0.5;\n        if origin is 'upper', x = j + 0.5, y = Nrows - i - 0.5\n        If extent is not None, x and y will be scaled to match,\n        as in imshow.\n        If origin is None and extent is not None, then extent\n        will give the minimum and maximum values of x and y.\n        \"\"\"\n        if z.ndim != 2:\n            raise TypeError(f\"Input z must be 2D, not {z.ndim}D\")\n        elif z.shape[0] < 2 or z.shape[1] < 2:\n            raise TypeError(f\"Input z must be at least a (2, 2) shaped array, \"\n                            f\"but has shape {z.shape}\")\n        else:\n            Ny, Nx = z.shape\n        if self.origin is None:  # Not for image-matching.\n            if self.extent is None:\n                return np.meshgrid(np.arange(Nx), np.arange(Ny))\n            else:\n                x0, x1, y0, y1 = self.extent\n                x = np.linspace(x0, x1, Nx)\n                y = np.linspace(y0, y1, Ny)\n                return np.meshgrid(x, y)\n        # Match image behavior:\n        if self.extent is None:\n            x0, x1, y0, y1 = (0, Nx, 0, Ny)\n        else:\n            x0, x1, y0, y1 = self.extent\n        dx = (x1 - x0) / Nx\n        dy = (y1 - y0) / Ny\n        x = x0 + (np.arange(Nx) + 0.5) * dx\n        y = y0 + (np.arange(Ny) + 0.5) * dy\n        if self.origin == 'upper':\n            y = y[::-1]\n        return np.meshgrid(x, y)"
        },
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "ge",
          "class_name": "_Edge_integer",
          "code": "def ge(self, x):\n        \"\"\"Return the smallest n: n*step >= x.\"\"\"\n        d, m = divmod(x, self.step)\n        if self.closeto(m / self.step, 0):\n            return d\n        return d + 1"
        }
      ]
    },
    {
      "pr_number": 10935,
      "pr_title": "FIX: Postscript allow empty markers",
      "pr_body": "## PR Summary\r\n\r\nIf you set a marker to have `markerfacecolor='none'` with the eps backend, no marker is drawn, even if the edgecolor is non-zero.  Thats silly.  This checks and if none, just doesn't issue the fill command instead of returning before anything is drawn...\r\n\r\nI couldn't figure out how to make a postscript test.  The saved image is in pdf, but the test spits out an eps, so...\r\n\r\nCloses #10921\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is PEP 8 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": 10921,
      "issue_title": "No markers in EPS if fillstyle='none'",
      "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\nIf `fillstyle='none'`, on `savefig()` to EPS there are no markers at all. \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 numpy as np\r\n\r\nb = np.random.rand(50)\r\n\r\nplt.figure()\r\nmarker_style = dict(markersize=10, fillstyle='none')\r\nplt.plot(b, '--s', **marker_style)\r\nplt.savefig('no_markers_without_filling.eps', format='eps')\r\nplt.close()\r\n```\r\n\r\n**Actual outcome**\r\n\r\nNo markers at all.\r\n\r\n**Expected outcome**\r\n\r\nNot filled markers, or at least filled markers.\r\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Operating system: Linux 4.15.13-1-ARCH x86_64 GNU/Linux\r\n  * Matplotlib version: 2.2.2\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): Qt5Agg\r\n  * Python version: 3.6.4 (default, Jan  5 2018, 02:35:40) [GCC 7.2.1 20171224]\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\nAll installed from ArchLinux's official repos.\r\n\r\n\r\n",
      "issue_closed_at": "2018-04-01T17:27:57Z",
      "base_commit": "3599db9798fa567436b6b8cd6c6dff68e91978ad",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "draw_markers",
          "class_name": "RendererPS",
          "code": "def draw_markers(\n            self, gc, marker_path, marker_trans, path, trans, rgbFace=None):\n        \"\"\"\n        Draw the markers defined by path at each of the positions in x\n        and y.  path coordinates are points, x and y coords will be\n        transformed by the transform\n        \"\"\"\n        if debugPS: self._pswriter.write('% draw_markers \\n')\n\n        if rgbFace:\n            if len(rgbFace) == 4 and rgbFace[3] == 0:\n                return\n            if rgbFace[0] == rgbFace[1] == rgbFace[2]:\n                ps_color = '%1.3f setgray' % rgbFace[0]\n            else:\n                ps_color = '%1.3f %1.3f %1.3f setrgbcolor' % rgbFace[:3]\n\n        # construct the generic marker command:\n        ps_cmd = ['/o {', 'gsave', 'newpath', 'translate'] # don't want the translate to be global\n\n        lw = gc.get_linewidth()\n        stroke = lw != 0.0\n        if stroke:\n            ps_cmd.append('%.1f setlinewidth' % lw)\n            jint = gc.get_joinstyle()\n            ps_cmd.append('%d setlinejoin' % jint)\n            cint = gc.get_capstyle()\n            ps_cmd.append('%d setlinecap' % cint)\n\n        ps_cmd.append(self._convert_path(marker_path, marker_trans,\n                                         simplify=False))\n\n        if rgbFace:\n            if stroke:\n                ps_cmd.append('gsave')\n            ps_cmd.extend([ps_color, 'fill'])\n            if stroke:\n                ps_cmd.append('grestore')\n\n        if stroke:\n            ps_cmd.append('stroke')\n        ps_cmd.extend(['grestore', '} bind def'])\n\n        for vertices, code in path.iter_segments(\n                trans,\n                clip=(0, 0, self.width*72, self.height*72),\n                simplify=False):\n            if len(vertices):\n                x, y = vertices[-2:]\n                ps_cmd.append(\"%g %g o\" % (x, y))\n\n        ps = '\\n'.join(ps_cmd)\n        self._draw_ps(ps, gc, rgbFace, fill=False, stroke=False)"
        },
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "draw_markers",
          "class_name": "RendererPS",
          "code": "def draw_markers(\n            self, gc, marker_path, marker_trans, path, trans, rgbFace=None):\n        \"\"\"\n        Draw the markers defined by path at each of the positions in x\n        and y.  path coordinates are points, x and y coords will be\n        transformed by the transform\n        \"\"\"\n        if debugPS: self._pswriter.write('% draw_markers \\n')\n\n        if rgbFace:\n            if len(rgbFace) == 4 and rgbFace[3] == 0:\n                return\n            if rgbFace[0] == rgbFace[1] == rgbFace[2]:\n                ps_color = '%1.3f setgray' % rgbFace[0]\n            else:\n                ps_color = '%1.3f %1.3f %1.3f setrgbcolor' % rgbFace[:3]\n\n        # construct the generic marker command:\n        ps_cmd = ['/o {', 'gsave', 'newpath', 'translate'] # don't want the translate to be global\n\n        lw = gc.get_linewidth()\n        stroke = lw != 0.0\n        if stroke:\n            ps_cmd.append('%.1f setlinewidth' % lw)\n            jint = gc.get_joinstyle()\n            ps_cmd.append('%d setlinejoin' % jint)\n            cint = gc.get_capstyle()\n            ps_cmd.append('%d setlinecap' % cint)\n\n        ps_cmd.append(self._convert_path(marker_path, marker_trans,\n                                         simplify=False))\n\n        if rgbFace:\n            if stroke:\n                ps_cmd.append('gsave')\n            ps_cmd.extend([ps_color, 'fill'])\n            if stroke:\n                ps_cmd.append('grestore')\n\n        if stroke:\n            ps_cmd.append('stroke')\n        ps_cmd.extend(['grestore', '} bind def'])\n\n        for vertices, code in path.iter_segments(\n                trans,\n                clip=(0, 0, self.width*72, self.height*72),\n                simplify=False):\n            if len(vertices):\n                x, y = vertices[-2:]\n                ps_cmd.append(\"%g %g o\" % (x, y))\n\n        ps = '\\n'.join(ps_cmd)\n        self._draw_ps(ps, gc, rgbFace, fill=False, stroke=False)"
        }
      ]
    },
    {
      "pr_number": 2231,
      "pr_title": "FIX See also in axes docstrings wasn't formatted properly",
      "pr_body": "closes #2224\n\nSorry about that... I had a syntax error in the docstring.\n",
      "issue_id": 2224,
      "issue_title": "New docs build failure",
      "issue_body": "There's a problem with the docs build on Launchpad again. Looks like #2209 is the most likely source.\n\n```\nreading sources... [  5%] api/pyplot_api\n\nException occurred:\n  File \"/usr/lib/python2.7/dist-packages/sphinx/ext/autodoc.py\", line 399, in format_signature\n    self.object, self.options, args, retann)\n  File \"/usr/lib/python2.7/dist-packages/sphinx/application.py\", line 318, in emit_firstresult\n    for result in self.emit(event, *args):\n  File \"/usr/lib/python2.7/dist-packages/sphinx/application.py\", line 314, in emit\n    results.append(callback(self, *args))\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/numpydoc.py\", line 83, in mangle_signature\n    doc = SphinxDocString(pydoc.getdoc(obj))\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/docscrape_sphinx.py\", line 8, in __init__\n    NumpyDocString.__init__(self, docstring, config=config)\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/docscrape.py\", line 110, in __init__\n    self._parse()\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/docscrape.py\", line 293, in _parse\n    self['See Also'] = self._parse_see_also(content)\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/docscrape.py\", line 231, in _parse_see_also\n    push_item(current_func, rest)\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/docscrape.py\", line 213, in push_item\n    name, role = parse_item_name(name)\n  File \"/usr/lib/python2.7/dist-packages/numpydoc/docscrape.py\", line 208, in parse_item_name\n    raise ValueError(\"%s is not a item name\" % text)\nValueError: `axhspan` for example plot and source code is not a item name\n```\n",
      "issue_closed_at": "2013-07-19T12:03:35Z",
      "base_commit": "037f25c36e8fa96a10f88ac626043863185e0edb",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_axes.py",
          "type": "function",
          "name": "axhline",
          "class_name": "Axes",
          "code": "def axhline(self, y=0, xmin=0, xmax=1, **kwargs):\n        \"\"\"\n        Add a horizontal line across the axis.\n\n        Parameters\n        ----------\n        y : scalar, optional, default: 0\n            y position in data coordinates of the horizontal line.\n\n        xmin : scalar, optional, default: 0\n            Should be between 0 and 1, 0 being the far left of the plot, 1 the\n            far right of the plot.\n\n        xmax : scalar, optional, default: 1\n            Should be between 0 and 1, 0 being the far left of the plot, 1 the\n            far right of the plot.\n\n        Returns\n        -------\n        `~matplotlib.lines.Line2D`\n\n        Notes\n        -----\n        kwargs are the same as kwargs to plot, and can be\n        used to control the line properties.  e.g.,\n\n        Examples\n        --------\n\n        * draw a thick red hline at 'y' = 0 that spans the xrange::\n\n            >>> axhline(linewidth=4, color='r')\n\n        * draw a default hline at 'y' = 1 that spans the xrange::\n\n            >>> axhline(y=1)\n\n        * draw a default hline at 'y' = .5 that spans the the middle half of\n          the xrange::\n\n            >>> axhline(y=.5, xmin=0.25, xmax=0.75)\n\n        Valid kwargs are :class:`~matplotlib.lines.Line2D` properties,\n        with the exception of 'transform':\n\n        %(Line2D)s\n\n        See also\n        --------\n        `axhspan` for example plot and source code\n        \"\"\"\n\n        if \"transform\" in kwargs:\n            raise ValueError(\n                \"'transform' is not allowed as a kwarg;\"\n                + \"axhline generates its own transform.\")\n        ymin, ymax = self.get_ybound()\n\n        # We need to strip away the units for comparison with\n        # non-unitized bounds\n        self._process_unit_info(ydata=y, kwargs=kwargs)\n        yy = self.convert_yunits(y)\n        scaley = (yy < ymin) or (yy > ymax)\n\n        trans = mtransforms.blended_transform_factory(\n            self.transAxes, self.transData)\n        l = mlines.Line2D([xmin, xmax], [y, y], transform=trans, **kwargs)\n        self.add_line(l)\n        self.autoscale_view(scalex=False, scaley=scaley)\n        return l"
        },
        {
          "file": "lib/matplotlib/axes/_axes.py",
          "type": "function",
          "name": "axvline",
          "class_name": "Axes",
          "code": "def axvline(self, x=0, ymin=0, ymax=1, **kwargs):\n        \"\"\"\n        Add a vertical line across the axes.\n\n        Parameters\n        ----------\n        x : scalar, optional, default: 0\n            y position in data coordinates of the vertical line.\n\n        ymin : scalar, optional, default: 0\n            Should be between 0 and 1, 0 being the far left of the plot, 1 the\n            far right of the plot.\n\n        ymax : scalar, optional, default: 1\n            Should be between 0 and 1, 0 being the far left of the plot, 1 the\n            far right of the plot.\n\n        Returns\n        -------\n        `~matplotlib.lines.Line2D`\n\n\n        Examples\n        ---------\n        * draw a thick red vline at *x* = 0 that spans the yrange::\n\n            >>> axvline(linewidth=4, color='r')\n\n        * draw a default vline at *x* = 1 that spans the yrange::\n\n            >>> axvline(x=1)\n\n        * draw a default vline at *x* = .5 that spans the the middle half of\n          the yrange::\n\n            >>> axvline(x=.5, ymin=0.25, ymax=0.75)\n\n        Valid kwargs are :class:`~matplotlib.lines.Line2D` properties,\n        with the exception of 'transform':\n\n        %(Line2D)s\n\n        See also\n        --------\n\n        `axhspan` for example plot and source code\n        \"\"\"\n\n        if \"transform\" in kwargs:\n            raise ValueError(\n                \"'transform' is not allowed as a kwarg;\"\n                + \"axvline generates its own transform.\")\n        xmin, xmax = self.get_xbound()\n\n        # We need to strip away the units for comparison with\n        # non-unitized bounds\n        self._process_unit_info(xdata=x, kwargs=kwargs)\n        xx = self.convert_xunits(x)\n        scalex = (xx < xmin) or (xx > xmax)\n\n        trans = mtransforms.blended_transform_factory(\n            self.transData, self.transAxes)\n        l = mlines.Line2D([x, x], [ymin, ymax], transform=trans, **kwargs)\n        self.add_line(l)\n        self.autoscale_view(scalex=scalex, scaley=False)\n        return l"
        }
      ]
    }
  ]
}