{
  "Selected_candidate": {
    "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)"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant because it involves a bug in a function of a similar component (RangeSlider) of the Matplotlib library. The issue stems from how values are set within the widget, specifically related to indices used in an array (similar to the IndexError in the CURRENT bug report). The structural similarity and patch relevance can provide insights into debugging the current issue with initializing values for the RangeSlider. Examining the fix for the improper handling of values will aid in determining how to correct the error stemming from index access in the CURRENT bug report.",
  "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"
}