{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate C deals with issues related to the `Arc` patch, which is directly tied to the functionality of the `LogNorm` norm in the CURRENT bug. Both bug reports focus on the rendering of portions of arcs, with Candidate C addressing the accuracy of angles in drawing arcs. As the CURRENT bug involves an `Invalid vmin` error triggered during plotting due to changes related to norm settings, the knowledge from debugging angle problems can provide insights into handling cases where valid parameters must be enforced, similar to how valid theta values are examined in arcs. The structural similarity in handling matplotlib patches also relates the two cases, making it highly relevant for debugging the CURRENT issue.",
  "instance_id": "matplotlib__matplotlib-25079",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-01-25T15:24:44Z",
  "problem_statement": "[Bug]: Setting norm with existing colorbar fails with 3.6.3\n### Bug summary\r\n\r\nSetting the norm to a `LogNorm` after the colorbar has been created (e.g. in interactive code) fails with an `Invalid vmin` value in matplotlib 3.6.3.\r\n\r\nThe same code worked in previous matplotlib versions.\r\n\r\nNot that vmin and vmax are explicitly set to values valid for `LogNorm` and no negative values (or values == 0) exist in the input data.\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nfrom matplotlib.colors import LogNorm\r\nimport numpy as np\r\n\r\n# create some random data to fill a 2d plot\r\nrng = np.random.default_rng(0)\r\nimg = rng.uniform(1, 5, (25, 25))\r\n\r\n# plot it\r\nfig, ax = plt.subplots(layout=\"constrained\")\r\nplot = ax.pcolormesh(img)\r\ncbar = fig.colorbar(plot, ax=ax)\r\n\r\nvmin = 1\r\nvmax = 5\r\n\r\nplt.ion()\r\nfig.show()\r\nplt.pause(0.5)\r\n\r\nplot.norm = LogNorm(vmin, vmax)\r\nplot.autoscale()\r\nplt.pause(0.5)\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n```\r\nTraceback (most recent call last):\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/backends/backend_qt.py\", line 454, in _draw_idle\r\n    self.draw()\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/backends/backend_agg.py\", line 405, in draw\r\n    self.figure.draw(self.renderer)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/artist.py\", line 74, in draw_wrapper\r\n    result = draw(artist, renderer, *args, **kwargs)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/figure.py\", line 3082, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/image.py\", line 131, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/axes/_base.py\", line 3100, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/image.py\", line 131, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/collections.py\", line 2148, in draw\r\n    self.update_scalarmappable()\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/collections.py\", line 891, in update_scalarmappable\r\n    self._mapped_colors = self.to_rgba(self._A, self._alpha)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/cm.py\", line 511, in to_rgba\r\n    x = self.norm(x)\r\n  File \"/home/mnoethe/.local/conda/envs/cta-dev/lib/python3.9/site-packages/matplotlib/colors.py\", line 1694, in __call__\r\n    raise ValueError(\"Invalid vmin or vmax\")\r\nValueError: Invalid vmin or vmax\r\n```\r\n\r\n### Expected outcome\r\n\r\nWorks, colorbar and mappable are updated with new norm.\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\nLinux\r\n\r\n### Matplotlib Version\r\n\r\n3.6.3 (works with 3.6.2)\r\n\r\n### Matplotlib Backend\r\n\r\nMultpiple backends tested, same error in all (Qt5Agg, TkAgg, agg, ...)\r\n\r\n### Python version\r\n\r\n3.9.15\r\n\r\n### Jupyter version\r\n\r\nnot in jupyter\r\n\r\n### Installation\r\n\r\nconda\n",
  "patch": "diff --git a/lib/matplotlib/colors.py b/lib/matplotlib/colors.py\n--- a/lib/matplotlib/colors.py\n+++ b/lib/matplotlib/colors.py\n@@ -1362,8 +1362,12 @@ def inverse(self, value):\n \n     def autoscale(self, A):\n         \"\"\"Set *vmin*, *vmax* to min, max of *A*.\"\"\"\n-        self.vmin = self.vmax = None\n-        self.autoscale_None(A)\n+        with self.callbacks.blocked():\n+            # Pause callbacks while we are updating so we only get\n+            # a single update signal at the end\n+            self.vmin = self.vmax = None\n+            self.autoscale_None(A)\n+        self._changed()\n \n     def autoscale_None(self, A):\n         \"\"\"If vmin or vmax are not set, use the min/max of *A* to set them.\"\"\"\n"
}