{
  "original_problem": {
    "instance_id": "matplotlib__matplotlib-26011",
    "repo": "matplotlib/matplotlib",
    "created_at": "2023-05-30T13:45:49Z",
    "problem_statement": "xlim_changed not emitted on shared axis\n<!--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\nWhen an axis is shared with another its registered \"xlim_changed\" callbacks does not get called when the change is induced by a shared axis (via sharex=). \r\n\r\nIn _base.py the set_xlim for sibling axis are called with emit=False:\r\n\r\n```\r\nmatplotlib/lib/matplotlib/axes/_base.py:\r\n\r\n/.../\r\ndef set_xlim(...)\r\n/.../\r\n        if emit:\r\n            self.callbacks.process('xlim_changed', self)\r\n            # Call all of the other x-axes that are shared with this one\r\n            for other in self._shared_x_axes.get_siblings(self):\r\n                if other is not self:\r\n                    other.set_xlim(self.viewLim.intervalx,\r\n                                   emit=False, auto=auto)\r\n```\r\n\r\nI'm very new to matplotlib, so perhaps there is a good reason for this? emit=False seems to disable both continued \"inheritance\" of axis (why?) and triggering of change callbacks (looking at the code above).\r\n\r\nIt seems like one would at least want to trigger the xlim_changed callbacks as they would be intended to react to any change in axis limits.\r\n\r\nEdit: Setting emit=True seems to introduce a recursion issue (not sure why but as inheritance seems to be passed along anyway it doesn't really matter). Moving the callback call to outside of the \"if emit:\"-statement seems to solve the issue as far as I can see when trying it out. Any reason to keep it inside the if-statement? \r\n\n",
    "patch": "diff --git a/lib/matplotlib/axis.py b/lib/matplotlib/axis.py\n--- a/lib/matplotlib/axis.py\n+++ b/lib/matplotlib/axis.py\n@@ -1241,11 +1241,13 @@ def _set_lim(self, v0, v1, *, emit=True, auto):\n             self.axes.callbacks.process(f\"{name}lim_changed\", self.axes)\n             # Call all of the other axes that are shared with this one\n             for other in self._get_shared_axes():\n-                if other is not self.axes:\n-                    other._axis_map[name]._set_lim(\n-                        v0, v1, emit=False, auto=auto)\n-                    if other.figure != self.figure:\n-                        other.figure.canvas.draw_idle()\n+                if other is self.axes:\n+                    continue\n+                other._axis_map[name]._set_lim(v0, v1, emit=False, auto=auto)\n+                if emit:\n+                    other.callbacks.process(f\"{name}lim_changed\", other)\n+                if other.figure != self.figure:\n+                    other.figure.canvas.draw_idle()\n \n         self.stale = True\n         return v0, v1\n"
  },
  "candidates_evaluated": 5,
  "judgment_result": {
    "candidates": [
      {
        "idx": 1,
        "id": "similar_10140",
        "decision": "Not useful",
        "confidence": "Medium",
        "reason": "The issue is about CPU usage and responsiveness, unrelated to callback handling or shared axis logic."
      },
      {
        "idx": 2,
        "id": "similar_25345",
        "decision": "Useful",
        "confidence": "High",
        "reason": "Both issues involve callback handling and the need to ensure proper execution without unintended side effects."
      },
      {
        "idx": 3,
        "id": "similar_13676",
        "decision": "Not useful",
        "confidence": "Medium",
        "reason": "The issue is about exception handling in animations, not related to callback or shared axis logic."
      },
      {
        "idx": 4,
        "id": "similar_6142",
        "decision": "Not useful",
        "confidence": "Medium",
        "reason": "The issue is about tick calculation, unrelated to callback handling or shared axis logic."
      },
      {
        "idx": 5,
        "id": "similar_19989",
        "decision": "Not useful",
        "confidence": "Medium",
        "reason": "The issue involves axis removal and rendering, not directly related to callback handling or shared axis logic."
      }
    ]
  },
  "raw_summaries": [
    {
      "similar_issue": {
        "issue_title": "Qt5Agg eats 100% CPU when plotting with block=True in interactive mode",
        "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\nWhen using Qt5Agg, Matplotlib consumes a full CPU core while an interactive window is open.  The cursor repeatedly shows a \"spinner\" when I mouse over the canvas.  This didn't used to be the case.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nplt.plot([1, 2], [3, 4])\r\nplt.ion()\r\nplt.show(block=True)\r\n```\r\n\r\n**Actual outcome**\r\n\r\nOne of the CPU cores is pegged at 100%.  Interface is very slow and barely responsive.\r\n\r\n**Expected outcome**\r\n\r\nShould not use up 100% of a CPU core when nothing is happening.\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating system: Linux 4.14.8\r\n  * Matplotlib version: 2.1.1+864.g92a93d698.dirty\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: 3.6.4\r\n\r\nThis was reproduced on both XMonad and XFCE4.  Affects both `python-matplotlib` (2.1.1) from official Arch repos and `python-matplotlib-git` from AUR.",
        "issue_id": 10140,
        "pr_number": 10142,
        "pr_title": "FIX: prevent recursive draws with plt.ion not in IPython with qt5agg",
        "pr_body": "## PR Summary\r\n\r\nIn 9b8a944db954c8240e14413e517e1b781881b80b the latch logic was moved\r\nabove the call to `self.draw` to de-latch the call to\r\n`__draw_idle_agg` in `paintEvent` to protect against `blit` calls\r\nduring a draw.\r\n\r\nHowever, this now fails to properly latch the base draw so with\r\n`plt.ion` (not in IPython) the stale callback will cause a (possibly\r\ninfinite) recursion in the draw due to stale being set as part of the\r\ndraw call (which will re-trigger the draw).\r\n\r\nThis adds a second flag to track if we are currently rendering and\r\nbail.\r\n\r\ncloses #10140\r\n\r\n\r\nI suspect a better fix is to re-think how the stale callbacks propagate or how the non-ipython stale callback works, but I think this is the minimal change which fixes a critical issue.\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_closed_at": "2018-01-04T20:28:55Z",
        "base_commit": "008da385fe73614059430073a722d674ade55014"
      },
      "summary": "### Summary:\nThis issue is related to excessive CPU usage when using the Matplotlib library with the Qt5Agg backend in interactive mode. The problem manifests when a plot is displayed with the `block=True` parameter, leading to one CPU core being fully utilized while the interactive window remains open. Users experience a persistent \"spinner\" cursor over the plot canvas, indicating an unresponsive interface. This behavior was not present in earlier versions of the software. The problem affects systems running on Linux, specifically with Matplotlib version 2.1.1, and has been observed in different desktop environments such as XMonad and XFCE4. The issue has the potential to significantly degrade system performance and user experience, as it renders the application sluggish and unresponsive. The technical resolution involved modifying several functions within the `backend_qt5agg.py` file, particularly within the `FigureCanvasQTAggBase` class, to address the excessive CPU consumption and improve the responsiveness of the plotting interface.",
      "prompt_used": "You are an expert in software issue reasoning analysis.\nGiven the following problem report and its fixed code elements, generate a comprehensive summary based on the entire document. Your goal is to abstract the information in the problem description into a more general description.\n\n## Original Issue Report:\nTitle: Qt5Agg eats 100% CPU when plotting with block=True in interactive mode\n\nBody:\n### Bug report\r\n\r\n**Bug summary**\r\n\r\nWhen using Qt5Agg, Matplotlib consumes a full CPU core while an interactive window is open.  The cursor repeatedly shows a \"spinner\" when I mouse over the canvas.  This didn't used to be the case.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nplt.plot([1, 2], [3, 4])\r\nplt.ion()\r\nplt.show(block=True)\r\n```\r\n\r\n**Actual outcome**\r\n\r\nOne of the CPU cores is pegged at 100%.  Interface is very slow and barely responsive.\r\n\r\n**Expected outcome**\r\n\r\nShould not use up 100% of a CPU core when nothing is happening.\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating system: Linux 4.14.8\r\n  * Matplotlib version: 2.1.1+864.g92a93d698.dirty\r\n  * Matplotlib backend: Qt5Agg\r\n  * Python version: 3.6.4\r\n\r\nThis was reproduced on both XMonad and XFCE4.  Affects both `python-matplotlib` (2.1.1) from official Arch repos and `python-matplotlib-git` from AUR.\n\n## Code elements fixed by the patch:\n{FIXED_CODE_ELEMENTS}\n\nPlease analyze the above issue report and provide a structured summary that includes:\n1. Problem description in general terms\n2. Key symptoms and behaviors observed\n3. Affected components or systems\n4. Potential impact or severity\n5. Any relevant technical details abstracted for broader understanding\n\nPlease return the summary with “### Summary:\", For example:\n### Summary: This issue is ...\n\nChanges Summary:\nlib/matplotlib/backends/backend_qt5agg.py\n  function: FigureCanvasQTAggBase.__init__\n  function: FigureCanvasQTAggBase.draw\n  function: FigureCanvasQTAggBase.draw_idle\n  function: FigureCanvasQTAggBase.__draw_idle_agg\n"
    },
    {
      "similar_issue": {
        "issue_title": "[Bug]: using clf and pyplot.draw in range slider on_changed callback blocks input to widgets",
        "issue_body": "### Bug summary\n\nWhen using clear figure, adding new widgets and then redrawing the current figure in the on_changed callback of a range slider the inputs to all the widgets in the figure are blocked. When doing the same in the button callback on_clicked, everything works fine.\n\n### Code for reproduction\n\n```python\nimport matplotlib.pyplot as pyplot\r\nimport matplotlib.widgets as widgets\r\n\r\ndef onchanged(values):\r\n    print(\"on changed\")\r\n    print(values)\r\n    pyplot.clf()\r\n    addElements()\r\n    pyplot.draw()\r\n\r\ndef onclick(e):\r\n    print(\"on click\")\r\n    pyplot.clf()\r\n    addElements()\r\n    pyplot.draw()\r\n\r\ndef addElements():\r\n    ax = pyplot.axes([0.1, 0.45, 0.8, 0.1])\r\n    global slider\r\n    slider = widgets.RangeSlider(ax, \"Test\", valmin=1, valmax=10, valinit=(1, 10))\r\n    slider.on_changed(onchanged)\r\n    ax = pyplot.axes([0.1, 0.30, 0.8, 0.1])\r\n    global button\r\n    button = widgets.Button(ax, \"Test\")\r\n    button.on_clicked(onclick)\r\n\r\naddElements()\r\n\r\npyplot.show()\n```\n\n\n### Actual outcome\n\nThe widgets can't receive any input from a mouse click, when redrawing in the on_changed callback of a range Slider. \r\nWhen using a button, there is no problem.\n\n### Expected outcome\n\nThe range slider callback on_changed behaves the same as the button callback on_clicked.\n\n### Additional information\n\nThe problem also occurred on Manjaro with:\r\n- Python version: 3.10.9\r\n- Matplotlib version: 3.6.2\r\n- Matplotlib backend: QtAgg\r\n- Installation of matplotlib via Linux package manager\r\n\n\n### Operating system\n\nWindows 10\n\n### Matplotlib Version\n\n3.6.2\n\n### Matplotlib Backend\n\nTkAgg\n\n### Python version\n\n3.11.0\n\n### Jupyter version\n\n_No response_\n\n### Installation\n\npip",
        "issue_id": 25345,
        "pr_number": 25433,
        "pr_title": "Release mouse grabs when owning Axes is removed",
        "pr_body": "## PR Summary\r\n\r\nOtherwise, _nothing_ will remove the grab, and no other Axes can grab the mouse (either raising the `RuntimeError` exception, or in the case of widgets, ignoring mouse events altogether.)\r\n\r\nFixes #25345\r\n\r\n## PR Checklist\r\n\r\n**Documentation and Tests**\r\n- [x] Has pytest style unit tests (and `pytest` passes)\r\n- [n/a] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [n/a] New plotting related features are documented with examples.\r\n\r\n**Release Notes**\r\n- [n/a] New features are marked with a `.. versionadded::` directive in the docstring and documented in `doc/users/next_whats_new/`\r\n- [n/a] API changes are marked with a `.. versionchanged::` directive in the docstring and documented in `doc/api/next_api_changes/`\r\n- [n/a] Release notes conform with instructions in  `next_whats_new/README.rst` or `next_api_changes/README.rst`",
        "issue_closed_at": "2023-03-14T06:22:33Z",
        "base_commit": "7eafdd8af3c523c1c77b027d378fb337dd489f18"
      },
      "summary": "### Summary:\n\nThis issue is related to a bug in the Matplotlib library, specifically when interacting with graphical widgets such as sliders and buttons within a plotted figure. The problem arises when the `on_changed` callback function of a range slider is used to clear and redraw the current figure. This action results in all widgets within the figure becoming unresponsive to input, effectively blocking user interaction. However, performing the same clear and redraw operation within a button's `on_clicked` callback does not exhibit this blocking behavior.\n\nKey symptoms include the inability of the widgets to receive input from mouse clicks after the figure is redrawn in the slider's callback. The expected behavior is for both the slider's and button's callbacks to handle figure redrawing without affecting widget interactivity.\n\nThe affected components are the graphical widgets managed by Matplotlib, particularly when using the TkAgg backend on Windows 10 and the QtAgg backend on Manjaro Linux. The problem seems to be related to how callbacks interact with figure rendering and widget states.\n\nThe potential impact includes a disruption in user experience, as widgets become unresponsive, hindering interactive data visualization tasks. This issue could affect applications or scripts that rely on dynamic figure updates triggered by slider movements or other widget interactions.\n\nRelevant technical details include the use of Python 3.10.9 and 3.11.0, and Matplotlib version 3.6.2. The issue manifests across different operating systems and backends, indicating a potential underlying problem in the Matplotlib library's handling of widget states during figure updates.",
      "prompt_used": "You are an expert in software issue reasoning analysis.\nGiven the following problem report and its fixed code elements, generate a comprehensive summary based on the entire document. Your goal is to abstract the information in the problem description into a more general description.\n\n## Original Issue Report:\nTitle: [Bug]: using clf and pyplot.draw in range slider on_changed callback blocks input to widgets\n\nBody:\n### Bug summary\n\nWhen using clear figure, adding new widgets and then redrawing the current figure in the on_changed callback of a range slider the inputs to all the widgets in the figure are blocked. When doing the same in the button callback on_clicked, everything works fine.\n\n### Code for reproduction\n\n```python\nimport matplotlib.pyplot as pyplot\r\nimport matplotlib.widgets as widgets\r\n\r\ndef onchanged(values):\r\n    print(\"on changed\")\r\n    print(values)\r\n    pyplot.clf()\r\n    addElements()\r\n    pyplot.draw()\r\n\r\ndef onclick(e):\r\n    print(\"on click\")\r\n    pyplot.clf()\r\n    addElements()\r\n    pyplot.draw()\r\n\r\ndef addElements():\r\n    ax = pyplot.axes([0.1, 0.45, 0.8, 0.1])\r\n    global slider\r\n    slider = widgets.RangeSlider(ax, \"Test\", valmin=1, valmax=10, valinit=(1, 10))\r\n    slider.on_changed(onchanged)\r\n    ax = pyplot.axes([0.1, 0.30, 0.8, 0.1])\r\n    global button\r\n    button = widgets.Button(ax, \"Test\")\r\n    button.on_clicked(onclick)\r\n\r\naddElements()\r\n\r\npyplot.show()\n```\n\n\n### Actual outcome\n\nThe widgets can't receive any input from a mouse click, when redrawing in the on_changed callback of a range Slider. \r\nWhen using a button, there is no problem.\n\n### Expected outcome\n\nThe range slider callback on_changed behaves the same as the button callback on_clicked.\n\n### Additional information\n\nThe problem also occurred on Manjaro with:\r\n- Python version: 3.10.9\r\n- Matplotlib version: 3.6.2\r\n- Matplotlib backend: QtAgg\r\n- Installation of matplotlib via Linux package manager\r\n\n\n### Operating system\n\nWindows 10\n\n### Matplotlib Version\n\n3.6.2\n\n### Matplotlib Backend\n\nTkAgg\n\n### Python version\n\n3.11.0\n\n### Jupyter version\n\n_No response_\n\n### Installation\n\npip\n\n## Code elements fixed by the patch:\n{FIXED_CODE_ELEMENTS}\n\nPlease analyze the above issue report and provide a structured summary that includes:\n1. Problem description in general terms\n2. Key symptoms and behaviors observed\n3. Affected components or systems\n4. Potential impact or severity\n5. Any relevant technical details abstracted for broader understanding\n\nPlease return the summary with “### Summary:\", For example:\n### Summary: This issue is ...\n\nChanges Summary:\nlib/matplotlib/figure.py\n  function: FigureBase._break_share_link\n"
    },
    {
      "similar_issue": {
        "issue_title": "FuncAnimation with generator causes crash on StopIteration",
        "issue_body": "### Bug report\r\n\r\nUsing FuncAnimation with a generator causes a crash when the generator raises a StopIteration.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nfrom matplotlib.animation import FuncAnimation\r\n\r\nfig, ax = plt.subplots()\r\nplt.plot([1,2,3],[2,4,3])\r\ndef update(frame):\r\n\tprint(frame)\r\n\treturn []\r\nanimation = FuncAnimation(fig, update, frames=iter(range(10)), blit=True, interval=100)\r\nplt.show()\r\n```\r\n\r\nOutput:\r\n\r\n```\r\n$ python animation_crash.py \r\n0\r\n1\r\n2\r\n3\r\n4\r\n5\r\n6\r\n7\r\n8\r\n9\r\nTraceback (most recent call last):\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/backend_bases.py\", line 1238, in _on_timer\r\n    ret = func(*args, **kwargs)\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/animation.py\", line 1462, in _step\r\n    self._init_draw()\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/animation.py\", line 1740, in _init_draw\r\n    self._draw_frame(next(self.new_frame_seq()))\r\nStopIteration\r\nAbgebrochen (Speicherabzug geschrieben)\r\n```\r\n\r\n**Matplotlib version**\r\nOperating System: Arch linux, 64-bit\r\nMatplotlib version: 3.0.3-1 (using pacman)\r\nMatplotlib backend: Qt5Agg\r\n\r\nCoredump:\r\n```\r\n[Current thread is 1 (Thread 0x7f2d24492600 (LWP 25813))]\r\n(gdb) info stack\r\n#0  0x00007f2d249d0d7f in raise () at /usr/lib/libc.so.6\r\n#1  0x00007f2d249bb672 in abort () at /usr/lib/libc.so.6\r\n#2  0x00007f2d1f79a7fc in  () at /usr/lib/libQt5Core.so.5\r\n#3  0x00007f2d1fd2b048 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#4  0x00007f2d1fe8b216 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#5  0x00007f2d1fe8bcd0 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#6  0x00007f2d1f9bb94d in QMetaObject::activate(QObject*, int, int, void**) () at /usr/lib/libQt5Core.so.5\r\n#7  0x00007f2d1f9c7ab8 in QTimer::timeout(QTimer::QPrivateSignal) () at /usr/lib/libQt5Core.so.5\r\n#8  0x00007f2d1fe34c9c in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#9  0x00007f2d1f9bc10b in QObject::event(QEvent*) () at /usr/lib/libQt5Core.so.5\r\n#10 0x00007f2d1fe34b4c in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#11 0x00007f2d1ba68e24 in QApplicationPrivate::notify_helper(QObject*, QEvent*) () at /usr/lib/libQt5Widgets.so.5\r\n#12 0x00007f2d1ba706e1 in QApplication::notify(QObject*, QEvent*) () at /usr/lib/libQt5Widgets.so.5\r\n#13 0x00007f2d1c114b3f in  () at /usr/lib/python3.7/site-packages/PyQt5/QtWidgets.so\r\n#14 0x00007f2d1f990e99 in QCoreApplication::notifyInternal2(QObject*, QEvent*) () at /usr/lib/libQt5Core.so.5\r\n#15 0x00007f2d1f9e6095 in QTimerInfoList::activateTimers() () at /usr/lib/libQt5Core.so.5\r\n#16 0x00007f2d1f9e6922 in  () at /usr/lib/libQt5Core.so.5\r\n#17 0x00007f2d20dd3a2f in g_main_context_dispatch () at /usr/lib/libglib-2.0.so.0\r\n#18 0x00007f2d20dd55e9 in  () at /usr/lib/libglib-2.0.so.0\r\n#19 0x00007f2d20dd562e in g_main_context_iteration () at /usr/lib/libglib-2.0.so.0\r\n#20 0x00007f2d1f9e6ce9 in QEventDispatcherGlib::processEvents(QFlags<QEventLoop::ProcessEventsFlag>) ()\r\n    at /usr/lib/libQt5Core.so.5\r\n#21 0x00007f2d1f98fb2c in QEventLoop::exec(QFlags<QEventLoop::ProcessEventsFlag>) () at /usr/lib/libQt5Core.so.5\r\n#22 0x00007f2d1f997e36 in QCoreApplication::exec() () at /usr/lib/libQt5Core.so.5\r\n#23 0x00007f2d1c1135cd in  () at /usr/lib/python3.7/site-packages/PyQt5/QtWidgets.so\r\n#24 0x00007f2d24775a78 in _PyMethodDef_RawFastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#25 0x00007f2d24775d11 in _PyCFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#26 0x00007f2d247ec8eb in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#27 0x00007f2d247752eb in _PyFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#28 0x00007f2d247ec2b3 in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n--Type <RET> for more, q to quit, c to continue without paging--\r\n#29 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#30 0x00007f2d2472fdec in _PyFunction_FastCallDict () at /usr/lib/libpython3.7m.so.1.0\r\n#31 0x00007f2d2473f908 in _PyObject_Call_Prepend () at /usr/lib/libpython3.7m.so.1.0\r\n#32 0x00007f2d2473033b in PyObject_Call () at /usr/lib/libpython3.7m.so.1.0\r\n#33 0x00007f2d247e943c in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#34 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#35 0x00007f2d24775492 in _PyFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#36 0x00007f2d247ec2b3 in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#37 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#38 0x00007f2d2472fab4 in PyEval_EvalCodeEx () at /usr/lib/libpython3.7m.so.1.0\r\n#39 0x00007f2d2472fadc in PyEval_EvalCode () at /usr/lib/libpython3.7m.so.1.0\r\n#40 0x00007f2d24859c94 in  () at /usr/lib/libpython3.7m.so.1.0\r\n#41 0x00007f2d2485b8be in PyRun_FileExFlags () at /usr/lib/libpython3.7m.so.1.0\r\n#42 0x00007f2d2485cc75 in PyRun_SimpleFileExFlags () at /usr/lib/libpython3.7m.so.1.0\r\n#43 0x00007f2d2485eeb7 in  () at /usr/lib/libpython3.7m.so.1.0\r\n#44 0x00007f2d2485f0fc in _Py_UnixMain () at /usr/lib/libpython3.7m.so.1.0\r\n#45 0x00007f2d249bd223 in __libc_start_main () at /usr/lib/libc.so.6\r\n#46 0x0000561e1056f05e in _start ()\r\n```",
        "issue_id": 13676,
        "pr_number": 14068,
        "pr_title": "Alternative fix for passing iterator as frames to FuncAnimation",
        "pr_body": "## PR Summary\r\n\r\nFixes #13676. Replaces #13679.\r\n\r\nThis was proposed in https://github.com/matplotlib/matplotlib/pull/13679#issuecomment-478905310 it's clearly the better approach as it's simpler and can handle generators. Thanks @anntzer.\r\n\r\nI do not really know how to add a test for this. Can't have an infinite loop in a unittest :smile:. Works fine as a standalone test with the following code\r\n\r\n~~~\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nfrom matplotlib.animation import FuncAnimation\r\n\r\nfig, ax = plt.subplots()\r\ns = ax.set_title('0')\r\nplt.plot([1,2,3],[2,4,3])\r\ndef update(frame):\r\n    print(frame)\r\n    s.set_text(str(frame))\r\n    return []\r\nanimation = FuncAnimation(fig, update, frames=(i for i in range(5)), blit=True, interval=100)\r\nplt.show()\r\n~~~\r\n",
        "issue_closed_at": "2019-05-22T21:21:43Z",
        "base_commit": "2753cf5679cbb943e1219433bbaf784556b8bda3"
      },
      "summary": "### Summary:\n\nThis issue is centered around a crash that occurs in the Matplotlib library when using the `FuncAnimation` class in conjunction with a generator that raises a `StopIteration` exception. The problem manifests when the generator, used to provide frames for the animation, exhausts its sequence of values and raises the `StopIteration` exception, which is not adequately handled by the `FuncAnimation` class, leading to a program crash.\n\n1. **Problem description in general terms**: The problem occurs when a generator used in an animation sequence finishes its iteration and raises a `StopIteration` exception. The animation framework fails to handle this exception gracefully, resulting in a crash.\n\n2. **Key symptoms and behaviors observed**: The primary symptom is a crash of the program when the generator used in `FuncAnimation` completes its iteration. The output includes a traceback ending with a `StopIteration` exception, indicating that the end of the generator's sequence was reached without proper handling.\n\n3. **Affected components or systems**: The issue affects the `FuncAnimation` class within the Matplotlib library, specifically impacting users who employ generators as a source of frames for animations.\n\n4. **Potential impact or severity**: The severity of this issue can be considered moderate, as it causes the program to crash, thereby disrupting the execution of animations. This is particularly relevant for applications relying on dynamic data visualization using Matplotlib's animation capabilities.\n\n5. **Relevant technical details abstracted for broader understanding**: The crash is due to the absence of exception handling for `StopIteration` within the `FuncAnimation` class's implementation. The `StopIteration` exception needs to be caught and managed to allow the animation to terminate gracefully instead of causing the application to abort unexpectedly. The fix involves modifying the `FuncAnimation.__init__` function within the `lib/matplotlib/animation.py` file to handle the exception properly.",
      "prompt_used": "You are an expert in software issue reasoning analysis.\nGiven the following problem report and its fixed code elements, generate a comprehensive summary based on the entire document. Your goal is to abstract the information in the problem description into a more general description.\n\n## Original Issue Report:\nTitle: FuncAnimation with generator causes crash on StopIteration\n\nBody:\n### Bug report\r\n\r\nUsing FuncAnimation with a generator causes a crash when the generator raises a StopIteration.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\nfrom matplotlib.animation import FuncAnimation\r\n\r\nfig, ax = plt.subplots()\r\nplt.plot([1,2,3],[2,4,3])\r\ndef update(frame):\r\n\tprint(frame)\r\n\treturn []\r\nanimation = FuncAnimation(fig, update, frames=iter(range(10)), blit=True, interval=100)\r\nplt.show()\r\n```\r\n\r\nOutput:\r\n\r\n```\r\n$ python animation_crash.py \r\n0\r\n1\r\n2\r\n3\r\n4\r\n5\r\n6\r\n7\r\n8\r\n9\r\nTraceback (most recent call last):\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/backend_bases.py\", line 1238, in _on_timer\r\n    ret = func(*args, **kwargs)\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/animation.py\", line 1462, in _step\r\n    self._init_draw()\r\n  File \"/usr/lib/python3.7/site-packages/matplotlib/animation.py\", line 1740, in _init_draw\r\n    self._draw_frame(next(self.new_frame_seq()))\r\nStopIteration\r\nAbgebrochen (Speicherabzug geschrieben)\r\n```\r\n\r\n**Matplotlib version**\r\nOperating System: Arch linux, 64-bit\r\nMatplotlib version: 3.0.3-1 (using pacman)\r\nMatplotlib backend: Qt5Agg\r\n\r\nCoredump:\r\n```\r\n[Current thread is 1 (Thread 0x7f2d24492600 (LWP 25813))]\r\n(gdb) info stack\r\n#0  0x00007f2d249d0d7f in raise () at /usr/lib/libc.so.6\r\n#1  0x00007f2d249bb672 in abort () at /usr/lib/libc.so.6\r\n#2  0x00007f2d1f79a7fc in  () at /usr/lib/libQt5Core.so.5\r\n#3  0x00007f2d1fd2b048 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#4  0x00007f2d1fe8b216 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#5  0x00007f2d1fe8bcd0 in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#6  0x00007f2d1f9bb94d in QMetaObject::activate(QObject*, int, int, void**) () at /usr/lib/libQt5Core.so.5\r\n#7  0x00007f2d1f9c7ab8 in QTimer::timeout(QTimer::QPrivateSignal) () at /usr/lib/libQt5Core.so.5\r\n#8  0x00007f2d1fe34c9c in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#9  0x00007f2d1f9bc10b in QObject::event(QEvent*) () at /usr/lib/libQt5Core.so.5\r\n#10 0x00007f2d1fe34b4c in  () at /usr/lib/python3.7/site-packages/PyQt5/QtCore.so\r\n#11 0x00007f2d1ba68e24 in QApplicationPrivate::notify_helper(QObject*, QEvent*) () at /usr/lib/libQt5Widgets.so.5\r\n#12 0x00007f2d1ba706e1 in QApplication::notify(QObject*, QEvent*) () at /usr/lib/libQt5Widgets.so.5\r\n#13 0x00007f2d1c114b3f in  () at /usr/lib/python3.7/site-packages/PyQt5/QtWidgets.so\r\n#14 0x00007f2d1f990e99 in QCoreApplication::notifyInternal2(QObject*, QEvent*) () at /usr/lib/libQt5Core.so.5\r\n#15 0x00007f2d1f9e6095 in QTimerInfoList::activateTimers() () at /usr/lib/libQt5Core.so.5\r\n#16 0x00007f2d1f9e6922 in  () at /usr/lib/libQt5Core.so.5\r\n#17 0x00007f2d20dd3a2f in g_main_context_dispatch () at /usr/lib/libglib-2.0.so.0\r\n#18 0x00007f2d20dd55e9 in  () at /usr/lib/libglib-2.0.so.0\r\n#19 0x00007f2d20dd562e in g_main_context_iteration () at /usr/lib/libglib-2.0.so.0\r\n#20 0x00007f2d1f9e6ce9 in QEventDispatcherGlib::processEvents(QFlags<QEventLoop::ProcessEventsFlag>) ()\r\n    at /usr/lib/libQt5Core.so.5\r\n#21 0x00007f2d1f98fb2c in QEventLoop::exec(QFlags<QEventLoop::ProcessEventsFlag>) () at /usr/lib/libQt5Core.so.5\r\n#22 0x00007f2d1f997e36 in QCoreApplication::exec() () at /usr/lib/libQt5Core.so.5\r\n#23 0x00007f2d1c1135cd in  () at /usr/lib/python3.7/site-packages/PyQt5/QtWidgets.so\r\n#24 0x00007f2d24775a78 in _PyMethodDef_RawFastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#25 0x00007f2d24775d11 in _PyCFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#26 0x00007f2d247ec8eb in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#27 0x00007f2d247752eb in _PyFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#28 0x00007f2d247ec2b3 in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n--Type <RET> for more, q to quit, c to continue without paging--\r\n#29 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#30 0x00007f2d2472fdec in _PyFunction_FastCallDict () at /usr/lib/libpython3.7m.so.1.0\r\n#31 0x00007f2d2473f908 in _PyObject_Call_Prepend () at /usr/lib/libpython3.7m.so.1.0\r\n#32 0x00007f2d2473033b in PyObject_Call () at /usr/lib/libpython3.7m.so.1.0\r\n#33 0x00007f2d247e943c in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#34 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#35 0x00007f2d24775492 in _PyFunction_FastCallKeywords () at /usr/lib/libpython3.7m.so.1.0\r\n#36 0x00007f2d247ec2b3 in _PyEval_EvalFrameDefault () at /usr/lib/libpython3.7m.so.1.0\r\n#37 0x00007f2d2472eb99 in _PyEval_EvalCodeWithName () at /usr/lib/libpython3.7m.so.1.0\r\n#38 0x00007f2d2472fab4 in PyEval_EvalCodeEx () at /usr/lib/libpython3.7m.so.1.0\r\n#39 0x00007f2d2472fadc in PyEval_EvalCode () at /usr/lib/libpython3.7m.so.1.0\r\n#40 0x00007f2d24859c94 in  () at /usr/lib/libpython3.7m.so.1.0\r\n#41 0x00007f2d2485b8be in PyRun_FileExFlags () at /usr/lib/libpython3.7m.so.1.0\r\n#42 0x00007f2d2485cc75 in PyRun_SimpleFileExFlags () at /usr/lib/libpython3.7m.so.1.0\r\n#43 0x00007f2d2485eeb7 in  () at /usr/lib/libpython3.7m.so.1.0\r\n#44 0x00007f2d2485f0fc in _Py_UnixMain () at /usr/lib/libpython3.7m.so.1.0\r\n#45 0x00007f2d249bd223 in __libc_start_main () at /usr/lib/libc.so.6\r\n#46 0x0000561e1056f05e in _start ()\r\n```\n\n## Code elements fixed by the patch:\n{FIXED_CODE_ELEMENTS}\n\nPlease analyze the above issue report and provide a structured summary that includes:\n1. Problem description in general terms\n2. Key symptoms and behaviors observed\n3. Affected components or systems\n4. Potential impact or severity\n5. Any relevant technical details abstracted for broader understanding\n\nPlease return the summary with “### Summary:\", For example:\n### Summary: This issue is ...\n\nChanges Summary:\nlib/matplotlib/animation.py\n  function: FuncAnimation.__init__\n"
    },
    {
      "similar_issue": {
        "issue_title": "matplotlib.ticker.LinearLocator view_limits algorithm improvement?",
        "issue_body": "Inspecting the code of matplotlib.ticker.LinearLocator\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1161\n\nyou can see that the view limits are chosen such that the difference between vmin and vmax is an interger multiple of scale, which is itself a power of 10. Therefore, the range  will be nicely divisible when divided by 10 (11 tickmarks).\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1213\n\nTherefore, the view_limits function determines the limits assuming there are 11 ticks. This assumption is implicit in two lines:\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1224\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1227\n\nCode is repeated here:\n\n```\nexponent, remainder = divmod(math.log10(vmax - vmin), 1)\nif remainder < 0.5:\n    exponent -= 1\nscale = 10 ** (-exponent)\nvmin = math.floor(scale * vmin) / scale\nvmax = math.ceil(scale * vmax) / scale\n```\n\nSince we know the number of ticks, from `self.num_ticks`, we can generalize the current algorithm to be better suited for any number of ticks. Suggested generalized algorithm:\n\n```\nexponent, remainder = divmod(math.log10(vmax - vmin), math.log10(self.num_ticks-1))\nif remainder < 0.5:\n    exponent -= 1\nscale = (self.num_ticks-1) ** (-exponent)\nvmin = math.floor(scale * vmin) / scale\nvmax = math.ceil(scale * vmax) / scale\n```\n\nThis generalized expression reduces to the current form when `self.num_ticks==11` (which is the current default). For other cases, here is an example:\nwhen num_ticks = 10, vmin = 20, vmax=90\nCurrent algorithm returns vmin = 20, vmax=90, corresponding to ticks spaced by 7.77778.\n\nThe proposed algorithm returns vmin = 18, vmax = 90, corresponding to ticks spaced by 8.\n\nIs this something worth doing? The patch is trivial -- just changing two lines of code. I can turn this in to a pull request to illustrate if it is helpful.\n",
        "issue_id": 6142,
        "pr_number": 6431,
        "pr_title": "Merge from v2.x",
        "pr_body": "This merge from v2.x into master also required a little editing, and it involves many changes, hence this PR.  I resolved conflicts in .travis.yml and lib/matplotlib/tests/test_colors.py.\n",
        "issue_closed_at": "2016-05-04T00:26:00Z",
        "base_commit": "22a7b955a0b9dc4dea8adf155041498dd355e4df"
      },
      "summary": "### Summary:\n\nThis issue pertains to the algorithm used by the `matplotlib.ticker.LinearLocator` component of the Matplotlib library, which is responsible for determining view limits in plots. The current algorithm assumes a fixed number of tick marks (11) when calculating the view limits, which can lead to suboptimal results when a different number of ticks is desired. The reporter suggests a generalized algorithm that can accommodate any number of tick marks, making the calculation more flexible and potentially improving the spacing of tick marks for different scenarios.\n\nKey symptoms and behaviors observed include the algorithm's limitation to a fixed number of ticks, resulting in tick mark spacing that may not be ideal for all use cases. For example, using 10 tick marks with the current algorithm results in non-uniform spacing, while the proposed method offers more even distribution.\n\nThe affected component is the `LinearLocator` within the `matplotlib.ticker` module. This component plays a crucial role in rendering plots with appropriate axis limits and tick spacing, which is critical for accurate data representation and visualization.\n\nThe potential impact of the issue is moderate, as it affects the aesthetics and readability of plots. However, the proposed solution is a simple change in the algorithm that could enhance the functionality and flexibility of the `LinearLocator` component, making it more adaptable to various plotting needs.\n\nRelevant technical details include the mathematical calculation of scale and view limits, which currently depend on the number of tick marks. The proposal involves modifying the algorithm to use the number of ticks specified by `self.num_ticks`, allowing for more precise control over tick mark spacing. The change is straightforward, involving the alteration of two lines of code.\n\nOverall, this issue highlights the importance of flexible and adaptable algorithms in data visualization libraries, ensuring that they can cater to a wide range of user requirements and preferences.",
      "prompt_used": "You are an expert in software issue reasoning analysis.\nGiven the following problem report and its fixed code elements, generate a comprehensive summary based on the entire document. Your goal is to abstract the information in the problem description into a more general description.\n\n## Original Issue Report:\nTitle: matplotlib.ticker.LinearLocator view_limits algorithm improvement?\n\nBody:\nInspecting the code of matplotlib.ticker.LinearLocator\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1161\n\nyou can see that the view limits are chosen such that the difference between vmin and vmax is an interger multiple of scale, which is itself a power of 10. Therefore, the range  will be nicely divisible when divided by 10 (11 tickmarks).\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1213\n\nTherefore, the view_limits function determines the limits assuming there are 11 ticks. This assumption is implicit in two lines:\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1224\nhttps://github.com/matplotlib/matplotlib/blob/7d1a7c2e4637efba239ad3b984928c0175d45f98/lib/matplotlib/ticker.py#L1227\n\nCode is repeated here:\n\n```\nexponent, remainder = divmod(math.log10(vmax - vmin), 1)\nif remainder < 0.5:\n    exponent -= 1\nscale = 10 ** (-exponent)\nvmin = math.floor(scale * vmin) / scale\nvmax = math.ceil(scale * vmax) / scale\n```\n\nSince we know the number of ticks, from `self.num_ticks`, we can generalize the current algorithm to be better suited for any number of ticks. Suggested generalized algorithm:\n\n```\nexponent, remainder = divmod(math.log10(vmax - vmin), math.log10(self.num_ticks-1))\nif remainder < 0.5:\n    exponent -= 1\nscale = (self.num_ticks-1) ** (-exponent)\nvmin = math.floor(scale * vmin) / scale\nvmax = math.ceil(scale * vmax) / scale\n```\n\nThis generalized expression reduces to the current form when `self.num_ticks==11` (which is the current default). For other cases, here is an example:\nwhen num_ticks = 10, vmin = 20, vmax=90\nCurrent algorithm returns vmin = 20, vmax=90, corresponding to ticks spaced by 7.77778.\n\nThe proposed algorithm returns vmin = 18, vmax = 90, corresponding to ticks spaced by 8.\n\nIs this something worth doing? The patch is trivial -- just changing two lines of code. I can turn this in to a pull request to illustrate if it is helpful.\n\n\n## Code elements fixed by the patch:\n{FIXED_CODE_ELEMENTS}\n\nPlease analyze the above issue report and provide a structured summary that includes:\n1. Problem description in general terms\n2. Key symptoms and behaviors observed\n3. Affected components or systems\n4. Potential impact or severity\n5. Any relevant technical details abstracted for broader understanding\n\nPlease return the summary with “### Summary:\", For example:\n### Summary: This issue is ...\n\nChanges Summary:\nlib/matplotlib/animation.py\n  line: line 36\n  class: MovieWriter\n  function: FuncAnimation.__init__\n  function: ImageMagickBase.isAvailable\n  class: FFMpegBase\n  function: ImageMagickBase.output_args\n  function: Animation.save\n  function: Animation.save\n  function: Animation.save\n  function: Animation.save\n  function: Animation.to_html5_video\n\nlib/matplotlib/axes/_base.py\n  function: _AxesBase.__init__\n\nlib/matplotlib/backends/backend_qt5.py\n  function: SubplotToolQt.__init__\n  function: SubplotToolQt.funcleft\n  function: SubplotToolQt.funcright\n  function: SubplotToolQt.funcbottom\n  function: SubplotToolQt.functop\n\nlib/matplotlib/cbook.py\n  function: issubclass_safe\n\nlib/matplotlib/contour.py\n  function: ContourSet.changed\n  function: ContourSet._contour_level_args\n  function: QuadContourSet._contour_args\n\nlib/matplotlib/dates.py\n  class: AutoDateFormatter\n  function: MicrosecondLocator.__init__\n\nlib/matplotlib/image.py\n  function: PcolorImage.set_data\n  function: PcolorImage.set_data\n  function: BboxImage.__init__\n  function: PcolorImage._check_unsampled_image\n  function: BboxImage.make_image\n\nlib/matplotlib/rcsetup.py\n  function: validate_animation_writer_path\n\nlib/matplotlib/ticker.py\n  function: OldAutoLocator.__init__\n  function: OldAutoLocator.tick_values\n\nsetupext.py\n  function: PdfToPs.check\n  function: PdfToPs.check\n"
    },
    {
      "similar_issue": {
        "issue_title": "Removal of y-shared polar axes causes crash at draw time",
        "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\nAll's in the title.\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug.\r\nPlease make sure to minimize the number of dependencies required, and provide\r\nany necessary plotted data.\r\nAvoid using threads, as Matplotlib is (explicitly) not thread-safe.-->\r\n\r\n```python\r\nfrom pylab import *\r\nax1, ax2 = gcf().subplots(2, sharey=True, subplot_kw={\"projection\": \"polar\"}); ax2.remove(); 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```\r\nTraceback (most recent call last):\r\nTraceback (most recent call last):\r\n  File \".../path/to/matplotlib/backends/backend_qt5.py\", line 440, in _draw_idle\r\n    self.draw()\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 269, in draw\r\n    self.get_renderer(_ensure_cleared=True, _ensure_drawn=True)\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 261, in get_renderer\r\n    return self._get_cached_or_new_renderer(\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 256, in _get_cached_or_new_renderer\r\n    self.figure.draw(renderer)\r\n  File \".../path/to/matplotlib/artist.py\", line 74, in draw_wrapper\r\n    result = draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/figure.py\", line 2730, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \".../path/to/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 447, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 447, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 994, in draw\r\n    super().draw(renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 421, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/axes/_base.py\", line 3102, in draw\r\n    mimage._draw_list_compositing_images(renderer, self, artists)\r\n  File \".../path/to/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/axis.py\", line 1124, in draw\r\n    ticks_to_draw = self._update_ticks()\r\n  File \".../path/to/matplotlib/axis.py\", line 1011, in _update_ticks\r\n    major_locs = self.get_majorticklocs()\r\n  File \".../path/to/matplotlib/axis.py\", line 1243, in get_majorticklocs\r\n    return self.major.locator()\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 432, in __call__\r\n    return [tick for tick in self.base() if tick > rorigin]\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 432, in __call__\r\n    return [tick for tick in self.base() if tick > rorigin]\r\n  File \".../path/to/matplotlib/ticker.py\", line 2265, in __call__\r\n    return self.tick_values(vmin, vmax)\r\n  File \".../path/to/matplotlib/ticker.py\", line 2273, in tick_values\r\n    locs = self._raw_ticks(vmin, vmax)\r\n  File \".../path/to/matplotlib/ticker.py\", line 2212, in _raw_ticks\r\n    nbins = np.clip(self.axis.get_tick_space(),\r\n  File \".../path/to/matplotlib/axis.py\", line 2513, in get_tick_space\r\n    length = ((ends[1][1] - ends[0][1]) / self.axes.figure.dpi) * 72\r\nAttributeError: 'NoneType' object has no attribute 'dpi'\r\n```\r\n\r\n**Expected outcome**\r\n\r\nNormal axes removal.\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\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): head\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): mplcairo\r\n  * Python version: 39\r\n  * Jupyter version (if applicable): \r\n  * Other libraries: \r\n\r\n(Note that this is a separate issue from https://github.com/matplotlib/matplotlib/issues/19988 as the root cause seems very different.)\r\n\r\nEdit: I have a fix, but the test is a bit simpler if https://github.com/matplotlib/matplotlib/pull/19994 goes in first.",
        "issue_id": 19989,
        "pr_number": 20009,
        "pr_title": "Fix removal of shared polar axes.",
        "pr_body": "There's really two separate fixes here:\r\n\r\n- Move isDefault_{maj,min}{loc,fmt} tracking to the Ticker instances,\r\n  where they logically belong (note the previous need to additionally\r\n  track them manually on axes removal, when that info was tracked on\r\n  the Axis).  This has the side effect of fixing removal of sharex'd\r\n  polar axes, as ThetaLocators rely on _AxisWrappers which don't have\r\n  that isDefault attribute.  (Note that the patch would have resulted\r\n  in a net decrease of lines of code if it didn't need to maintain\r\n  backcompat on isDefault_foos).  Closes #19988.  Split out of #13482.\r\n\r\n- Ensure that RadialLocator correctly propagates Axis information to\r\n  the linear locator it wraps (consistently with ThetaLocator), so that\r\n  when an axes is removed the wrapped linear locator doesn't stay\r\n  pointing at an obsolete axes.  This, together with the first patch,\r\n  fixes removal of sharey'd polar axes.  Closes #19989.\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n\r\n- [ ] Has pytest style unit tests (and `pytest` passes).\r\n- [ ] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [ ] New features are documented, with examples if plot related.\r\n- [ ] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [ ] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [ ] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [ ] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\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_closed_at": "2021-04-29T20:19:03Z",
        "base_commit": "a94acb3d2c6829d7ce3a4ef2cc33058c677e0526"
      },
      "summary": "### Summary:\nThis issue pertains to a software crash occurring in the Matplotlib library when attempting to remove y-shared polar axes during runtime. The problem arises specifically when a subplot with shared y-axes, using a polar projection, is removed and then the figure is drawn. \n\n1. **Problem description in general terms:**\n   The core issue is a software crash triggered by the removal of polar axes that are shared along the y-axis, leading to an error when the figure tries to render. This is due to the system attempting to access properties of a non-existent or improperly maintained object.\n\n2. **Key symptoms and behaviors observed:**\n   The primary symptom is a traceback error indicating an `AttributeError`, where the system attempts to access the `dpi` attribute of a `NoneType` object. This occurs during the drawing process of the figure, particularly within the functions responsible for updating and drawing ticks on the axes.\n\n3. **Affected components or systems:**\n   The issue affects the Matplotlib library, specifically within the backend handling of polar axes and the drawing functions related to tick updating and rendering. The crash is closely associated with the `mplcairo` backend and involves components such as `axis.py`, `figure.py`, and `projections/polar.py`.\n\n4. **Potential impact or severity:**\n   The severity of this issue is significant for users utilizing polar plots with shared y-axes, as it prevents normal operation and visualization rendering, potentially disrupting data analysis workflows that rely on Matplotlib for plotting.\n\n5. **Relevant technical details abstracted for broader understanding:**\n   The root cause is linked to the improper handling of axis removal and the ensuing state of the figure's draw process. Specifically, when an axis is removed, its associated properties may not be updated or reset correctly, leading to attempts to access properties like `dpi` that are no longer valid. The fix involves modifications to several functions in the Matplotlib codebase to ensure proper initialization and resetting of locators and formatters, particularly for polar axes.",
      "prompt_used": "You are an expert in software issue reasoning analysis.\nGiven the following problem report and its fixed code elements, generate a comprehensive summary based on the entire document. Your goal is to abstract the information in the problem description into a more general description.\n\n## Original Issue Report:\nTitle: Removal of y-shared polar axes causes crash at draw time\n\nBody:\n<!--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\nAll's in the title.\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug.\r\nPlease make sure to minimize the number of dependencies required, and provide\r\nany necessary plotted data.\r\nAvoid using threads, as Matplotlib is (explicitly) not thread-safe.-->\r\n\r\n```python\r\nfrom pylab import *\r\nax1, ax2 = gcf().subplots(2, sharey=True, subplot_kw={\"projection\": \"polar\"}); ax2.remove(); 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```\r\nTraceback (most recent call last):\r\nTraceback (most recent call last):\r\n  File \".../path/to/matplotlib/backends/backend_qt5.py\", line 440, in _draw_idle\r\n    self.draw()\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 269, in draw\r\n    self.get_renderer(_ensure_cleared=True, _ensure_drawn=True)\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 261, in get_renderer\r\n    return self._get_cached_or_new_renderer(\r\n  File \"/home/antony/src/local/mplcairo/lib/mplcairo/base.py\", line 256, in _get_cached_or_new_renderer\r\n    self.figure.draw(renderer)\r\n  File \".../path/to/matplotlib/artist.py\", line 74, in draw_wrapper\r\n    result = draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/figure.py\", line 2730, in draw\r\n    mimage._draw_list_compositing_images(\r\n  File \".../path/to/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 447, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 447, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 994, in draw\r\n    super().draw(renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/_api/deprecation.py\", line 421, in wrapper\r\n    return func(*inner_args, **inner_kwargs)\r\n  File \".../path/to/matplotlib/axes/_base.py\", line 3102, in draw\r\n    mimage._draw_list_compositing_images(renderer, self, artists)\r\n  File \".../path/to/matplotlib/image.py\", line 132, in _draw_list_compositing_images\r\n    a.draw(renderer)\r\n  File \".../path/to/matplotlib/artist.py\", line 51, in draw_wrapper\r\n    return draw(artist, renderer, *args, **kwargs)\r\n  File \".../path/to/matplotlib/axis.py\", line 1124, in draw\r\n    ticks_to_draw = self._update_ticks()\r\n  File \".../path/to/matplotlib/axis.py\", line 1011, in _update_ticks\r\n    major_locs = self.get_majorticklocs()\r\n  File \".../path/to/matplotlib/axis.py\", line 1243, in get_majorticklocs\r\n    return self.major.locator()\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 432, in __call__\r\n    return [tick for tick in self.base() if tick > rorigin]\r\n  File \".../path/to/matplotlib/projections/polar.py\", line 432, in __call__\r\n    return [tick for tick in self.base() if tick > rorigin]\r\n  File \".../path/to/matplotlib/ticker.py\", line 2265, in __call__\r\n    return self.tick_values(vmin, vmax)\r\n  File \".../path/to/matplotlib/ticker.py\", line 2273, in tick_values\r\n    locs = self._raw_ticks(vmin, vmax)\r\n  File \".../path/to/matplotlib/ticker.py\", line 2212, in _raw_ticks\r\n    nbins = np.clip(self.axis.get_tick_space(),\r\n  File \".../path/to/matplotlib/axis.py\", line 2513, in get_tick_space\r\n    length = ((ends[1][1] - ends[0][1]) / self.axes.figure.dpi) * 72\r\nAttributeError: 'NoneType' object has no attribute 'dpi'\r\n```\r\n\r\n**Expected outcome**\r\n\r\nNormal axes removal.\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\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): head\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): mplcairo\r\n  * Python version: 39\r\n  * Jupyter version (if applicable): \r\n  * Other libraries: \r\n\r\n(Note that this is a separate issue from https://github.com/matplotlib/matplotlib/issues/19988 as the root cause seems very different.)\r\n\r\nEdit: I have a fix, but the test is a bit simpler if https://github.com/matplotlib/matplotlib/pull/19994 goes in first.\n\n## Code elements fixed by the patch:\n{FIXED_CODE_ELEMENTS}\n\nPlease analyze the above issue report and provide a structured summary that includes:\n1. Problem description in general terms\n2. Key symptoms and behaviors observed\n3. Affected components or systems\n4. Potential impact or severity\n5. Any relevant technical details abstracted for broader understanding\n\nPlease return the summary with “### Summary:\", For example:\n### Summary: This issue is ...\n\nChanges Summary:\nlib/matplotlib/axis.py\n  function: YAxis.__init__\n\nlib/matplotlib/figure.py\n  function: FigureBase._reset_locators_and_formatters\n\nlib/matplotlib/projections/polar.py\n  function: PolarAxes.__init__\n"
    }
  ]
}