{
  "Selected_candidate": {
    "pr_number": 17784,
    "pr_title": "Allow passing empty list of ticks to FixedLocator",
    "pr_body": "## PR Summary\r\nFixes #17736. See https://github.com/matplotlib/matplotlib/issues/17736#issuecomment-648496870 for reasons why I haven't added a deprecation; we can always add that later anyway if desired. The test errors without the fix, but maybe I should make it a figure comparison to make sure the ticks are being removed?\r\n\r\n## PR Checklist\r\n\r\n- [x] Has Pytest style unit tests\r\n- [x] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) compliant\r\n- [ ] New features are documented, with examples if plot related\r\n- [ ] Documentation is sphinx and numpydoc compliant\r\n- [ ] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [ ] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
    "issue_id": 17736,
    "issue_title": "ax.set_xticklabels([]) for categorical plots is broken in 3.3.0rc1",
    "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\n#17266 requires tick labels set by `FixedFormatter` to be equal to the ticks set by `FixedLocator`. `FixedLocator` is used in many cases where categorical data is plotted, e.g. `bar` and `box plot`. A way of removing all tick labels (while maintaining ticks and gridlines) is to do `ax.set_xticklabels([])` or `ax.set_yticklabels([])`, by passing just an empty list. This works on 3.2.2, but raises a `ValueError` with 3.3.0rc1. \r\n\r\nThe new behavior is not mentioned in the \"What's new?\" section of 3.3.0rc1. I suspect this may break existing code, as it broke seaborn and [this stackoverflow thread](https://stackoverflow.com/questions/2176424/hiding-axis-text-in-matplotlib-plots) seems very popular.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\n# Example modified from bar example page\r\n\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\n\r\nlabels = ['G1', 'G2', 'G3', 'G4', 'G5']\r\nmen_means = [20, 34, 30, 35, 27]\r\nwomen_means = [25, 32, 34, 20, 25]\r\n\r\nx = np.arange(len(labels))  # the label locations\r\nwidth = 0.35  # the width of the bars\r\n\r\nfig, ax = plt.subplots()\r\nrects1 = ax.bar(x - width/2, men_means, width, label='Men')\r\nrects2 = ax.bar(x + width/2, women_means, width, label='Women')\r\n\r\nax.set_xticks(x)\r\nax.set_xticklabels([])\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\n# 3.3.0rc1\r\nValueError: The number of FixedLocator locations (5), usually from a call to set_ticks, does not match the number of ticklabels (0).\r\n```\r\n\r\n**Expected outcome**\r\n\r\nUnset tick labels but not error out, or raise a deprecation warning that this won't be possible in a future release.\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: macOS\r\n  * Matplotlib version: 3.3.0rc1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): MacOSX\r\n  * Python version: 3.8.3\r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\nInstalled using pip",
    "issue_closed_at": "2020-06-28T17:52:18Z",
    "base_commit": "56d4f69923d5a4bb1158c7d61fbc7e24bd10d9b9",
    "changes": [
      {
        "file": "lib/matplotlib/axis.py",
        "type": "function",
        "name": "set_ticklabels",
        "class_name": "Axis",
        "code": "def set_ticklabels(self, ticklabels, *, minor=False, **kwargs):\n        r\"\"\"\n        Set the text values of the tick labels.\n\n        .. warning::\n            This method should only be used after fixing the tick positions\n            using `.Axis.set_ticks`. Otherwise, the labels may end up in\n            unexpected positions.\n\n        Parameters\n        ----------\n        ticklabels : sequence of str or of `.Text`\\s\n            Texts for labeling each tick location in the sequence set by\n            `.Axis.set_ticks`; the number of labels must match the number of\n            locations.\n        minor : bool\n            If True, set minor ticks instead of major ticks.\n        **kwargs\n            Text properties.\n\n        Returns\n        -------\n        list of `.Text`\\s\n            For each tick, includes ``tick.label1`` if it is visible, then\n            ``tick.label2`` if it is visible, in that order.\n        \"\"\"\n        ticklabels = [t.get_text() if hasattr(t, 'get_text') else t\n                      for t in ticklabels]\n        locator = (self.get_minor_locator() if minor\n                   else self.get_major_locator())\n        if isinstance(locator, mticker.FixedLocator):\n            if len(locator.locs) != len(ticklabels):\n                raise ValueError(\n                    \"The number of FixedLocator locations\"\n                    f\" ({len(locator.locs)}), usually from a call to\"\n                    \" set_ticks, does not match\"\n                    f\" the number of ticklabels ({len(ticklabels)}).\")\n            tickd = {loc: lab for loc, lab in zip(locator.locs, ticklabels)}\n            func = functools.partial(self._format_with_dict, tickd)\n            formatter = mticker.FuncFormatter(func)\n        else:\n            formatter = mticker.FixedFormatter(ticklabels)\n\n        if minor:\n            self.set_minor_formatter(formatter)\n            locs = self.get_minorticklocs()\n            ticks = self.get_minor_ticks(len(locs))\n        else:\n            self.set_major_formatter(formatter)\n            locs = self.get_majorticklocs()\n            ticks = self.get_major_ticks(len(locs))\n\n        ret = []\n        for pos, (loc, tick) in enumerate(zip(locs, ticks)):\n            tick.update_position(loc)\n            tick_label = formatter(loc, pos)\n            # deal with label1\n            tick.label1.set_text(tick_label)\n            tick.label1.update(kwargs)\n            # deal with label2\n            tick.label2.set_text(tick_label)\n            tick.label2.update(kwargs)\n            # only return visible tick labels\n            if tick.label1.get_visible():\n                ret.append(tick.label1)\n            if tick.label2.get_visible():\n                ret.append(tick.label2)\n\n        self.stale = True\n        return ret"
      }
    ]
  },
  "Justification": "CANDIDATE_BUG_REPORT_A is the most relevant because it addresses issues specifically related to tick labels and validation, similar to the CURRENT_BUG_REPORT about `Axis.set_ticks`. The shared module (axis), symptoms (issues with tick handling), and similar patches indicate that this candidate might provide valuable insights for diagnosing the CURRENT_bug. The learning gained from the handling of `FixedFormatter` and `FixedLocator` in Candidate A may directly inform the necessary changes in the current issue, especially regarding validation and error handling for tick labels.",
  "instance_id": "matplotlib__matplotlib-24334",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-11-01T18:11:43Z",
  "problem_statement": "[ENH]: Axes.set_xticks/Axis.set_ticks only validates kwargs if ticklabels are set, but they should\n### Problem\n\nPer the doc of `Axis.set_ticks`:\r\n```\r\n        **kwargs\r\n            `.Text` properties for the labels. These take effect only if you\r\n            pass *labels*. In other cases, please use `~.Axes.tick_params`.\r\n```\r\nThis means that in e.g. `ax.set_xticks([0, 1], xticklabels=[\"a\", \"b\"])`, the incorrect `xticklabels` silently do nothing; they are not even validated (because `labels` has not been passed).\n\n### Proposed solution\n\nWe should at least check that `kwargs` are valid Text properties in all cases; we could even consider making any kwargs an error if `labels` is not set.\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@@ -2029,6 +2029,9 @@ def set_ticks(self, ticks, labels=None, *, minor=False, **kwargs):\n         other limits, you should set the limits explicitly after setting the\n         ticks.\n         \"\"\"\n+        if labels is None and kwargs:\n+            raise ValueError('labels argument cannot be None when '\n+                             'kwargs are passed')\n         result = self._set_tick_locations(ticks, minor=minor)\n         if labels is not None:\n             self.set_ticklabels(labels, minor=minor, **kwargs)\n"
}