{
  "instance_id": "matplotlib__matplotlib-24265",
  "repo": "matplotlib/matplotlib",
  "created_at": "2022-10-25T02:03:19Z",
  "problem_statement": "[Bug]: Setting matplotlib.pyplot.style.library['seaborn-colorblind'] result in key error on matplotlib v3.6.1\n### Bug summary\n\nI have code that executes:\r\n```\r\nimport matplotlib.pyplot as plt\r\nthe_rc = plt.style.library[\"seaborn-colorblind\"]\r\n```\r\n\r\nUsing version 3.4.3 of matplotlib, this works fine. I recently installed my code on a machine with matplotlib version 3.6.1 and upon importing my code, this generated a key error for line `the_rc = plt.style.library[\"seaborn-colorblind\"]` saying \"seaborn-colorblind\" was a bad key.\n\n### Code for reproduction\n\n```python\nimport matplotlib.pyplot as plt\r\nthe_rc = plt.style.library[\"seaborn-colorblind\"]\n```\n\n\n### Actual outcome\n\nTraceback (most recent call last):\r\nKeyError: 'seaborn-colorblind'\n\n### Expected outcome\n\nseaborn-colorblind should be set as the matplotlib library style and I should be able to continue plotting with that style.\n\n### Additional information\n\n- Bug occurs with matplotlib version 3.6.1\r\n- Bug does not occur with matplotlib version 3.4.3\r\n- Tested on MacOSX and Ubuntu (same behavior on both)\n\n### Operating system\n\nOS/X\n\n### Matplotlib Version\n\n3.6.1\n\n### Matplotlib Backend\n\nMacOSX\n\n### Python version\n\n3.9.7\n\n### Jupyter version\n\n_No response_\n\n### Installation\n\npip\n",
  "patch": "diff --git a/lib/matplotlib/style/core.py b/lib/matplotlib/style/core.py\n--- a/lib/matplotlib/style/core.py\n+++ b/lib/matplotlib/style/core.py\n@@ -43,6 +43,32 @@ class __getattr__:\n     'toolbar', 'timezone', 'figure.max_open_warning',\n     'figure.raise_window', 'savefig.directory', 'tk.window_focus',\n     'docstring.hardcopy', 'date.epoch'}\n+_DEPRECATED_SEABORN_STYLES = {\n+    s: s.replace(\"seaborn\", \"seaborn-v0_8\")\n+    for s in [\n+        \"seaborn\",\n+        \"seaborn-bright\",\n+        \"seaborn-colorblind\",\n+        \"seaborn-dark\",\n+        \"seaborn-darkgrid\",\n+        \"seaborn-dark-palette\",\n+        \"seaborn-deep\",\n+        \"seaborn-muted\",\n+        \"seaborn-notebook\",\n+        \"seaborn-paper\",\n+        \"seaborn-pastel\",\n+        \"seaborn-poster\",\n+        \"seaborn-talk\",\n+        \"seaborn-ticks\",\n+        \"seaborn-white\",\n+        \"seaborn-whitegrid\",\n+    ]\n+}\n+_DEPRECATED_SEABORN_MSG = (\n+    \"The seaborn styles shipped by Matplotlib are deprecated since %(since)s, \"\n+    \"as they no longer correspond to the styles shipped by seaborn. However, \"\n+    \"they will remain available as 'seaborn-v0_8-<style>'. Alternatively, \"\n+    \"directly use the seaborn API instead.\")\n \n \n def _remove_blacklisted_style_params(d, warn=True):\n@@ -113,31 +139,9 @@ def use(style):\n     def fix_style(s):\n         if isinstance(s, str):\n             s = style_alias.get(s, s)\n-            if s in [\n-                \"seaborn\",\n-                \"seaborn-bright\",\n-                \"seaborn-colorblind\",\n-                \"seaborn-dark\",\n-                \"seaborn-darkgrid\",\n-                \"seaborn-dark-palette\",\n-                \"seaborn-deep\",\n-                \"seaborn-muted\",\n-                \"seaborn-notebook\",\n-                \"seaborn-paper\",\n-                \"seaborn-pastel\",\n-                \"seaborn-poster\",\n-                \"seaborn-talk\",\n-                \"seaborn-ticks\",\n-                \"seaborn-white\",\n-                \"seaborn-whitegrid\",\n-            ]:\n-                _api.warn_deprecated(\n-                    \"3.6\", message=\"The seaborn styles shipped by Matplotlib \"\n-                    \"are deprecated since %(since)s, as they no longer \"\n-                    \"correspond to the styles shipped by seaborn. However, \"\n-                    \"they will remain available as 'seaborn-v0_8-<style>'. \"\n-                    \"Alternatively, directly use the seaborn API instead.\")\n-                s = s.replace(\"seaborn\", \"seaborn-v0_8\")\n+            if s in _DEPRECATED_SEABORN_STYLES:\n+                _api.warn_deprecated(\"3.6\", message=_DEPRECATED_SEABORN_MSG)\n+                s = _DEPRECATED_SEABORN_STYLES[s]\n         return s\n \n     for style in map(fix_style, styles):\n@@ -244,17 +248,26 @@ def update_nested_dict(main_dict, new_dict):\n     return main_dict\n \n \n+class _StyleLibrary(dict):\n+    def __getitem__(self, key):\n+        if key in _DEPRECATED_SEABORN_STYLES:\n+            _api.warn_deprecated(\"3.6\", message=_DEPRECATED_SEABORN_MSG)\n+            key = _DEPRECATED_SEABORN_STYLES[key]\n+\n+        return dict.__getitem__(self, key)\n+\n+\n # Load style library\n # ==================\n _base_library = read_style_directory(BASE_LIBRARY_PATH)\n-library = None\n+library = _StyleLibrary()\n available = []\n \n \n def reload_library():\n     \"\"\"Reload the style library.\"\"\"\n-    global library\n-    library = update_user_library(_base_library)\n+    library.clear()\n+    library.update(update_user_library(_base_library))\n     available[:] = sorted(library.keys())\n \n \n",
  "similar_bug_items": [
    {
      "pr_number": 10772,
      "pr_title": "API: check locator and formatter args when passed",
      "pr_body": "## PR Summary\r\n\r\nRight now, `ax.set_major_formatter` and `ax.set_major_locator` throw pretty deep error messages if the wrong type of data is passed to them.  This catches at entry. \r\n\r\nCloses #10769\r\n\r\n```python\r\nimport matplotlib\r\nimport matplotlib.pyplot as plt\r\n\r\nf, ax = plt.subplots(1, 1, figsize=(15, 10))\r\nplt.plot([10**4, 11*10**4], [100, 150])\r\nax.xaxis.set_major_formatter(matplotlib.ticker.LogLocator())\r\nplt.show()\r\n```\r\nNow returns:\r\n```\r\nTraceback (most recent call last):\r\n  File \"testEngTicker.py\", line 6, in <module>\r\n    ax.xaxis.set_major_formatter(matplotlib.ticker.LogLocator())\r\n  File \"/Users/jklymak/matplotlib/lib/matplotlib/axis.py\", line 1573, in set_major_formatter\r\n    \"formatter argument must be matplotlib.ticker.Formatter\")\r\nValueError: formatter argument must be matplotlib.ticker.Formatter\r\n```\r\ninstead of something down in the bowels of drawing the ticker...\r\n\r\n## PR Checklist\r\n\r\n- [x] Code is PEP 8 compliant\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->",
      "issue_id": 10769,
      "issue_title": "DOC: set_major_locator could check that its getting a Locator (was EngFormatter broken?)",
      "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`matplotlib.ticker.EngFormatter` seems to be broken:\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n```\r\nf, ax = plt.subplots(1, 1, figsize=(15, 10))\r\nplt.plot([10**4, 11*10**4], [100, 150])\r\nax.xaxis.set_major_locator(matplotlib.ticker.EngFormatter())\r\n```\r\n```\r\nError in callback <function install_repl_displayhook.<locals>.post_execute at 0x7f60a9b120d0> (for post_execute):\r\n\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/pyplot.py in post_execute()\r\n    147             def post_execute():\r\n    148                 if matplotlib.is_interactive():\r\n--> 149                     draw_all()\r\n    150 \r\n    151             # IPython >= 2\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/_pylab_helpers.py in draw_all(cls, force)\r\n    134         for f_mgr in cls.get_all_fig_managers():\r\n    135             if force or f_mgr.canvas.figure.stale:\r\n--> 136                 f_mgr.canvas.draw_idle()\r\n    137 \r\n    138 atexit.register(Gcf.destroy_all)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backend_bases.py in draw_idle(self, *args, **kwargs)\r\n   2053         if not self._is_idle_drawing:\r\n   2054             with self._idle_draw_cntx():\r\n-> 2055                 self.draw(*args, **kwargs)\r\n   2056 \r\n   2057     def draw_cursor(self, event):\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in draw(self)\r\n    429             # if toolbar:\r\n    430             #     toolbar.set_cursor(cursors.WAIT)\r\n--> 431             self.figure.draw(self.renderer)\r\n    432             # A GUI class may be need to update a window using this draw, so\r\n    433             # don't forget to call the superclass.\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py in draw(self, renderer)\r\n   1473 \r\n   1474             mimage._draw_list_compositing_images(\r\n-> 1475                 renderer, self, artists, self.suppressComposite)\r\n   1476 \r\n   1477             renderer.close_group('figure')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py in draw(self, renderer, inframe)\r\n   2605             renderer.stop_rasterizing()\r\n   2606 \r\n-> 2607         mimage._draw_list_compositing_images(renderer, self, artists)\r\n   2608 \r\n   2609         renderer.close_group('axes')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in draw(self, renderer, *args, **kwargs)\r\n   1185         renderer.open_group(__name__)\r\n   1186 \r\n-> 1187         ticks_to_draw = self._update_ticks(renderer)\r\n   1188         ticklabelBoxes, ticklabelBoxes2 = self._get_tick_bboxes(ticks_to_draw,\r\n   1189                                                                 renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in _update_ticks(self, renderer)\r\n   1023 \r\n   1024         interval = self.get_view_interval()\r\n-> 1025         tick_tups = list(self.iter_ticks())  # iter_ticks calls the locator\r\n   1026         if self._smart_bounds and tick_tups:\r\n   1027             # handle inverted limits\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in iter_ticks(self)\r\n    966         Iterate through all of the major and minor ticks.\r\n    967         \"\"\"\r\n--> 968         majorLocs = self.major.locator()\r\n    969         majorTicks = self.get_major_ticks(len(majorLocs))\r\n    970         self.major.formatter.set_locs(majorLocs)\r\n\r\nTypeError: __call__() missing 1 required positional argument: 'x'\r\n\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/formatters.py in __call__(self, obj)\r\n    339                 pass\r\n    340             else:\r\n--> 341                 return printer(obj)\r\n    342             # Finally look for special method names\r\n    343             method = get_real_method(obj, self.print_method)\r\n\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/pylabtools.py in <lambda>(fig)\r\n    236 \r\n    237     if 'png' in formats:\r\n--> 238         png_formatter.for_type(Figure, lambda fig: print_figure(fig, 'png', **kwargs))\r\n    239     if 'retina' in formats or 'png2x' in formats:\r\n    240         png_formatter.for_type(Figure, lambda fig: retina_figure(fig, **kwargs))\r\n\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/pylabtools.py in print_figure(fig, fmt, bbox_inches, **kwargs)\r\n    120 \r\n    121     bytes_io = BytesIO()\r\n--> 122     fig.canvas.print_figure(bytes_io, **kw)\r\n    123     data = bytes_io.getvalue()\r\n    124     if fmt == 'svg':\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backend_bases.py in print_figure(self, filename, dpi, facecolor, edgecolor, orientation, format, **kwargs)\r\n   2208                     orientation=orientation,\r\n   2209                     dryrun=True,\r\n-> 2210                     **kwargs)\r\n   2211                 renderer = self.figure._cachedRenderer\r\n   2212                 bbox_inches = self.figure.get_tightbbox(renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in print_png(self, filename_or_obj, *args, **kwargs)\r\n    509 \r\n    510     def print_png(self, filename_or_obj, *args, **kwargs):\r\n--> 511         FigureCanvasAgg.draw(self)\r\n    512         renderer = self.get_renderer()\r\n    513         original_dpi = renderer.dpi\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in draw(self)\r\n    429             # if toolbar:\r\n    430             #     toolbar.set_cursor(cursors.WAIT)\r\n--> 431             self.figure.draw(self.renderer)\r\n    432             # A GUI class may be need to update a window using this draw, so\r\n    433             # don't forget to call the superclass.\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py in draw(self, renderer)\r\n   1473 \r\n   1474             mimage._draw_list_compositing_images(\r\n-> 1475                 renderer, self, artists, self.suppressComposite)\r\n   1476 \r\n   1477             renderer.close_group('figure')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py in draw(self, renderer, inframe)\r\n   2605             renderer.stop_rasterizing()\r\n   2606 \r\n-> 2607         mimage._draw_list_compositing_images(renderer, self, artists)\r\n   2608 \r\n   2609         renderer.close_group('axes')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in draw(self, renderer, *args, **kwargs)\r\n   1185         renderer.open_group(__name__)\r\n   1186 \r\n-> 1187         ticks_to_draw = self._update_ticks(renderer)\r\n   1188         ticklabelBoxes, ticklabelBoxes2 = self._get_tick_bboxes(ticks_to_draw,\r\n   1189                                                                 renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in _update_ticks(self, renderer)\r\n   1023 \r\n   1024         interval = self.get_view_interval()\r\n-> 1025         tick_tups = list(self.iter_ticks())  # iter_ticks calls the locator\r\n   1026         if self._smart_bounds and tick_tups:\r\n   1027             # handle inverted limits\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in iter_ticks(self)\r\n    966         Iterate through all of the major and minor ticks.\r\n    967         \"\"\"\r\n--> 968         majorLocs = self.major.locator()\r\n    969         majorTicks = self.get_major_ticks(len(majorLocs))\r\n    970         self.major.formatter.set_locs(majorLocs)\r\n\r\nTypeError: __call__() missing 1 required positional argument: 'x'\r\n\r\n<Figure size 1080x720 with 1 Axes>\r\n```\r\n<!--A minimum code snippet required to reproduce the bug, also minimizing the number of dependencies required-->\r\n\r\n\r\n<!--A description of the expected outcome from the code snippet-->\r\n<!--If this used to work in an earlier version of Matplotlib, please note the version it used to work on-->\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: Ubuntu 16.04\r\n  * Matplotlib version: 2.2.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): module://ipykernel.pylab.backend_inline\r\n  * Python version: 3.5.2 (IPython 6.2.1)\r\n  * Jupyter version (if applicable): Jupyter Lab beta 0.31.8\r\n\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\n",
      "issue_closed_at": "2018-03-14T06:41:19Z",
      "base_commit": "8af1fac8baf9bbf06160d8a91575d3d6d8fdca04",
      "changes": [
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_major_formatter",
          "class_name": "Axis",
          "code": "def set_major_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the major ticker\n\n        ACCEPTS: A :class:`~matplotlib.ticker.Formatter` instance\n        \"\"\"\n        self.isDefault_majfmt = False\n        self.major.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_minor_formatter",
          "class_name": "Axis",
          "code": "def set_minor_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the minor ticker\n\n        ACCEPTS: A :class:`~matplotlib.ticker.Formatter` instance\n        \"\"\"\n        self.isDefault_minfmt = False\n        self.minor.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_major_locator",
          "class_name": "Axis",
          "code": "def set_major_locator(self, locator):\n        \"\"\"\n        Set the locator of the major ticker\n\n        ACCEPTS: a :class:`~matplotlib.ticker.Locator` instance\n        \"\"\"\n        self.isDefault_majloc = False\n        self.major.locator = locator\n        locator.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_minor_locator",
          "class_name": "Axis",
          "code": "def set_minor_locator(self, locator):\n        \"\"\"\n        Set the locator of the minor ticker\n\n        ACCEPTS: a :class:`~matplotlib.ticker.Locator` instance\n        \"\"\"\n        self.isDefault_minloc = False\n        self.minor.locator = locator\n        locator.set_axis(self)\n        self.stale = True"
        }
      ]
    },
    {
      "pr_number": 13983,
      "pr_title": "Fix locator/formatter setting when removing shared Axes",
      "pr_body": "When a shared Axes was removed from a plot, any other Axes it was shared with had their Axis locators/formatters set to the locator/formatter on the removed Axes. This PR makes sure that setting only happens if the locator/formatter has previously been manually set. This allows the locator/formatter to be automatically set in the future by e.g. units being plotted.\r\n\r\n\r\nFixes  #12853",
      "issue_id": 12853,
      "issue_title": "Remove()ing a shared axes prevents the remaining axes from using unit-provided formatters",
      "issue_body": "Consider\r\n```\r\nfrom pylab import *\r\nfrom datetime import date\r\n\r\nfig, axs = plt.subplots(1, 2, sharex=True)\r\naxs[1].remove()\r\naxs[0].plot([date(2000, 1, 1), date(2000, 2, 1)], [0, 1])\r\nplt.show()\r\n```\r\n\r\nOne gets\r\n![test](https://user-images.githubusercontent.com/1322974/48794454-4c3f5c00-ecfa-11e8-9e1f-83ff6015782c.png)\r\n\r\ni.e. the call to `axs[1].remove()` prevented the axs[0] from acquiring the correct tick formatter and locator.\r\n\r\nInterestingly, using `fig.delaxes(axs[1])` doesn't exhibit the same bug.\r\n\r\nLooks like the problem comes from\r\n```\r\n    def _remove_ax(self, ax):\r\n        def _reset_loc_form(axis):\r\n            axis.set_major_formatter(axis.get_major_formatter())\r\n            axis.set_major_locator(axis.get_major_locator())\r\n            axis.set_minor_formatter(axis.get_minor_formatter())\r\n            axis.set_minor_locator(axis.get_minor_locator())\r\n\r\n        def _break_share_link(ax, grouper):\r\n            siblings = grouper.get_siblings(ax)\r\n            if len(siblings) > 1:\r\n                grouper.remove(ax)\r\n                for last_ax in siblings:\r\n                    if ax is not last_ax:\r\n                        return last_ax\r\n            return None\r\n\r\n        self.delaxes(ax)\r\n        last_ax = _break_share_link(ax, ax._shared_y_axes)\r\n        if last_ax is not None:\r\n            _reset_loc_form(last_ax.yaxis)\r\n\r\n        last_ax = _break_share_link(ax, ax._shared_x_axes)\r\n        if last_ax is not None:\r\n            _reset_loc_form(last_ax.xaxis)\r\n```\r\nwhere the call to `set_major_formatter` (etc.), which basically call `formatter.set_axis(axis)` (to update the axis seen by the formatter) also make Matplotlib believe that we had a user-provided formatter (isDefault_majloc = False, etc.) which should not be overridden by the unit framework.\r\n\r\nmpl master (ca. 3.0.2)",
      "issue_closed_at": "2019-04-20T20:29:01Z",
      "base_commit": "76db50151a65927c19c83a8c3c195c87dbcc0556",
      "changes": [
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_major_formatter",
          "class_name": "Axis",
          "code": "def set_major_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the major ticker.\n\n        Parameters\n        ----------\n        formatter : ~matplotlib.ticker.Formatter\n        \"\"\"\n        if not isinstance(formatter, mticker.Formatter):\n            raise TypeError(\"formatter argument should be instance of \"\n                    \"matplotlib.ticker.Formatter\")\n        self.isDefault_majfmt = False\n        self.major.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/figure.py",
          "type": "function",
          "name": "subplots",
          "class_name": "Figure",
          "code": "def subplots(self, nrows=1, ncols=1, sharex=False, sharey=False,\n                 squeeze=True, subplot_kw=None, gridspec_kw=None):\n        \"\"\"\n        Add a set of subplots to this figure.\n\n        This utility wrapper makes it convenient to create common layouts of\n        subplots in a single call.\n\n        Parameters\n        ----------\n        nrows, ncols : int, optional, default: 1\n            Number of rows/columns of the subplot grid.\n\n        sharex, sharey : bool or {'none', 'all', 'row', 'col'}, default: False\n            Controls sharing of properties among x (`sharex`) or y (`sharey`)\n            axes:\n\n                - True or 'all': x- or y-axis will be shared among all\n                  subplots.\n                - False or 'none': each subplot x- or y-axis will be\n                  independent.\n                - 'row': each subplot row will share an x- or y-axis.\n                - 'col': each subplot column will share an x- or y-axis.\n\n            When subplots have a shared x-axis along a column, only the x tick\n            labels of the bottom subplot are created. Similarly, when subplots\n            have a shared y-axis along a row, only the y tick labels of the\n            first column subplot are created. To later turn other subplots'\n            ticklabels on, use `~matplotlib.axes.Axes.tick_params`.\n\n        squeeze : bool, optional, default: True\n            - If True, extra dimensions are squeezed out from the returned\n              array of Axes:\n\n                - if only one subplot is constructed (nrows=ncols=1), the\n                  resulting single Axes object is returned as a scalar.\n                - for Nx1 or 1xM subplots, the returned object is a 1D numpy\n                  object array of Axes objects.\n                - for NxM, subplots with N>1 and M>1 are returned\n                  as a 2D array.\n\n            - If False, no squeezing at all is done: the returned Axes object\n              is always a 2D array containing Axes instances, even if it ends\n              up being 1x1.\n\n        subplot_kw : dict, optional\n            Dict with keywords passed to the\n            :meth:`~matplotlib.figure.Figure.add_subplot` call used to create\n            each subplot.\n\n        gridspec_kw : dict, optional\n            Dict with keywords passed to the\n            `~matplotlib.gridspec.GridSpec` constructor used to create\n            the grid the subplots are placed on.\n\n        Returns\n        -------\n        ax : `~.axes.Axes` object or array of Axes objects.\n            *ax* can be either a single `~matplotlib.axes.Axes` object or\n            an array of Axes objects if more than one subplot was created. The\n            dimensions of the resulting array can be controlled with the\n            squeeze keyword, see above.\n\n        Examples\n        --------\n        ::\n\n            # First create some toy data:\n            x = np.linspace(0, 2*np.pi, 400)\n            y = np.sin(x**2)\n\n            # Create a figure\n            plt.figure()\n\n            # Creates a subplot\n            ax = fig.subplots()\n            ax.plot(x, y)\n            ax.set_title('Simple plot')\n\n            # Creates two subplots and unpacks the output array immediately\n            ax1, ax2 = fig.subplots(1, 2, sharey=True)\n            ax1.plot(x, y)\n            ax1.set_title('Sharing Y axis')\n            ax2.scatter(x, y)\n\n            # Creates four polar axes, and accesses them through the\n            # returned array\n            axes = fig.subplots(2, 2, subplot_kw=dict(polar=True))\n            axes[0, 0].plot(x, y)\n            axes[1, 1].scatter(x, y)\n\n            # Share a X axis with each column of subplots\n            fig.subplots(2, 2, sharex='col')\n\n            # Share a Y axis with each row of subplots\n            fig.subplots(2, 2, sharey='row')\n\n            # Share both X and Y axes with all subplots\n            fig.subplots(2, 2, sharex='all', sharey='all')\n\n            # Note that this is the same as\n            fig.subplots(2, 2, sharex=True, sharey=True)\n\n            See Also\n            --------\n            .pyplot.subplots\n            .Figure.add_subplot\n            .pyplot.subplot\n            \"\"\"\n\n        if isinstance(sharex, bool):\n            sharex = \"all\" if sharex else \"none\"\n        if isinstance(sharey, bool):\n            sharey = \"all\" if sharey else \"none\"\n        # This check was added because it is very easy to type\n        # `subplots(1, 2, 1)` when `subplot(1, 2, 1)` was intended.\n        # In most cases, no error will ever occur, but mysterious behavior\n        # will result because what was intended to be the subplot index is\n        # instead treated as a bool for sharex.\n        if isinstance(sharex, Integral):\n            cbook._warn_external(\n                \"sharex argument to subplots() was an integer.  Did you \"\n                \"intend to use subplot() (without 's')?\")\n        cbook._check_in_list([\"all\", \"row\", \"col\", \"none\"],\n                             sharex=sharex, sharey=sharey)\n        if subplot_kw is None:\n            subplot_kw = {}\n        if gridspec_kw is None:\n            gridspec_kw = {}\n        # don't mutate kwargs passed by user...\n        subplot_kw = subplot_kw.copy()\n        gridspec_kw = gridspec_kw.copy()\n\n        if self.get_constrained_layout():\n            gs = GridSpec(nrows, ncols, figure=self, **gridspec_kw)\n        else:\n            # this should turn constrained_layout off if we don't want it\n            gs = GridSpec(nrows, ncols, figure=None, **gridspec_kw)\n        self._gridspecs.append(gs)\n\n        # Create array to hold all axes.\n        axarr = np.empty((nrows, ncols), dtype=object)\n        for row in range(nrows):\n            for col in range(ncols):\n                shared_with = {\"none\": None, \"all\": axarr[0, 0],\n                               \"row\": axarr[row, 0], \"col\": axarr[0, col]}\n                subplot_kw[\"sharex\"] = shared_with[sharex]\n                subplot_kw[\"sharey\"] = shared_with[sharey]\n                axarr[row, col] = self.add_subplot(gs[row, col], **subplot_kw)\n\n        # turn off redundant tick labeling\n        if sharex in [\"col\", \"all\"]:\n            # turn off all but the bottom row\n            for ax in axarr[:-1, :].flat:\n                ax.xaxis.set_tick_params(which='both',\n                                         labelbottom=False, labeltop=False)\n                ax.xaxis.offsetText.set_visible(False)\n        if sharey in [\"row\", \"all\"]:\n            # turn off all but the first column\n            for ax in axarr[:, 1:].flat:\n                ax.yaxis.set_tick_params(which='both',\n                                         labelleft=False, labelright=False)\n                ax.yaxis.offsetText.set_visible(False)\n\n        if squeeze:\n            # Discarding unneeded dimensions that equal 1.  If we only have one\n            # subplot, just return it instead of a 1-element array.\n            return axarr.item() if axarr.size == 1 else axarr.squeeze()\n        else:\n            # Returned axis array will be always 2-d, even if nrows=ncols=1.\n            return axarr"
        }
      ]
    },
    {
      "pr_number": 4186,
      "pr_title": "Close clipped paths",
      "pr_body": "Fix #4185.\n",
      "issue_id": 4185,
      "issue_title": "Colorbar outline has broken path in vector backends",
      "issue_body": "Hi again,\nIn all of the vector graphics backends that I tested (pdf, eps, ps, svg), the colorbar outline is drawn with a broken path at the corner where the path starts and ends.  This does not happen with the raster graphic or interactive backends.  Here is a minimal working example:\n\n``` python\nimport numpy as np\nimport matplotlib as mpl\nimport matplotlib.pyplot as plt\n\nmpl.rcParams.update({'axes.linewidth':10})\n\nfig = plt.figure(figsize=(8,6))\nax1 = fig.add_axes([0.05, 0.85, 0.9, 0.1])\nax2 = fig.add_axes([0.05, 0.65, 0.9, 0.1])\nax3 = fig.add_axes([0.05, 0.45, 0.9, 0.1])\nax4 = fig.add_axes([0.05, 0.25, 0.9, 0.1])\nax5 = fig.add_axes([0.05, 0.05, 0.9, 0.1])\n\ncmap = mpl.cm.jet\ncmap.set_under('w')\ncmap.set_over('w')\n\nim = ax1.pcolormesh(np.linspace(0,10,16).reshape((4,4)))\n\nplt.colorbar(im,cax=ax2,cmap=cmap,orientation='horizontal',\n             extend='both',extendfrac=0.5)\nplt.colorbar(im,cax=ax3,cmap=cmap,orientation='horizontal',\n             extend='both',)\nplt.colorbar(im,cax=ax4,cmap=cmap,orientation='horizontal',\n             extend='both',extendrect=True)\nplt.colorbar(im,cax=ax5,cmap=cmap,orientation='horizontal',\n             extend='neither')\n\nplt.savefig('colorbar_tip.pdf')\nplt.savefig('colorbar_tip.eps')\nplt.savefig('colorbar_tip.svg')\n```\n\nTested with matplotlib 1.4.3, and python 2.7, on Linux and Windows. I have exaggerated the axes linewidth to make the problem more visible.  Here is a screenshot of the PDF output:\n\n![colorbar_tip](https://cloud.githubusercontent.com/assets/8595627/6471312/c556da04-c1b7-11e4-8711-25ba893eb043.png)\n\nI have been able to fix this by making the following change to the `ColorbarBase` class in colorbar.py:\n\n``` python\ndef _config_axes(self, X, Y):\n\n    ...\n\n    if self.outline is not None:\n        self.outline.remove()\n\n    xy2 = np.append(xy,xy[1:2,:],axis=0) # <-- added\n\n    self.outline = mpatches.Polygon(\n        xy2, edgecolor=mpl.rcParams['axes.edgecolor'], # <-- modified\n        facecolor='none',\n        linewidth=mpl.rcParams['axes.linewidth'],\n        closed=True,\n        zorder=2)\n\n    ...\n```\n\nThis essentially just copies the second point of the path to the end of the path.  However, I don't know if this is the best fix.  I don't know anything about vector graphics drawing directives, but it seems to me that there should be a builtin function to have paths wrap-around enough so that it creates a closed loop, including drawing the correct corner.  Clearly the `closed=True` option of the `mpatches.Polygon` is not doing this.\n\nAgain, thanks to anyone who looks into this.\n",
      "issue_closed_at": "2015-03-04T22:48:51Z",
      "base_commit": "6387fcaaeeff4777373429a54d79ca270cbd2522",
      "changes": [
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "print_figure_impl",
          "class_name": null,
          "code": "def print_figure_impl():\n            # write the PostScript headers\n            if isEPSF: print(\"%!PS-Adobe-3.0 EPSF-3.0\", file=fh)\n            else: print(\"%!PS-Adobe-3.0\", file=fh)\n            if title: print(\"%%Title: \"+title, file=fh)\n            print((\"%%Creator: matplotlib version \"\n                         +__version__+\", http://matplotlib.org/\"), file=fh)\n            print(\"%%CreationDate: \"+time.ctime(time.time()), file=fh)\n            print(\"%%Orientation: \" + orientation, file=fh)\n            if not isEPSF: print(\"%%DocumentPaperSizes: \"+papertype, file=fh)\n            print(\"%%%%BoundingBox: %d %d %d %d\" % bbox, file=fh)\n            if not isEPSF: print(\"%%Pages: 1\", file=fh)\n            print(\"%%EndComments\", file=fh)\n\n            Ndict = len(psDefs)\n            print(\"%%BeginProlog\", file=fh)\n            if not rcParams['ps.useafm']:\n                Ndict += len(ps_renderer.used_characters)\n            print(\"/mpldict %d dict def\"%Ndict, file=fh)\n            print(\"mpldict begin\", file=fh)\n            for d in psDefs:\n                d=d.strip()\n                for l in d.split('\\n'):\n                    print(l.strip(), file=fh)\n            if not rcParams['ps.useafm']:\n                for font_filename, chars in six.itervalues(ps_renderer.used_characters):\n                    if len(chars):\n                        font = FT2Font(font_filename)\n                        cmap = font.get_charmap()\n                        glyph_ids = []\n                        for c in chars:\n                            gind = cmap.get(c) or 0\n                            glyph_ids.append(gind)\n\n                        fonttype = rcParams['ps.fonttype']\n\n                        # Can not use more than 255 characters from a\n                        # single font for Type 3\n                        if len(glyph_ids) > 255:\n                            fonttype = 42\n\n                        # The ttf to ps (subsetting) support doesn't work for\n                        # OpenType fonts that are Postscript inside (like the\n                        # STIX fonts).  This will simply turn that off to avoid\n                        # errors.\n                        if is_opentype_cff_font(font_filename):\n                            raise RuntimeError(\"OpenType CFF fonts can not be saved using the internal Postscript backend at this time.\\nConsider using the Cairo backend.\")\n                        else:\n                            fh.flush()\n                            convert_ttf_to_ps(\n                                font_filename.encode(sys.getfilesystemencoding()),\n                                fh, fonttype, glyph_ids)\n            print(\"end\", file=fh)\n            print(\"%%EndProlog\", file=fh)\n\n            if not isEPSF: print(\"%%Page: 1 1\", file=fh)\n            print(\"mpldict begin\", file=fh)\n            #print >>fh, \"gsave\"\n            print(\"%s translate\"%_nums_to_str(xo, yo), file=fh)\n            if rotation: print(\"%d rotate\"%rotation, file=fh)\n            print(\"%s clipbox\"%_nums_to_str(width*72, height*72, 0, 0), file=fh)\n\n            # write the figure\n            content = self._pswriter.getvalue()\n            if not isinstance(content, six.text_type):\n                content = content.decode('ascii')\n            print(content, file=fh)\n\n            # write the trailer\n            #print >>fh, \"grestore\"\n            print(\"end\", file=fh)\n            print(\"showpage\", file=fh)\n            if not isEPSF: print(\"%%EOF\", file=fh)\n            fh.flush()"
        },
        {
          "file": "lib/matplotlib/backends/backend_ps.py",
          "type": "function",
          "name": "write",
          "class_name": "NullWriter",
          "code": "def write(self, *kl, **kwargs):\n                    pass"
        },
        {
          "file": "lib/matplotlib/backends/backend_svg.py",
          "type": "function",
          "name": "_write_default_style",
          "class_name": "RendererSVG",
          "code": "def _write_default_style(self):\n        writer = self.writer\n        default_style = generate_css({\n            'stroke-linejoin': 'round',\n            'stroke-linecap': 'butt'})\n        writer.start('defs')\n        writer.start('style', type='text/css')\n        writer.data('*{%s}\\n' % default_style)\n        writer.end('style')\n        writer.end('defs')"
        }
      ]
    },
    {
      "pr_number": 17569,
      "pr_title": "Delay \\usepackage{textcomp} until after the custom tex preamble.",
      "pr_body": "... and only load textcomp if it hasn't already been loaded.  This\r\navoids option clashes in case the custom preamble loads textcomp with\r\nother options.\r\n\r\nCloses #9118.  Simpler than to #17563, I guess.\r\n\r\n## PR Summary\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) compliant\r\n- [ ] New features are documented, with examples if plot related\r\n- [ ] Documentation is sphinx and numpydoc compliant\r\n- [ ] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [ ] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
      "issue_id": 9118,
      "issue_title": "make TeXManager more user-configurable",
      "issue_body": "### Bug report\r\n\r\nMatplotlib cannot be used with the LaTeX font NewTX. Matplotlib loads the LaTeX `texcomp` package, without any options, while the `newtxtext` package requires the `full` option.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\nfrom matplotlib import rc\r\n\r\nrc('text', usetex=True)\r\nrc('mathtext', fontset='custom')\r\nrc('text.latex', preamble=r'\\usepackage{newtxtext}\\usepackage{newtxmath}')\r\n\r\nx = np.linspace(0, 10, 100)\r\nplt.plot(x, np.sin(x))\r\nplt.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\n! LaTeX Error: Option clash for package textcomp.\r\n```\r\n\r\n**Expected outcome**\r\n\r\nTo be able to use the newtxtex package. Maybe to have the ability to override the default packages or set their options in some ways.\r\n\r\n**Matplotlib version**\r\n\r\n  * Operating System: Debian GNU/Linux 8\r\n  * Matplotlib Version: 2.0\r\n  * Python Version: 3.5\r\n\r\nInstalled matplotlib as Debian package.\r\n",
      "issue_closed_at": "2020-06-17T16:23:11Z",
      "base_commit": "89fa0e43b63512c595387a37bdfd37196ced69be",
      "changes": [
        {
          "file": "lib/matplotlib/texmanager.py",
          "type": "function",
          "name": "get_font_config",
          "class_name": "TexManager",
          "code": "def get_font_config(self):\n        ff = rcParams['font.family']\n        if len(ff) == 1 and ff[0].lower() in self.font_families:\n            self.font_family = ff[0].lower()\n        else:\n            _log.info('font.family must be one of (%s) when text.usetex is '\n                      'True. serif will be used by default.',\n                      ', '.join(self.font_families))\n            self.font_family = 'serif'\n\n        fontconfig = [self.font_family]\n        for font_family in self.font_families:\n            for font in rcParams['font.' + font_family]:\n                if font.lower() in self.font_info:\n                    self._fonts[font_family] = self.font_info[font.lower()]\n                    _log.debug('family: %s, font: %s, info: %s',\n                               font_family, font, self.font_info[font.lower()])\n                    break\n                else:\n                    _log.debug('%s font is not compatible with usetex.', font)\n            else:\n                _log.info('No LaTeX-compatible font found for the %s font '\n                          'family in rcParams. Using default.', font_family)\n                self._fonts[font_family] = self.font_info[font_family]\n            fontconfig.append(self._fonts[font_family][0])\n        # Add a hash of the latex preamble to fontconfig so that the\n        # correct png is selected for strings rendered with same font and dpi\n        # even if the latex preamble changes within the session\n        preamble_bytes = self.get_custom_preamble().encode('utf-8')\n        fontconfig.append(hashlib.md5(preamble_bytes).hexdigest())\n\n        # The following packages and commands need to be included in the latex\n        # file's preamble:\n        cmd = [self._fonts['serif'][1],\n               self._fonts['sans-serif'][1],\n               self._fonts['monospace'][1]]\n        if self.font_family == 'cursive':\n            cmd.append(self._fonts['cursive'][1])\n        self._font_preamble = '\\n'.join(\n            [r'\\usepackage{type1cm}', *cmd, r'\\usepackage{textcomp}'])\n\n        return ''.join(fontconfig)"
        },
        {
          "file": "lib/matplotlib/texmanager.py",
          "type": "function",
          "name": "_get_preamble",
          "class_name": "TexManager",
          "code": "def _get_preamble(self):\n        return \"\\n\".join([\n            r\"\\documentclass{article}\",\n            # Pass-through \\mathdefault, which is used in non-usetex mode to\n            # use the default text font but was historically suppressed in\n            # usetex mode.\n            r\"\\newcommand{\\mathdefault}[1]{#1}\",\n            self._font_preamble,\n            r\"\\usepackage[utf8]{inputenc}\",\n            r\"\\DeclareUnicodeCharacter{2212}{\\ensuremath{-}}\",\n            # Needs to come early so that the custom preamble can change the\n            # geometry, e.g. in convert_psfrags.\n            r\"\\usepackage[papersize=72in,body=70in,margin=1in]{geometry}\",\n            self.get_custom_preamble(),\n        ])"
        }
      ]
    },
    {
      "pr_number": 12678,
      "pr_title": "FIX: properly set tz for YearLocator",
      "pr_body": "## PR Summary\r\n\r\n`dates.YearLocator` was not always being called with the timezone info.  Now it is...\r\n\r\nCloses #12675:\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport pytz\r\nfrom datetime import datetime, timedelta, timezone\r\n\r\nx = [datetime(2010, 1, 1).astimezone(pytz.timezone('America/New_York')) + timedelta(i) for i in range(2000)]\r\nplt.plot(x, [0] * len(x));\r\nplt.show()\r\n```\r\n\r\nNow has the first tick at 2010 instead of 2009....\r\n\r\n## PR Checklist\r\n\r\n- [ ] Has Pytest style unit tests\r\n- [ ] Code is [Flake 8](http://flake8.pycqa.org/en/latest/) compliant\r\n- [ ] New features are documented, with examples if plot related\r\n- [ ] Documentation is sphinx and numpydoc compliant\r\n- [ ] Added an entry to doc/users/next_whats_new/ if major new feature (follow instructions in README.rst there)\r\n- [ ] Documented in doc/api/api_changes.rst if API changed in a backward-incompatible way\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
      "issue_id": 12675,
      "issue_title": "Off-by-one bug in annual axis labels when localized time crosses year boundary",
      "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\nWhen plotting dates with time zone information on the x-axis, under certain circumstances, the wrong year will be displayed as the label.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport pytz\r\nfrom datetime import datetime, timedelta\r\n\r\nx = [datetime(2010, 1, 1, tzinfo=pytz.timezone('America/New_York')) + timedelta(i) for i in range(2000)]\r\nplt.plot(x, [0] * len(x));\r\n```\r\n\r\n**Actual outcome**\r\n\r\nThe first label on the plot is 2009.\r\n\r\n**Expected outcome**\r\n\r\nThe first label on the plot is 2010.\r\n\r\n**Matplotlib version**\r\n\r\nBuilt this morning from master. Also confirmed the issue on 3.0.0 and confirmed that 2.2.2 does not have the issue (or at least not with the reproduction steps I found.)\r\n\r\n  * Operating system: Debian 7.11\r\n  * Matplotlib version: from source\r\n  * Matplotlib backend module://ipykernel.pylab.backend_inline\r\n  * Python version: 3.7.0\r\n  * Jupyter version (if applicable): 4.4.0\r\n  * Other libraries: pytz 2018.5\r\n\r\nI built matplotlib from source to verify that this bug affects master. The checks against 3.0.0 and 2.2.2 were done on a conda install of those versions and may have had different versions of pytz and jupyter.",
      "issue_closed_at": "2019-01-28T20:12:28Z",
      "base_commit": "c22847b0ce518be6b58908f5dacf946c92855168",
      "changes": [
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ConciseDateConverter",
          "code": "def __init__(self, formats=None, zero_formats=None, offset_formats=None,\n                show_offset=True):\n        self._formats = formats\n        self._zero_formats = zero_formats\n        self._offset_formats = offset_formats\n        self._show_offset = show_offset\n        super().__init__()"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "get_locator",
          "class_name": "AutoDateLocator",
          "code": "def get_locator(self, dmin, dmax):\n        'Pick the best locator based on a distance.'\n        delta = relativedelta(dmax, dmin)\n        tdelta = dmax - dmin\n\n        # take absolute difference\n        if dmin > dmax:\n            delta = -delta\n            tdelta = -tdelta\n\n        # The following uses a mix of calls to relativedelta and timedelta\n        # methods because there is incomplete overlap in the functionality of\n        # these similar functions, and it's best to avoid doing our own math\n        # whenever possible.\n        numYears = float(delta.years)\n        numMonths = numYears * MONTHS_PER_YEAR + delta.months\n        numDays = tdelta.days   # Avoids estimates of days/month, days/year\n        numHours = numDays * HOURS_PER_DAY + delta.hours\n        numMinutes = numHours * MIN_PER_HOUR + delta.minutes\n        numSeconds = np.floor(tdelta.total_seconds())\n        numMicroseconds = np.floor(tdelta.total_seconds() * 1e6)\n\n        nums = [numYears, numMonths, numDays, numHours, numMinutes,\n                numSeconds, numMicroseconds]\n\n        use_rrule_locator = [True] * 6 + [False]\n\n        # Default setting of bymonth, etc. to pass to rrule\n        # [unused (for year), bymonth, bymonthday, byhour, byminute,\n        #  bysecond, unused (for microseconds)]\n        byranges = [None, 1, 1, 0, 0, 0, None]\n\n        # Loop over all the frequencies and try to find one that gives at\n        # least a minticks tick positions.  Once this is found, look for\n        # an interval from an list specific to that frequency that gives no\n        # more than maxticks tick positions. Also, set up some ranges\n        # (bymonth, etc.) as appropriate to be passed to rrulewrapper.\n        for i, (freq, num) in enumerate(zip(self._freqs, nums)):\n            # If this particular frequency doesn't give enough ticks, continue\n            if num < self.minticks:\n                # Since we're not using this particular frequency, set\n                # the corresponding by_ to None so the rrule can act as\n                # appropriate\n                byranges[i] = None\n                continue\n\n            # Find the first available interval that doesn't give too many\n            # ticks\n            for interval in self.intervald[freq]:\n                if num <= interval * (self.maxticks[freq] - 1):\n                    break\n            else:\n                # We went through the whole loop without breaking, default to\n                # the last interval in the list and raise a warning\n                cbook._warn_external('AutoDateLocator was unable to pick an '\n                                     'appropriate interval for this date '\n                                     'range. It may be necessary to add an '\n                                     'interval value to the '\n                                     'AutoDateLocator\\'s intervald '\n                                     'dictionary. Defaulting to {0}.'\n                                     .format(interval))\n\n            # Set some parameters as appropriate\n            self._freq = freq\n\n            if self._byranges[i] and self.interval_multiples:\n                byranges[i] = self._byranges[i][::interval]\n                if i in (DAILY, WEEKLY):\n                    if interval == 14:\n                        # just make first and 15th.  Avoids 30th.\n                        byranges[i] = [1, 15]\n                    elif interval == 7:\n                        byranges[i] = [1, 8, 15, 22]\n\n                interval = 1\n            else:\n                byranges[i] = self._byranges[i]\n            break\n        else:\n            raise ValueError('No sensible date limit could be found in the '\n                             'AutoDateLocator.')\n\n        if (freq == YEARLY) and self.interval_multiples:\n            locator = YearLocator(interval)\n        elif use_rrule_locator[i]:\n            _, bymonth, bymonthday, byhour, byminute, bysecond, _ = byranges\n            rrule = rrulewrapper(self._freq, interval=interval,\n                                 dtstart=dmin, until=dmax,\n                                 bymonth=bymonth, bymonthday=bymonthday,\n                                 byhour=byhour, byminute=byminute,\n                                 bysecond=bysecond)\n\n            locator = RRuleLocator(rrule, self.tz)\n        else:\n            locator = MicrosecondLocator(interval, tz=self.tz)\n            if dmin.year > 20 and interval < 1000:\n                _log.warning('Plotting microsecond time intervals is not well '\n                             'supported. Please see the MicrosecondLocator '\n                             'documentation for details.')\n\n        locator.set_axis(self.axis)\n\n        if self.axis is not None:\n            locator.set_view_interval(*self.axis.get_view_interval())\n            locator.set_data_interval(*self.axis.get_data_interval())\n        return locator"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ConciseDateConverter",
          "code": "def __init__(self, formats=None, zero_formats=None, offset_formats=None,\n                show_offset=True):\n        self._formats = formats\n        self._zero_formats = zero_formats\n        self._offset_formats = offset_formats\n        self._show_offset = show_offset\n        super().__init__()"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "tick_values",
          "class_name": "MicrosecondLocator",
          "code": "def tick_values(self, vmin, vmax):\n        nmin, nmax = date2num((vmin, vmax))\n        nmin *= MUSECONDS_PER_DAY\n        nmax *= MUSECONDS_PER_DAY\n        ticks = self._wrapped_locator.tick_values(nmin, nmax)\n        ticks = [tick / MUSECONDS_PER_DAY for tick in ticks]\n        return ticks"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "autoscale",
          "class_name": "YearLocator",
          "code": "def autoscale(self):\n        \"\"\"\n        Set the view limits to include the data range.\n        \"\"\"\n        dmin, dmax = self.datalim_to_dt()\n\n        ymin = self.base.le(dmin.year)\n        ymax = self.base.ge(dmax.year)\n        vmin = dmin.replace(year=ymin, **self.replaced)\n        vmax = dmax.replace(year=ymax, **self.replaced)\n\n        vmin = date2num(vmin)\n        vmax = date2num(vmax)\n        return self.nonsingular(vmin, vmax)"
        }
      ]
    }
  ]
}