{
  "Selected_candidate": {
    "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_id": 19989,
    "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_closed_at": "2021-04-29T20:19:03Z",
    "base_commit": "a94acb3d2c6829d7ce3a4ef2cc33058c677e0526",
    "changes": [
      {
        "file": "lib/matplotlib/axis.py",
        "type": "function",
        "name": "__init__",
        "class_name": "YAxis",
        "code": "def __init__(self, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        # x in display coords, y in axes coords (to be updated at draw time by\n        # _update_label_positions and _update_offset_text_position).\n        self.label.set(\n            x=0, y=0.5,\n            verticalalignment='bottom', horizontalalignment='center',\n            rotation='vertical', rotation_mode='anchor',\n            transform=mtransforms.blended_transform_factory(\n                mtransforms.IdentityTransform(), self.axes.transAxes),\n        )\n        self.label_position = 'left'\n        # x in axes coords, y in display coords(!).\n        self.offsetText.set(\n            x=0, y=0.5,\n            verticalalignment='baseline', horizontalalignment='left',\n            transform=mtransforms.blended_transform_factory(\n                self.axes.transAxes, mtransforms.IdentityTransform()),\n            fontsize=mpl.rcParams['ytick.labelsize'],\n            color=mpl.rcParams['ytick.color'],\n        )\n        self.offset_text_position = 'left'"
      },
      {
        "file": "lib/matplotlib/figure.py",
        "type": "function",
        "name": "_reset_locators_and_formatters",
        "class_name": "FigureBase",
        "code": "def _reset_locators_and_formatters(axis):\n            # Set the formatters and locators to be associated with axis\n            # (where previously they may have been associated with another\n            # Axis instance)\n            #\n            # Because set_major_formatter() etc. force isDefault_* to be False,\n            # we have to manually check if the original formatter was a\n            # default and manually set isDefault_* if that was the case.\n            majfmt = axis.get_major_formatter()\n            isDefault = majfmt.axis.isDefault_majfmt\n            axis.set_major_formatter(majfmt)\n            if isDefault:\n                majfmt.axis.isDefault_majfmt = True\n\n            majloc = axis.get_major_locator()\n            isDefault = majloc.axis.isDefault_majloc\n            axis.set_major_locator(majloc)\n            if isDefault:\n                majloc.axis.isDefault_majloc = True\n\n            minfmt = axis.get_minor_formatter()\n            isDefault = majloc.axis.isDefault_minfmt\n            axis.set_minor_formatter(minfmt)\n            if isDefault:\n                minfmt.axis.isDefault_minfmt = True\n\n            minloc = axis.get_minor_locator()\n            isDefault = majloc.axis.isDefault_minloc\n            axis.set_minor_locator(minloc)\n            if isDefault:\n                minloc.axis.isDefault_minloc = True"
      },
      {
        "file": "lib/matplotlib/projections/polar.py",
        "type": "function",
        "name": "__init__",
        "class_name": "PolarAxes",
        "code": "def __init__(self, *args,\n                 theta_offset=0, theta_direction=1, rlabel_position=22.5,\n                 **kwargs):\n        # docstring inherited\n        self._default_theta_offset = theta_offset\n        self._default_theta_direction = theta_direction\n        self._default_rlabel_position = np.deg2rad(rlabel_position)\n        super().__init__(*args, **kwargs)\n        self.use_sticky_edges = True\n        self.set_aspect('equal', adjustable='box', anchor='C')\n        self.cla()"
      }
    ]
  },
  "Justification": "Candidate E is the most helpful for the CURRENT bug report, as it directly deals with the management of shared axes in Matplotlib, which is a critical aspect of the CURRENT bug involving the \"xlim_changed\" callback not being emitted correctly on shared axes. The structural similarity is evident in how both reports discuss issues with axis management and callback triggering, particularly in shared contexts. Fixes in this report address propagation of attributes necessary for ensuring shared axis behavior works as intended, which can provide insights into modifying the existing callback behavior in the CURRENT bug.",
  "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"
}