{
  "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",
  "similar_bug_items": [
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 20047,
      "pr_title": "Add labels parameter to set_ticks()",
      "pr_body": "## PR Summary\r\n\r\nAs proposed in https://github.com/matplotlib/matplotlib/issues/18848#issuecomment-720154479 and discussed on today's dev call.\r\n\r\nCloses #18848, #19016.\r\n\r\n**kwargs:** I've added kwargs for text properties. While this makes the function do a lot (borderline of too much), it is necessary if we want a full replacement of `set_ticklabels(labels, **text_properties)`.\r\nOne can argue that this is actually too much, in which case we'd have to advertise\r\n~~~\r\n# Replace\r\nset_ticks(locs)\r\nset_ticklabels(labels, **text_properties)\r\n# by\r\nset_ticks(locs, labels)\r\ntick_params(**text_properties)\r\n~~~\r\n\r\n\r\n\r\n**Return value:** `set_ticks()` has an undocumented return value (list of the Tick instances). While `plt.xticks()` returns a tuple `(locs, labels)` (list of Ticks and list of Texts), I don't want to do this here. First, I don't think the setter should return anything at all (but that's how it is for now and deprecating that is probably not worth it). Second, since labels are optional, I'm not even sure if we would want to give the texts only if `labels` is not None.\r\n\r\n\r\n\r\n",
      "issue_id": 18848,
      "issue_title": "Setting xticklabels causes warning related to FixedFormatter",
      "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\nI get a warning related to `FixedFormatter` even though I am not setting a formatter.\r\n\r\nThanks to user Rational-IM [on StackOverflow](https://stackoverflow.com/questions/63723514/userwarning-fixedformatter-should-only-be-used-together-with-fixedlocator) and the good folks at [pandas ](https://github.com/pandas-dev/pandas/issues/35684) for isolating and reproducing the bug.\r\n\r\n**Code for reproduction**\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\n\r\nfig, ax = plt.subplots()\r\nax.scatter(0.5, 0.5)\r\nax.set_xticklabels(['0', r'$T/2$', r'$T$'])\r\n\r\n# Optional line\r\nax.xaxis.set_major_locator(FixedLocator([0, 0.5, 1]))\r\n\r\nNone\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<ipython-input-3-52a3699e4278>:5: UserWarning: FixedFormatter should only be used together with FixedLocator\r\n  ax.set_xticklabels(['0', r'$T/2$', r'$T$'])\r\n```\r\n\r\n**Expected outcome**\r\n\r\nWorks on 3.2.2 with no warning.\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: Win 10\r\n  * Matplotlib version: 3.3.2\r\n  * Matplotlib backend: module://ipykernel.pylab.backend_inline\r\n  * Python version: 3.8.5\r\n  * Jupyter version (if applicable): 6.1.4\r\n  * Other libraries: none\r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\nInstalled from 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": "2021-05-14T03:50:35Z",
      "base_commit": "19aac39fe9e70decc72ce5b3e2261b5c602a8c78",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_secondary_axes.py",
          "type": "line",
          "name": "line 4",
          "code": "import matplotlib.docstring as docstring\nimport matplotlib.ticker as mticker\nfrom matplotlib.axes._base import _AxesBase, _TransformedBoundsLocator\n\n\nclass SecondaryAxis(_AxesBase):"
        },
        {
          "file": "lib/matplotlib/axes/_secondary_axes.py",
          "type": "function",
          "name": "apply_aspect",
          "class_name": "SecondaryAxis",
          "code": "def apply_aspect(self, position=None):\n        # docstring inherited.\n        self._set_lims()\n        super().apply_aspect(position)"
        },
        {
          "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            # Passing [] as a list of ticklabels is often used as a way to\n            # remove all tick labels, so only error for > 0 ticklabels\n            if len(locator.locs) != len(ticklabels) and len(ticklabels) != 0:\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"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "_set_ticklabels",
          "class_name": "Axis",
          "code": "def _set_ticklabels(self, labels, fontdict=None, minor=False, **kwargs):\n        \"\"\"\n        Set this Axis' labels with list of string 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        labels : list of str\n            The label texts.\n\n        fontdict : dict, optional\n            A dictionary controlling the appearance of the ticklabels.\n            The default *fontdict* is::\n\n               {'fontsize': rcParams['axes.titlesize'],\n                'fontweight': rcParams['axes.titleweight'],\n                'verticalalignment': 'baseline',\n                'horizontalalignment': loc}\n\n        minor : bool, default: False\n            Whether to set the minor ticklabels rather than the major ones.\n\n        Returns\n        -------\n        list of `~.Text`\n            The labels.\n\n        Other Parameters\n        ----------------\n        **kwargs : `~.text.Text` properties.\n        \"\"\"\n        if fontdict is not None:\n            kwargs.update(fontdict)\n        return self.set_ticklabels(labels, minor=minor, **kwargs)"
        },
        {
          "file": "lib/matplotlib/axis.py",
          "type": "function",
          "name": "set_ticks",
          "class_name": "Axis",
          "code": "def set_ticks(self, ticks, *, minor=False):\n        \"\"\"\n        Set this Axis' tick locations.\n\n        If necessary, the view limits of the Axis are expanded so that all\n        given ticks are visible.\n\n        Parameters\n        ----------\n        ticks : list of floats\n            List of tick locations.\n        minor : bool, default: False\n            If ``False``, set the major ticks; if ``True``, the minor ticks.\n\n        Notes\n        -----\n        The mandatory expansion of the view limits is an intentional design\n        choice to prevent the surprise of a non-visible tick. If you need\n        other limits, you should set the limits explicitly after setting the\n        ticks.\n        \"\"\"\n        # XXX if the user changes units, the information will be lost here\n        ticks = self.convert_units(ticks)\n        if self is self.axes.xaxis:\n            shared = [\n                ax.xaxis\n                for ax in self.axes.get_shared_x_axes().get_siblings(self.axes)\n            ]\n        elif self is self.axes.yaxis:\n            shared = [\n                ax.yaxis\n                for ax in self.axes.get_shared_y_axes().get_siblings(self.axes)\n            ]\n        elif hasattr(self.axes, \"zaxis\") and self is self.axes.zaxis:\n            shared = [\n                ax.zaxis\n                for ax in self.axes._shared_z_axes.get_siblings(self.axes)\n            ]\n        else:\n            shared = [self]\n        for axis in shared:\n            if len(ticks) > 1:\n                xleft, xright = axis.get_view_interval()\n                if xright > xleft:\n                    axis.set_view_interval(min(ticks), max(ticks))\n                else:\n                    axis.set_view_interval(max(ticks), min(ticks))\n        self.axes.stale = True\n        if minor:\n            self.set_minor_locator(mticker.FixedLocator(ticks))\n            return self.get_minor_ticks(len(ticks))\n        else:\n            self.set_major_locator(mticker.FixedLocator(ticks))\n            return self.get_major_ticks(len(ticks))"
        }
      ]
    },
    {
      "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_id": 6142,
      "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_closed_at": "2016-05-04T00:26:00Z",
      "base_commit": "22a7b955a0b9dc4dea8adf155041498dd355e4df",
      "changes": [
        {
          "file": "lib/matplotlib/animation.py",
          "type": "line",
          "name": "line 36",
          "code": "import abc\nimport contextlib\nimport tempfile\nfrom matplotlib.cbook import iterable, is_string_like\nfrom matplotlib.compat import subprocess\nfrom matplotlib import verbose\nfrom matplotlib import rcParams, rcParamsDefault\n\n# Process creation flag for subprocess to prevent it raising a terminal\n# window. See for example:"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "class",
          "name": "MovieWriter",
          "code": "class MovieWriter(AbstractMovieWriter):\n    '''\n    Base class for writing movies. Fundamentally, what a MovieWriter does\n    is provide is a way to grab frames by calling grab_frame(). setup()\n    is called to start the process and finish() is called afterwards.\n    This class is set up to provide for writing movie frame data to a pipe.\n    saving() is provided as a context manager to facilitate this process as::\n\n      with moviewriter.saving(fig, outfile='myfile.mp4', dpi=100):\n          # Iterate over frames\n          moviewriter.grab_frame(**savefig_kwargs)\n\n    The use of the context manager ensures that setup and cleanup are\n    performed as necessary.\n\n    frame_format: string\n        The format used in writing frame data, defaults to 'rgba'\n    '''\n    def __init__(self, fps=5, codec=None, bitrate=None, extra_args=None,\n                 metadata=None):\n        '''\n        Construct a new MovieWriter object.\n\n        fps: int\n            Framerate for movie.\n        codec: string or None, optional\n            The codec to use. If None (the default) the setting in the\n            rcParam `animation.codec` is used.\n        bitrate: int or None, optional\n            The bitrate for the saved movie file, which is one way to control\n            the output file size and quality. The default value is None,\n            which uses the value stored in the rcParam `animation.bitrate`.\n            A value of -1 implies that the bitrate should be determined\n            automatically by the underlying utility.\n        extra_args: list of strings or None\n            A list of extra string arguments to be passed to the underlying\n            movie utiltiy. The default is None, which passes the additional\n            argurments in the 'animation.extra_args' rcParam.\n        metadata: dict of string:string or None\n            A dictionary of keys and values for metadata to include in the\n            output file. Some keys that may be of use include:\n            title, artist, genre, subject, copyright, srcform, comment.\n        '''\n        self.fps = fps\n        self.frame_format = 'rgba'\n\n        if codec is None:\n            self.codec = rcParams['animation.codec']\n        else:\n            self.codec = codec\n\n        if bitrate is None:\n            self.bitrate = rcParams['animation.bitrate']\n        else:\n            self.bitrate = bitrate\n\n        if extra_args is None:\n            self.extra_args = list(rcParams[self.args_key])\n        else:\n            self.extra_args = extra_args\n\n        if metadata is None:\n            self.metadata = dict()\n        else:\n            self.metadata = metadata\n\n    @property\n    def frame_size(self):\n        'A tuple (width,height) in pixels of a movie frame.'\n        width_inches, height_inches = self.fig.get_size_inches()\n        return width_inches * self.dpi, height_inches * self.dpi\n\n    def setup(self, fig, outfile, dpi):\n        '''\n        Perform setup for writing the movie file.\n\n        fig: `matplotlib.Figure` instance\n            The figure object that contains the information for frames\n        outfile: string\n            The filename of the resulting movie file\n        dpi: int\n            The DPI (or resolution) for the file.  This controls the size\n            in pixels of the resulting movie file.\n        '''\n        self.outfile = outfile\n        self.fig = fig\n        self.dpi = dpi\n\n        # Run here so that grab_frame() can write the data to a pipe. This\n        # eliminates the need for temp files.\n        self._run()\n\n    def _run(self):\n        # Uses subprocess to call the program for assembling frames into a\n        # movie file.  *args* returns the sequence of command line arguments\n        # from a few configuration options.\n        command = self._args()\n        if verbose.ge('debug'):\n            output = sys.stdout\n        else:\n            output = subprocess.PIPE\n        verbose.report('MovieWriter.run: running command: %s' %\n                       ' '.join(command))\n        self._proc = subprocess.Popen(command, shell=False,\n                                      stdout=output, stderr=output,\n                                      stdin=subprocess.PIPE,\n                                      creationflags=subprocess_creation_flags)\n\n    def finish(self):\n        'Finish any processing for writing the movie.'\n        self.cleanup()\n\n    def grab_frame(self, **savefig_kwargs):\n        '''\n        Grab the image information from the figure and save as a movie frame.\n        All keyword arguments in savefig_kwargs are passed on to the 'savefig'\n        command that saves the figure.\n        '''\n        verbose.report('MovieWriter.grab_frame: Grabbing frame.',\n                       level='debug')\n        try:\n            # Tell the figure to save its data to the sink, using the\n            # frame format and dpi.\n            self.fig.savefig(self._frame_sink(), format=self.frame_format,\n                             dpi=self.dpi, **savefig_kwargs)\n        except RuntimeError:\n            out, err = self._proc.communicate()\n            verbose.report('MovieWriter -- Error '\n                           'running proc:\\n%s\\n%s' % (out,\n                                                      err), level='helpful')\n            raise\n\n    def _frame_sink(self):\n        'Returns the place to which frames should be written.'\n        return self._proc.stdin\n\n    def _args(self):\n        'Assemble list of utility-specific command-line arguments.'\n        return NotImplementedError(\"args needs to be implemented by subclass.\")\n\n    def cleanup(self):\n        'Clean-up and collect the process used to write the movie file.'\n        out, err = self._proc.communicate()\n        self._frame_sink().close()\n        verbose.report('MovieWriter -- '\n                       'Command stdout:\\n%s' % out, level='debug')\n        verbose.report('MovieWriter -- '\n                       'Command stderr:\\n%s' % err, level='debug')\n\n    @classmethod\n    def bin_path(cls):\n        '''\n        Returns the binary path to the commandline tool used by a specific\n        subclass. This is a class method so that the tool can be looked for\n        before making a particular MovieWriter subclass available.\n        '''\n        return rcParams[cls.exec_key]\n\n    @classmethod\n    def isAvailable(cls):\n        '''\n        Check to see if a MovieWriter subclass is actually available by\n        running the commandline tool.\n        '''\n        bin_path = cls.bin_path()\n        if not bin_path:\n            return False\n        try:\n            p = subprocess.Popen(bin_path,\n                             shell=False,\n                             stdout=subprocess.PIPE,\n                             stderr=subprocess.PIPE,\n                             creationflags=subprocess_creation_flags)\n            p.communicate()\n            return True\n        except OSError:\n            return False"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "__init__",
          "class_name": "FuncAnimation",
          "code": "def __init__(self, fig, func, frames=None, init_func=None, fargs=None,\n                 save_count=None, **kwargs):\n        if fargs:\n            self._args = fargs\n        else:\n            self._args = ()\n        self._func = func\n\n        # Amount of framedata to keep around for saving movies. This is only\n        # used if we don't know how many frames there will be: in the case\n        # of no generator or in the case of a callable.\n        self.save_count = save_count\n\n        # Set up a function that creates a new iterable when needed. If nothing\n        # is passed in for frames, just use itertools.count, which will just\n        # keep counting from 0. A callable passed in for frames is assumed to\n        # be a generator. An iterable will be used as is, and anything else\n        # will be treated as a number of frames.\n        if frames is None:\n            self._iter_gen = itertools.count\n        elif six.callable(frames):\n            self._iter_gen = frames\n        elif iterable(frames):\n            self._iter_gen = lambda: iter(frames)\n            if hasattr(frames, '__len__'):\n                self.save_count = len(frames)\n        else:\n            self._iter_gen = lambda: xrange(frames).__iter__()\n            self.save_count = frames\n\n        # If we're passed in and using the default, set it to 100.\n        if self.save_count is None:\n            self.save_count = 100\n\n        self._init_func = init_func\n\n        # Needs to be initialized so the draw functions work without checking\n        self._save_seq = []\n\n        TimedAnimation.__init__(self, fig, **kwargs)\n\n        # Need to reset the saved seq, since right now it will contain data\n        # for a single frame from init, which is not what we want.\n        self._save_seq = []"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "isAvailable",
          "class_name": "ImageMagickBase",
          "code": "def isAvailable(cls):\n        '''\n        Check to see if a ImageMagickWriter is actually available\n\n        Done by first checking the windows registry (if applicable) and then\n        running the commandline tool.\n        '''\n        bin_path = cls.bin_path()\n        if bin_path == \"convert\":\n            cls._init_from_registry()\n        return super(ImageMagickBase, cls).isAvailable()"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "class",
          "name": "FFMpegBase",
          "code": "class FFMpegBase(object):\n    exec_key = 'animation.ffmpeg_path'\n    args_key = 'animation.ffmpeg_args'\n\n    @property\n    def output_args(self):\n        # The %dk adds 'k' as a suffix so that ffmpeg treats our bitrate as in\n        # kbps\n        args = ['-vcodec', self.codec]\n        # For h264, the default format is yuv444p, which is not compatible\n        # with quicktime (and others). Specifying yuv420p fixes playback on\n        # iOS,as well as HTML5 video in firefox and safari (on both Win and\n        # OSX). Also fixes internet explorer. This is as of 2015/10/29.\n        if self.codec == 'h264' and '-pix_fmt' not in self.extra_args:\n            args.extend(['-pix_fmt', 'yuv420p'])\n        if self.bitrate > 0:\n            args.extend(['-b', '%dk' % self.bitrate])\n        if self.extra_args:\n            args.extend(self.extra_args)\n        for k, v in six.iteritems(self.metadata):\n            args.extend(['-metadata', '%s=%s' % (k, v)])\n\n        return args + ['-y', self.outfile]"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "output_args",
          "class_name": "ImageMagickBase",
          "code": "def output_args(self):\n        return [self.outfile]"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "save",
          "class_name": "Animation",
          "code": "def save(self, filename, writer=None, fps=None, dpi=None, codec=None,\n             bitrate=None, extra_args=None, metadata=None, extra_anim=None,\n             savefig_kwargs=None):\n        '''\n        Saves a movie file by drawing every frame.\n\n        *filename* is the output filename, e.g., :file:`mymovie.mp4`\n\n        *writer* is either an instance of :class:`AbstractMovieWriter` or\n        a string key that identifies a class to use, such as 'ffmpeg' or\n        'mencoder'.  If nothing is passed, the value of the rcparam\n        `animation.writer` is used.\n\n        *fps* is the frames per second in the movie. Defaults to None,\n        which will use the animation's specified interval to set the frames\n        per second.\n\n        *dpi* controls the dots per inch for the movie frames. This combined\n        with the figure's size in inches controls the size of the movie.\n\n        *codec* is the video codec to be used. Not all codecs are supported\n        by a given :class:`MovieWriter`. If none is given, this defaults to the\n        value specified by the rcparam `animation.codec`.\n\n        *bitrate* specifies the amount of bits used per second in the\n        compressed movie, in kilobits per second. A higher number means a\n        higher quality movie, but at the cost of increased file size. If no\n        value is given, this defaults to the value given by the rcparam\n        `animation.bitrate`.\n\n        *extra_args* is a list of extra string arguments to be passed to the\n        underlying movie utiltiy. The default is None, which passes the\n        additional argurments in the 'animation.extra_args' rcParam.\n\n        *metadata* is a dictionary of keys and values for metadata to include\n        in the output file. Some keys that may be of use include:\n        title, artist, genre, subject, copyright, srcform, comment.\n\n        *extra_anim* is a list of additional `Animation` objects that should\n        be included in the saved movie file. These need to be from the same\n        `matplotlib.Figure` instance. Also, animation frames will just be\n        simply combined, so there should be a 1:1 correspondence between\n        the frames from the different animations.\n\n        *savefig_kwargs* is a dictionary containing keyword arguments to be\n        passed on to the 'savefig' command which is called repeatedly to save\n        the individual frames. This can be used to set tight bounding boxes,\n        for example.\n        '''\n        if savefig_kwargs is None:\n            savefig_kwargs = {}\n\n        # FIXME: Using 'bbox_inches' doesn't currently work with\n        # writers that pipe the data to the command because this\n        # requires a fixed frame size (see Ryan May's reply in this\n        # thread: [1]). Thus we drop the 'bbox_inches' argument if it\n        # exists in savefig_kwargs.\n        #\n        # [1] (http://matplotlib.1069221.n5.nabble.com/\n        # Animation-class-let-save-accept-kwargs-which-\n        # are-passed-on-to-savefig-td39627.html)\n        #\n        if 'bbox_inches' in savefig_kwargs:\n            if not (writer in ['ffmpeg_file', 'mencoder_file'] or\n                    isinstance(writer,\n                               (FFMpegFileWriter, MencoderFileWriter))):\n                print(\"Warning: discarding the 'bbox_inches' argument in \"\n                      \"'savefig_kwargs' as it is only currently supported \"\n                      \"with the writers 'ffmpeg_file' and 'mencoder_file' \"\n                      \"(writer used: \"\n                      \"'{0}').\".format(\n                          writer if isinstance(writer, six.string_types)\n                          else writer.__class__.__name__))\n                savefig_kwargs.pop('bbox_inches')\n\n        # Need to disconnect the first draw callback, since we'll be doing\n        # draws. Otherwise, we'll end up starting the animation.\n        if self._first_draw_id is not None:\n            self._fig.canvas.mpl_disconnect(self._first_draw_id)\n            reconnect_first_draw = True\n        else:\n            reconnect_first_draw = False\n\n        if fps is None and hasattr(self, '_interval'):\n            # Convert interval in ms to frames per second\n            fps = 1000. / self._interval\n\n        # If the writer is None, use the rc param to find the name of the one\n        # to use\n        if writer is None:\n            writer = rcParams['animation.writer']\n\n        # Re-use the savefig DPI for ours if none is given\n        if dpi is None:\n            dpi = rcParams['savefig.dpi']\n        if dpi == 'figure':\n            dpi = self._fig.dpi\n\n        if codec is None:\n            codec = rcParams['animation.codec']\n\n        if bitrate is None:\n            bitrate = rcParams['animation.bitrate']\n\n        all_anim = [self]\n        if extra_anim is not None:\n            all_anim.extend(anim\n                            for anim\n                            in extra_anim if anim._fig is self._fig)\n\n        # If we have the name of a writer, instantiate an instance of the\n        # registered class.\n        if is_string_like(writer):\n            if writer in writers.avail:\n                writer = writers[writer](fps, codec, bitrate,\n                                         extra_args=extra_args,\n                                         metadata=metadata)\n            else:\n                import warnings\n                warnings.warn(\"MovieWriter %s unavailable\" % writer)\n\n                try:\n                    writer = writers[writers.list()[0]](fps, codec, bitrate,\n                                                        extra_args=extra_args,\n                                                        metadata=metadata)\n                except IndexError:\n                    raise ValueError(\"Cannot save animation: no writers are \"\n                                     \"available. Please install mencoder or \"\n                                     \"ffmpeg to save animations.\")\n\n        verbose.report('Animation.save using %s' % type(writer),\n                       level='helpful')\n        # Create a new sequence of frames for saved data. This is different\n        # from new_frame_seq() to give the ability to save 'live' generated\n        # frame information to be saved later.\n        # TODO: Right now, after closing the figure, saving a movie won't work\n        # since GUI widgets are gone. Either need to remove extra code to\n        # allow for this non-existant use case or find a way to make it work.\n        with writer.saving(self._fig, filename, dpi):\n            for anim in all_anim:\n                # Clear the initial frame\n                anim._init_draw()\n            for data in zip(*[a.new_saved_frame_seq()\n                              for a in all_anim]):\n                for anim, d in zip(all_anim, data):\n                    # TODO: Need to see if turning off blit is really necessary\n                    anim._draw_next_frame(d, blit=False)\n                writer.grab_frame(**savefig_kwargs)\n\n        # Reconnect signal for first draw if necessary\n        if reconnect_first_draw:\n            self._first_draw_id = self._fig.canvas.mpl_connect('draw_event',\n                                                               self._start)"
        },
        {
          "file": "lib/matplotlib/animation.py",
          "type": "function",
          "name": "to_html5_video",
          "class_name": "Animation",
          "code": "def to_html5_video(self):\n        r'''Returns animation as an HTML5 video tag.\n\n        This saves the animation as an h264 video, encoded in base64\n        directly into the HTML5 video tag. This respects the rc parameters\n        for the writer as well as the bitrate. This also makes use of the\n        ``interval`` to control the speed, and uses the ``repeat``\n        paramter to decide whether to loop.\n        '''\n        VIDEO_TAG = r'''<video {size} {options}>\n  <source type=\"video/mp4\" src=\"data:video/mp4;base64,{video}\">\n  Your browser does not support the video tag.\n</video>'''\n        # Cache the the rendering of the video as HTML\n        if not hasattr(self, '_base64_video'):\n            # First write the video to a tempfile. Set delete to False\n            # so we can re-open to read binary data.\n            with tempfile.NamedTemporaryFile(suffix='.m4v',\n                                             delete=False) as f:\n                # We create a writer manually so that we can get the\n                # appropriate size for the tag\n                Writer = writers[rcParams['animation.writer']]\n                writer = Writer(codec='h264',\n                                bitrate=rcParams['animation.bitrate'],\n                                fps=1000. / self._interval)\n                self.save(f.name, writer=writer)\n\n            # Now open and base64 encode\n            with open(f.name, 'rb') as video:\n                vid64 = encodebytes(video.read())\n                self._base64_video = vid64.decode('ascii')\n                self._video_size = 'width=\"{0}\" height=\"{1}\"'.format(\n                        *writer.frame_size)\n\n            # Now we can remove\n            os.remove(f.name)\n\n        # Default HTML5 options are to autoplay and to display video controls\n        options = ['controls', 'autoplay']\n\n        # If we're set to repeat, make it loop\n        if self.repeat:\n            options.append('loop')\n        return VIDEO_TAG.format(video=self._base64_video,\n                                size=self._video_size,\n                                options=' '.join(options))"
        },
        {
          "file": "lib/matplotlib/axes/_base.py",
          "type": "function",
          "name": "__init__",
          "class_name": "_AxesBase",
          "code": "def __init__(self, fig, rect,\n                 facecolor=None,  # defaults to rc axes.facecolor\n                 frameon=True,\n                 sharex=None,  # use Axes instance's xaxis info\n                 sharey=None,  # use Axes instance's yaxis info\n                 label='',\n                 xscale=None,\n                 yscale=None,\n                 axisbg=None,  # This will be removed eventually\n                 **kwargs\n                 ):\n        \"\"\"\n        Build an :class:`Axes` instance in\n        :class:`~matplotlib.figure.Figure` *fig* with\n        *rect=[left, bottom, width, height]* in\n        :class:`~matplotlib.figure.Figure` coordinates\n\n        Optional keyword arguments:\n\n          ================   =========================================\n          Keyword            Description\n          ================   =========================================\n          *adjustable*       [ 'box' | 'datalim' | 'box-forced']\n          *alpha*            float: the alpha transparency (can be None)\n          *anchor*           [ 'C', 'SW', 'S', 'SE', 'E', 'NE', 'N',\n                               'NW', 'W' ]\n          *aspect*           [ 'auto' | 'equal' | aspect_ratio ]\n          *autoscale_on*     [ *True* | *False* ] whether or not to\n                             autoscale the *viewlim*\n          *axisbelow*        [ *True* | *False* | 'line'] draw the grids\n                             and ticks below or above most other artists,\n                             or below lines but above patches\n          *cursor_props*     a (*float*, *color*) tuple\n          *figure*           a :class:`~matplotlib.figure.Figure`\n                             instance\n          *frame_on*         a boolean - draw the axes frame\n          *label*            the axes label\n          *navigate*         [ *True* | *False* ]\n          *navigate_mode*    [ 'PAN' | 'ZOOM' | None ] the navigation\n                             toolbar button status\n          *position*         [left, bottom, width, height] in\n                             class:`~matplotlib.figure.Figure` coords\n          *sharex*           an class:`~matplotlib.axes.Axes` instance\n                             to share the x-axis with\n          *sharey*           an class:`~matplotlib.axes.Axes` instance\n                             to share the y-axis with\n          *title*            the title string\n          *visible*          [ *True* | *False* ] whether the axes is\n                             visible\n          *xlabel*           the xlabel\n          *xlim*             (*xmin*, *xmax*) view limits\n          *xscale*           [%(scale)s]\n          *xticklabels*      sequence of strings\n          *xticks*           sequence of floats\n          *ylabel*           the ylabel strings\n          *ylim*             (*ymin*, *ymax*) view limits\n          *yscale*           [%(scale)s]\n          *yticklabels*      sequence of strings\n          *yticks*           sequence of floats\n          ================   =========================================\n        \"\"\" % {'scale': ' | '.join(\n            [repr(x) for x in mscale.get_scale_names()])}\n        martist.Artist.__init__(self)\n        if isinstance(rect, mtransforms.Bbox):\n            self._position = rect\n        else:\n            self._position = mtransforms.Bbox.from_bounds(*rect)\n        self._originalPosition = self._position.frozen()\n        # self.set_axes(self)\n        self.axes = self\n        self.set_aspect('auto')\n        self._adjustable = 'box'\n        self.set_anchor('C')\n        self._sharex = sharex\n        self._sharey = sharey\n        if sharex is not None:\n            self._shared_x_axes.join(self, sharex)\n            if sharex._adjustable == 'box':\n                sharex._adjustable = 'datalim'\n                # warnings.warn(\n                #    'shared axes: \"adjustable\" is being changed to \"datalim\"')\n            self._adjustable = 'datalim'\n        if sharey is not None:\n            self._shared_y_axes.join(self, sharey)\n            if sharey._adjustable == 'box':\n                sharey._adjustable = 'datalim'\n                # warnings.warn(\n                #    'shared axes: \"adjustable\" is being changed to \"datalim\"')\n            self._adjustable = 'datalim'\n        self.set_label(label)\n        self.set_figure(fig)\n\n        self.set_axes_locator(kwargs.get(\"axes_locator\", None))\n\n        self.spines = self._gen_axes_spines()\n\n        # this call may differ for non-sep axes, e.g., polar\n        self._init_axis()\n        if axisbg is not None and facecolor is not None:\n            raise TypeError('Both axisbg and facecolor are not None. '\n                            'These keywords are aliases, only one may be '\n                            'provided.')\n        if axisbg is not None:\n            cbook.warn_deprecated(\n                '2.0', name='axisbg', alternative='facecolor')\n            facecolor = axisbg\n        if facecolor is None:\n            facecolor = rcParams['axes.facecolor']\n        self._facecolor = facecolor\n        self._frameon = frameon\n        self._axisbelow = rcParams['axes.axisbelow']\n\n        self._rasterization_zorder = None\n\n        self._hold = rcParams['axes.hold']\n        self._connected = {}  # a dict from events to (id, func)\n        self.cla()\n        # funcs used to format x and y - fall back on major formatters\n        self.fmt_xdata = None\n        self.fmt_ydata = None\n\n        self.set_cursor_props((1, 'k'))  # set the cursor properties for axes\n\n        self._cachedRenderer = None\n        self.set_navigate(True)\n        self.set_navigate_mode(None)\n\n        if xscale:\n            self.set_xscale(xscale)\n        if yscale:\n            self.set_yscale(yscale)\n\n        if len(kwargs):\n            self.update(kwargs)\n\n        if self.xaxis is not None:\n            self._xcid = self.xaxis.callbacks.connect('units finalize',\n                                                      self.relim)\n\n        if self.yaxis is not None:\n            self._ycid = self.yaxis.callbacks.connect('units finalize',\n                                                      self.relim)\n        self.tick_params(top=rcParams['xtick.top'],\n                         bottom=rcParams['xtick.bottom'],\n                         left=rcParams['ytick.left'],\n                         right=rcParams['ytick.right'])"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "__init__",
          "class_name": "SubplotToolQt",
          "code": "def __init__(self, targetfig, parent):\n        UiSubplotTool.__init__(self, None)\n\n        self.targetfig = targetfig\n        self.parent = parent\n        self.donebutton.clicked.connect(self.close)\n        self.resetbutton.clicked.connect(self.reset)\n        self.tightlayout.clicked.connect(self.functight)\n\n        # constraints\n        self.sliderleft.valueChanged.connect(self.sliderright.setMinimum)\n        self.sliderright.valueChanged.connect(self.sliderleft.setMaximum)\n        self.sliderbottom.valueChanged.connect(self.slidertop.setMinimum)\n        self.slidertop.valueChanged.connect(self.sliderbottom.setMaximum)\n\n        self.defaults = {}\n        for attr in ('left', 'bottom', 'right', 'top', 'wspace', 'hspace', ):\n            self.defaults[attr] = getattr(self.targetfig.subplotpars, attr)\n            slider = getattr(self, 'slider' + attr)\n            slider.setMinimum(0)\n            slider.setMaximum(1000)\n            slider.setSingleStep(5)\n            slider.valueChanged.connect(getattr(self, 'func' + attr))\n\n        self._setSliderPositions()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "funcleft",
          "class_name": "SubplotToolQt",
          "code": "def funcleft(self, val):\n        if val == self.sliderright.value():\n            val -= 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(left=val)\n        self.leftvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "funcright",
          "class_name": "SubplotToolQt",
          "code": "def funcright(self, val):\n        if val == self.sliderleft.value():\n            val += 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(right=val)\n        self.rightvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "funcbottom",
          "class_name": "SubplotToolQt",
          "code": "def funcbottom(self, val):\n        if val == self.slidertop.value():\n            val -= 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(bottom=val)\n        self.bottomvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/backends/backend_qt5.py",
          "type": "function",
          "name": "functop",
          "class_name": "SubplotToolQt",
          "code": "def functop(self, val):\n        if val == self.sliderbottom.value():\n            val += 1\n        val /= 1000.\n        self.targetfig.subplots_adjust(top=val)\n        self.topvalue.setText(\"%.2f\" % val)\n        if self.drawon:\n            self.targetfig.canvas.draw()"
        },
        {
          "file": "lib/matplotlib/cbook.py",
          "type": "function",
          "name": "issubclass_safe",
          "class_name": null,
          "code": "def issubclass_safe(x, klass):\n    'return issubclass(x, klass) and return False on a TypeError'\n\n    try:\n        return issubclass(x, klass)\n    except TypeError:\n        return False"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "changed",
          "class_name": "ContourSet",
          "code": "def changed(self):\n        tcolors = [(tuple(rgba),)\n                   for rgba in self.to_rgba(self.cvalues, alpha=self.alpha)]\n        self.tcolors = tcolors\n        hatches = self.hatches * len(tcolors)\n        for color, hatch, collection in zip(tcolors, hatches,\n                                            self.collections):\n            if self.filled:\n                collection.set_facecolor(color)\n                # update the collection's hatch (may be None)\n                collection.set_hatch(hatch)\n            else:\n                collection.set_color(color)\n        for label, cv in zip(self.labelTexts, self.labelCValues):\n            label.set_alpha(self.alpha)\n            label.set_color(self.labelMappable.to_rgba(cv))\n        # add label colors\n        cm.ScalarMappable.changed(self)"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_contour_level_args",
          "class_name": "ContourSet",
          "code": "def _contour_level_args(self, z, args):\n        \"\"\"\n        Determine the contour levels and store in self.levels.\n        \"\"\"\n        if self.filled:\n            fn = 'contourf'\n        else:\n            fn = 'contour'\n        self._auto = False\n        if self.levels is None:\n            if len(args) == 0:\n                lev = self._autolev(z, 7)\n            else:\n                level_arg = args[0]\n                try:\n                    if type(level_arg) == int:\n                        lev = self._autolev(z, level_arg)\n                    else:\n                        lev = np.asarray(level_arg).astype(np.float64)\n                except:\n                    raise TypeError(\n                        \"Last %s arg must give levels; see help(%s)\" %\n                        (fn, fn))\n            self.levels = lev\n        if self.filled and len(self.levels) < 2:\n            raise ValueError(\"Filled contours require at least 2 levels.\")\n\n        if len(self.levels) > 1 and np.amin(np.diff(self.levels)) <= 0.0:\n            if hasattr(self, '_corner_mask') and self._corner_mask == 'legacy':\n                warnings.warn(\"Contour levels are not increasing\")\n            else:\n                raise ValueError(\"Contour levels must be increasing\")"
        },
        {
          "file": "lib/matplotlib/contour.py",
          "type": "function",
          "name": "_contour_args",
          "class_name": "QuadContourSet",
          "code": "def _contour_args(self, args, kwargs):\n        if self.filled:\n            fn = 'contourf'\n        else:\n            fn = 'contour'\n        Nargs = len(args)\n        if Nargs <= 2:\n            z = ma.asarray(args[0], dtype=np.float64)\n            x, y = self._initialize_x_y(z)\n            args = args[1:]\n        elif Nargs <= 4:\n            x, y, z = self._check_xyz(args[:3], kwargs)\n            args = args[3:]\n        else:\n            raise TypeError(\"Too many arguments to %s; see help(%s)\" %\n                            (fn, fn))\n        z = ma.masked_invalid(z, copy=False)\n        self.zmax = ma.maximum(z)\n        self.zmin = ma.minimum(z)\n        if self.logscale and self.zmin <= 0:\n            z = ma.masked_where(z <= 0, z)\n            warnings.warn('Log scale: values of z <= 0 have been masked')\n            self.zmin = z.min()\n        self._contour_level_args(z, args)\n        return (x, y, z)"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "class",
          "name": "AutoDateFormatter",
          "code": "class AutoDateFormatter(ticker.Formatter):\n    \"\"\"\n    This class attempts to figure out the best format to use.  This is\n    most useful when used with the :class:`AutoDateLocator`.\n\n\n    The AutoDateFormatter has a scale dictionary that maps the scale\n    of the tick (the distance in days between one major tick) and a\n    format string.  The default looks like this::\n\n        self.scaled = {\n            DAYS_PER_YEAR: rcParams['date.autoformat.year'],\n            DAYS_PER_MONTH: rcParams['date.autoformat.month'],\n            1.0: rcParams['date.autoformat.day'],\n            1. / HOURS_PER_DAY: rcParams['date.autoformat.hour'],\n            1. / (MINUTES_PER_DAY): rcParams['date.autoformat.minute'],\n            1. / (SEC_PER_DAY): rcParams['date.autoformat.second']}\n\n\n    The algorithm picks the key in the dictionary that is >= the\n    current scale and uses that format string.  You can customize this\n    dictionary by doing::\n\n\n    >>> locator = AutoDateLocator()\n    >>> formatter = AutoDateFormatter(locator)\n    >>> formatter.scaled[1/(24.*60.)] = '%M:%S' # only show min and sec\n\n    A custom :class:`~matplotlib.ticker.FuncFormatter` can also be used.\n    The following example shows how to use a custom format function to strip\n    trailing zeros from decimal seconds and adds the date to the first\n    ticklabel::\n\n        >>> def my_format_function(x, pos=None):\n        ...     x = matplotlib.dates.num2date(x)\n        ...     if pos == 0:\n        ...         fmt = '%D %H:%M:%S.%f'\n        ...     else:\n        ...         fmt = '%H:%M:%S.%f'\n        ...     label = x.strftime(fmt)\n        ...     label = label.rstrip(\"0\")\n        ...     label = label.rstrip(\".\")\n        ...     return label\n        >>> from matplotlib.ticker import FuncFormatter\n        >>> formatter.scaled[1/(24.*60.)] = FuncFormatter(my_format_function)\n    \"\"\"\n\n    # This can be improved by providing some user-level direction on\n    # how to choose the best format (precedence, etc...)\n\n    # Perhaps a 'struct' that has a field for each time-type where a\n    # zero would indicate \"don't show\" and a number would indicate\n    # \"show\" with some sort of priority.  Same priorities could mean\n    # show all with the same priority.\n\n    # Or more simply, perhaps just a format string for each\n    # possibility...\n\n    def __init__(self, locator, tz=None, defaultfmt='%Y-%m-%d'):\n        \"\"\"\n        Autoformat the date labels.  The default format is the one to use\n        if none of the values in ``self.scaled`` are greater than the unit\n        returned by ``locator._get_unit()``.\n        \"\"\"\n        self._locator = locator\n        self._tz = tz\n        self.defaultfmt = defaultfmt\n        self._formatter = DateFormatter(self.defaultfmt, tz)\n        self.scaled = {DAYS_PER_YEAR: rcParams['date.autoformatter.year'],\n                       DAYS_PER_MONTH: rcParams['date.autoformatter.month'],\n                       1.0: rcParams['date.autoformatter.day'],\n                       1. / HOURS_PER_DAY: rcParams['date.autoformatter.hour'],\n                       1. / (MINUTES_PER_DAY):\n                           rcParams['date.autoformatter.minute'],\n                       1. / (SEC_PER_DAY):\n                           rcParams['date.autoformatter.second']}\n\n    def __call__(self, x, pos=None):\n        locator_unit_scale = float(self._locator._get_unit())\n        fmt = self.defaultfmt\n\n        # Pick the first scale which is greater than the locator unit.\n        for possible_scale in sorted(self.scaled):\n            if possible_scale >= locator_unit_scale:\n                fmt = self.scaled[possible_scale]\n                break\n\n        if isinstance(fmt, six.string_types):\n            self._formatter = DateFormatter(fmt, self._tz)\n            result = self._formatter(x, pos)\n        elif six.callable(fmt):\n            result = fmt(x, pos)\n        else:\n            raise TypeError('Unexpected type passed to {0!r}.'.format(self))\n\n        return result"
        },
        {
          "file": "lib/matplotlib/dates.py",
          "type": "function",
          "name": "__init__",
          "class_name": "MicrosecondLocator",
          "code": "def __init__(self, interval=1, tz=None):\n        \"\"\"\n        *interval* is the interval between each iteration.  For\n        example, if ``interval=2``, mark every second microsecond.\n\n        \"\"\"\n        self._interval = interval\n        self._wrapped_locator = ticker.MultipleLocator(interval)\n        self.tz = tz"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "set_data",
          "class_name": "PcolorImage",
          "code": "def set_data(self, x, y, A):\n        A = cbook.safe_masked_invalid(A)\n        if x is None:\n            x = np.arange(0, A.shape[1]+1, dtype=np.float64)\n        else:\n            x = np.asarray(x, np.float64).ravel()\n        if y is None:\n            y = np.arange(0, A.shape[0]+1, dtype=np.float64)\n        else:\n            y = np.asarray(y, np.float64).ravel()\n\n        if A.shape[:2] != (y.size-1, x.size-1):\n            raise ValueError(\n                \"Axes don't match array shape. Got %s, expected %s.\" %\n                (A.shape[:2], (y.size - 1, x.size - 1)))\n        if A.ndim not in [2, 3]:\n            raise ValueError(\"A must be 2D or 3D\")\n        if A.ndim == 3 and A.shape[2] == 1:\n            A.shape = A.shape[:2]\n        self.is_grayscale = False\n        if A.ndim == 3:\n            if A.shape[2] in [3, 4]:\n                if ((A[:, :, 0] == A[:, :, 1]).all() and\n                        (A[:, :, 0] == A[:, :, 2]).all()):\n                    self.is_grayscale = True\n            else:\n                raise ValueError(\"3D arrays must have RGB or RGBA as last dim\")\n        self._A = A\n        self._Ax = x\n        self._Ay = y\n        self._rgbacache = None\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "set_data",
          "class_name": "PcolorImage",
          "code": "def set_data(self, x, y, A):\n        A = cbook.safe_masked_invalid(A)\n        if x is None:\n            x = np.arange(0, A.shape[1]+1, dtype=np.float64)\n        else:\n            x = np.asarray(x, np.float64).ravel()\n        if y is None:\n            y = np.arange(0, A.shape[0]+1, dtype=np.float64)\n        else:\n            y = np.asarray(y, np.float64).ravel()\n\n        if A.shape[:2] != (y.size-1, x.size-1):\n            raise ValueError(\n                \"Axes don't match array shape. Got %s, expected %s.\" %\n                (A.shape[:2], (y.size - 1, x.size - 1)))\n        if A.ndim not in [2, 3]:\n            raise ValueError(\"A must be 2D or 3D\")\n        if A.ndim == 3 and A.shape[2] == 1:\n            A.shape = A.shape[:2]\n        self.is_grayscale = False\n        if A.ndim == 3:\n            if A.shape[2] in [3, 4]:\n                if ((A[:, :, 0] == A[:, :, 1]).all() and\n                        (A[:, :, 0] == A[:, :, 2]).all()):\n                    self.is_grayscale = True\n            else:\n                raise ValueError(\"3D arrays must have RGB or RGBA as last dim\")\n        self._A = A\n        self._Ax = x\n        self._Ay = y\n        self._rgbacache = None\n        self.stale = True"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "__init__",
          "class_name": "BboxImage",
          "code": "def __init__(self, bbox,\n                 cmap=None,\n                 norm=None,\n                 interpolation=None,\n                 origin=None,\n                 filternorm=1,\n                 filterrad=4.0,\n                 resample=False,\n                 interp_at_native=True,\n                 **kwargs\n                 ):\n\n        \"\"\"\n        cmap is a colors.Colormap instance\n        norm is a colors.Normalize instance to map luminance to 0-1\n\n        interp_at_native is a flag that determines whether or not\n        interpolation should still be applied when the image is\n        displayed at its native resolution.  A common use case for this\n        is when displaying an image for annotational purposes; it is\n        treated similarly to Photoshop (interpolation is only used when\n        displaying the image at non-native resolutions).\n\n\n        kwargs are an optional list of Artist keyword args\n\n        \"\"\"\n        super(BboxImage, self).__init__(\n            None,\n            cmap=cmap,\n            norm=norm,\n            interpolation=interpolation,\n            origin=origin,\n            filternorm=filternorm,\n            filterrad=filterrad,\n            resample=resample,\n            **kwargs\n        )\n\n        self.bbox = bbox\n        self.interp_at_native = interp_at_native\n        self._transform = IdentityTransform()"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "_check_unsampled_image",
          "class_name": "PcolorImage",
          "code": "def _check_unsampled_image(self, renderer):\n        return False"
        },
        {
          "file": "lib/matplotlib/image.py",
          "type": "function",
          "name": "make_image",
          "class_name": "BboxImage",
          "code": "def make_image(self, renderer, magnification=1.0, unsampled=False):\n        width, height = renderer.get_canvas_width_height()\n\n        bbox_in = self.get_window_extent(renderer).frozen()\n        bbox_in._points /= [width, height]\n        bbox_out = self.get_window_extent(renderer)\n        clip = Bbox([[0, 0], [width, height]])\n        self._transform = BboxTransform(Bbox([[0, 0], [1, 1]]), clip)\n\n        return self._make_image(\n            self._A,\n            bbox_in, bbox_out, clip, magnification, unsampled=unsampled)"
        },
        {
          "file": "lib/matplotlib/rcsetup.py",
          "type": "function",
          "name": "validate_animation_writer_path",
          "class_name": null,
          "code": "def validate_animation_writer_path(p):\n    # Make sure it's a string and then figure out if the animations\n    # are already loaded and reset the writers (which will validate\n    # the path on next call)\n    if not isinstance(p, six.text_type):\n        raise ValueError(\"path must be a (unicode) string\")\n    from sys import modules\n    # set dirty, so that the next call to the registry will re-evaluate\n    # the state.\n    # only set dirty if already loaded. If not loaded, the load will\n    # trigger the checks.\n    if \"matplotlib.animation\" in modules:\n        modules[\"matplotlib.animation\"].writers.set_dirty()\n    return p"
        },
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "__init__",
          "class_name": "OldAutoLocator",
          "code": "def __init__(self):\n        self._locator = LinearLocator()"
        },
        {
          "file": "lib/matplotlib/ticker.py",
          "type": "function",
          "name": "tick_values",
          "class_name": "OldAutoLocator",
          "code": "def tick_values(self, vmin, vmax):\n        raise NotImplementedError('Cannot get tick locations for a '\n                                  '%s type.' % type(self))"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "check",
          "class_name": "PdfToPs",
          "code": "def check(self):\n        try:\n            output = check_output('pdftops -v', shell=True,\n                                  stderr=subprocess.STDOUT)\n            for line in output.splitlines():\n                line = line.decode()\n                if 'version' in line:\n                    return \"version %s\" % line.split()[2]\n        except (IndexError, ValueError, subprocess.CalledProcessError):\n            pass\n\n        raise CheckFailed()"
        },
        {
          "file": "setupext.py",
          "type": "function",
          "name": "check",
          "class_name": "PdfToPs",
          "code": "def check(self):\n        try:\n            output = check_output('pdftops -v', shell=True,\n                                  stderr=subprocess.STDOUT)\n            for line in output.splitlines():\n                line = line.decode()\n                if 'version' in line:\n                    return \"version %s\" % line.split()[2]\n        except (IndexError, ValueError, subprocess.CalledProcessError):\n            pass\n\n        raise CheckFailed()"
        }
      ]
    },
    {
      "pr_number": 17408,
      "pr_title": "FIX: cancel pending autoscale on manually setting limits",
      "pr_body": "closes #17331\r\n\r\n## PR Summary\r\nApplies patch suggested by @anntzer  in https://github.com/matplotlib/matplotlib/issues/17331#issuecomment-624109727\r\n\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\r\n",
      "issue_id": 17331,
      "issue_title": "Surprising/changed axis limit (autoscale) behavior",
      "issue_body": "### Bug report\r\n\r\n**Bug summary**\r\n\r\nThe interaction of setting axis limits and autoscaling changed in 3.2 with little guidance, introducing unexpected behavior that doesn't always make sense.\r\n\r\n**Code for reproduction**\r\n\r\nHere is a plot that looks very different on 3.1.2 and 3.2.1\r\n\r\n```python\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\nf, ax = plt.subplots()\r\nx = np.arange(100)\r\ny = np.random.uniform(-.1, .1, 100)\r\nax.scatter(x, y)\r\nassert ax._autoscaleYon\r\nax.set_ylim((-.5, .5), auto=None)\r\n```\r\n\r\n**Actual outcome**\r\n\r\nOn 3.2.1\r\n\r\n![image](https://user-images.githubusercontent.com/315810/81074711-656ca600-8eb7-11ea-9929-cef357be7cf1.png)\r\n\r\n**Expected outcome**\r\n\r\nOn 3.1.2\r\n\r\n![image](https://user-images.githubusercontent.com/315810/81074777-79b0a300-8eb7-11ea-8510-2bddabb72dd1.png)\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.2.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): pylab inline\r\n  * Python version: 3.7\r\n\r\n",
      "issue_closed_at": "2020-05-28T18:09:41Z",
      "base_commit": "4a8aa8ce227e4b823305e2adb616facd1435764b",
      "changes": [
        {
          "file": "lib/matplotlib/axes/_base.py",
          "type": "function",
          "name": "set_xlim",
          "class_name": "_AxesBase",
          "code": "def set_xlim(self, left=None, right=None, emit=True, auto=False,\n                 *, xmin=None, xmax=None):\n        \"\"\"\n        Set the x-axis view limits.\n\n        Parameters\n        ----------\n        left : float, optional\n            The left xlim in data coordinates. Passing *None* leaves the\n            limit unchanged.\n\n            The left and right xlims may also be passed as the tuple\n            (*left*, *right*) as the first positional argument (or as\n            the *left* keyword argument).\n\n            .. ACCEPTS: (bottom: float, top: float)\n\n        right : float, optional\n            The right xlim in data coordinates. Passing *None* leaves the\n            limit unchanged.\n\n        emit : bool, default: True\n            Whether to notify observers of limit change.\n\n        auto : bool or None, default: False\n            Whether to turn on autoscaling of the x-axis. True turns on,\n            False turns off, None leaves unchanged.\n\n        xmin, xmax : float, optional\n            They are equivalent to left and right respectively,\n            and it is an error to pass both *xmin* and *left* or\n            *xmax* and *right*.\n\n        Returns\n        -------\n        left, right : (float, float)\n            The new x-axis limits in data coordinates.\n\n        See Also\n        --------\n        get_xlim\n        set_xbound, get_xbound\n        invert_xaxis, xaxis_inverted\n\n        Notes\n        -----\n        The *left* value may be greater than the *right* value, in which\n        case the x-axis values will decrease from left to right.\n\n        Examples\n        --------\n        >>> set_xlim(left, right)\n        >>> set_xlim((left, right))\n        >>> left, right = set_xlim(left, right)\n\n        One limit may be left unchanged.\n\n        >>> set_xlim(right=right_lim)\n\n        Limits may be passed in reverse order to flip the direction of\n        the x-axis. For example, suppose *x* represents the number of\n        years before present. The x-axis limits might be set like the\n        following so 5000 years ago is on the left of the plot and the\n        present is on the right.\n\n        >>> set_xlim(5000, 0)\n\n        \"\"\"\n        if right is None and np.iterable(left):\n            left, right = left\n        if xmin is not None:\n            if left is not None:\n                raise TypeError('Cannot pass both `xmin` and `left`')\n            left = xmin\n        if xmax is not None:\n            if right is not None:\n                raise TypeError('Cannot pass both `xmax` and `right`')\n            right = xmax\n\n        self._process_unit_info(xdata=(left, right))\n        left = self._validate_converted_limits(left, self.convert_xunits)\n        right = self._validate_converted_limits(right, self.convert_xunits)\n\n        if left is None or right is None:\n            # Axes init calls set_xlim(0, 1) before get_xlim() can be called,\n            # so only grab the limits if we really need them.\n            old_left, old_right = self.get_xlim()\n            if left is None:\n                left = old_left\n            if right is None:\n                right = old_right\n\n        if self.get_xscale() == 'log' and (left <= 0 or right <= 0):\n            # Axes init calls set_xlim(0, 1) before get_xlim() can be called,\n            # so only grab the limits if we really need them.\n            old_left, old_right = self.get_xlim()\n            if left <= 0:\n                cbook._warn_external(\n                    'Attempted to set non-positive left xlim on a '\n                    'log-scaled axis.\\n'\n                    'Invalid limit will be ignored.')\n                left = old_left\n            if right <= 0:\n                cbook._warn_external(\n                    'Attempted to set non-positive right xlim on a '\n                    'log-scaled axis.\\n'\n                    'Invalid limit will be ignored.')\n                right = old_right\n        if left == right:\n            cbook._warn_external(\n                f\"Attempting to set identical left == right == {left} results \"\n                f\"in singular transformations; automatically expanding.\")\n        reverse = left > right\n        left, right = self.xaxis.get_major_locator().nonsingular(left, right)\n        left, right = self.xaxis.limit_range_for_scale(left, right)\n        # cast to bool to avoid bad interaction between python 3.8 and np.bool_\n        left, right = sorted([left, right], reverse=bool(reverse))\n\n        self._viewLim.intervalx = (left, right)\n        if auto is not None:\n            self._autoscaleXon = bool(auto)\n\n        if emit:\n            self.callbacks.process('xlim_changed', self)\n            # Call all of the other x-axes that are shared with this one\n            for other in self._shared_x_axes.get_siblings(self):\n                if other is not self:\n                    other.set_xlim(self.viewLim.intervalx,\n                                   emit=False, auto=auto)\n                    if other.figure != self.figure:\n                        other.figure.canvas.draw_idle()\n        self.stale = True\n        return left, right"
        },
        {
          "file": "lib/matplotlib/axes/_base.py",
          "type": "function",
          "name": "set_ylim",
          "class_name": "_AxesBase",
          "code": "def set_ylim(self, bottom=None, top=None, emit=True, auto=False,\n                 *, ymin=None, ymax=None):\n        \"\"\"\n        Set the y-axis view limits.\n\n        Parameters\n        ----------\n        bottom : float, optional\n            The bottom ylim in data coordinates. Passing *None* leaves the\n            limit unchanged.\n\n            The bottom and top ylims may also be passed as the tuple\n            (*bottom*, *top*) as the first positional argument (or as\n            the *bottom* keyword argument).\n\n            .. ACCEPTS: (bottom: float, top: float)\n\n        top : float, optional\n            The top ylim in data coordinates. Passing *None* leaves the\n            limit unchanged.\n\n        emit : bool, default: True\n            Whether to notify observers of limit change.\n\n        auto : bool or None, default: False\n            Whether to turn on autoscaling of the y-axis. *True* turns on,\n            *False* turns off, *None* leaves unchanged.\n\n        ymin, ymax : float, optional\n            They are equivalent to bottom and top respectively,\n            and it is an error to pass both *ymin* and *bottom* or\n            *ymax* and *top*.\n\n        Returns\n        -------\n        bottom, top : (float, float)\n            The new y-axis limits in data coordinates.\n\n        See Also\n        --------\n        get_ylim\n        set_ybound, get_ybound\n        invert_yaxis, yaxis_inverted\n\n        Notes\n        -----\n        The *bottom* value may be greater than the *top* value, in which\n        case the y-axis values will decrease from *bottom* to *top*.\n\n        Examples\n        --------\n        >>> set_ylim(bottom, top)\n        >>> set_ylim((bottom, top))\n        >>> bottom, top = set_ylim(bottom, top)\n\n        One limit may be left unchanged.\n\n        >>> set_ylim(top=top_lim)\n\n        Limits may be passed in reverse order to flip the direction of\n        the y-axis. For example, suppose ``y`` represents depth of the\n        ocean in m. The y-axis limits might be set like the following\n        so 5000 m depth is at the bottom of the plot and the surface,\n        0 m, is at the top.\n\n        >>> set_ylim(5000, 0)\n        \"\"\"\n        if top is None and np.iterable(bottom):\n            bottom, top = bottom\n        if ymin is not None:\n            if bottom is not None:\n                raise TypeError('Cannot pass both `ymin` and `bottom`')\n            bottom = ymin\n        if ymax is not None:\n            if top is not None:\n                raise TypeError('Cannot pass both `ymax` and `top`')\n            top = ymax\n\n        self._process_unit_info(ydata=(bottom, top))\n        bottom = self._validate_converted_limits(bottom, self.convert_yunits)\n        top = self._validate_converted_limits(top, self.convert_yunits)\n\n        if bottom is None or top is None:\n            # Axes init calls set_ylim(0, 1) before get_ylim() can be called,\n            # so only grab the limits if we really need them.\n            old_bottom, old_top = self.get_ylim()\n            if bottom is None:\n                bottom = old_bottom\n            if top is None:\n                top = old_top\n\n        if self.get_yscale() == 'log' and (bottom <= 0 or top <= 0):\n            # Axes init calls set_xlim(0, 1) before get_xlim() can be called,\n            # so only grab the limits if we really need them.\n            old_bottom, old_top = self.get_ylim()\n            if bottom <= 0:\n                cbook._warn_external(\n                    'Attempted to set non-positive bottom ylim on a '\n                    'log-scaled axis.\\n'\n                    'Invalid limit will be ignored.')\n                bottom = old_bottom\n            if top <= 0:\n                cbook._warn_external(\n                    'Attempted to set non-positive top ylim on a '\n                    'log-scaled axis.\\n'\n                    'Invalid limit will be ignored.')\n                top = old_top\n        if bottom == top:\n            cbook._warn_external(\n                f\"Attempting to set identical bottom == top == {bottom} \"\n                f\"results in singular transformations; automatically \"\n                f\"expanding.\")\n        reverse = bottom > top\n        bottom, top = self.yaxis.get_major_locator().nonsingular(bottom, top)\n        bottom, top = self.yaxis.limit_range_for_scale(bottom, top)\n        # cast to bool to avoid bad interaction between python 3.8 and np.bool_\n        bottom, top = sorted([bottom, top], reverse=bool(reverse))\n\n        self._viewLim.intervaly = (bottom, top)\n        if auto is not None:\n            self._autoscaleYon = bool(auto)\n\n        if emit:\n            self.callbacks.process('ylim_changed', self)\n            # Call all of the other y-axes that are shared with this one\n            for other in self._shared_y_axes.get_siblings(self):\n                if other is not self:\n                    other.set_ylim(self.viewLim.intervaly,\n                                   emit=False, auto=auto)\n                    if other.figure != self.figure:\n                        other.figure.canvas.draw_idle()\n        self.stale = True\n        return bottom, top"
        }
      ]
    },
    {
      "pr_number": 17994,
      "pr_title": "Special case degree-1 Bezier curves.",
      "pr_body": "This greatly speeds up extent computation for the common case of\r\npolylines.  (We were previously only special-casing the degree-0 case.)\r\n\r\nPartially fixes #17974 (4x speedup locally).  My guess is that the real fix will involve restoring a special-case in Path.get_extents for paths that do not contain curves, but this fix is still useful by itself with no additional complexity cost.\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/next_api_changes/* 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": 17974,
      "issue_title": "Major speed regression introduced in \"plt.bar\" definition clipping between 3.0.3 and 3.3.0.",
      "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\nAs I updated Matplolib from version 3.0.3 to version 3.3.0 I noticed that [Colour](https://github.com/colour-science/colour/) unit test time skyrocketed: our plotting sub-package test suite went from 120 seconds to around 3000 seconds!\r\n\r\nI quickly noticed that it was related to definitions using the spectrum plot:\r\n\r\n![image](https://user-images.githubusercontent.com/99779/87933498-78b80680-cae1-11ea-8bf5-b09af15cec50.png)\r\n\r\nThis definition is using `plt.bar` and as I suspected that polygon clipping might be the culprit, I ran some tests on one of the offending definitions while keeping and removing the polygon clipping:\r\n\r\n*With Polygon Clipping*\r\n\r\n```python\r\n>>> import timeit\r\n>>> import colour\r\n>>> timeit.timeit(lambda : colour.plotting.plot_single_illuminant_sd(filename='test.png'), number=1)\r\n59.81594165299998\r\n```\r\n\r\n*Without Polygon Clipping*\r\n```python\r\n>>> import timeit\r\n>>> import colour\r\n>>> timeit.timeit(lambda : colour.plotting.plot_single_illuminant_sd(filename='test.png'), number=1)\r\n1.3159556400000056\r\n```\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\nI haven't created a reproducible case but my assumption is that the clipping code is orders of magnitude slower than before, the definition I used is available here: https://github.com/colour-science/colour/blob/d5f68005f62fc86ba59745bd4c8bb8a01bb5dcb4/colour/plotting/colorimetry.py#L183 and the meat is roughly as follows:\r\n\r\n```python\r\n    polygon = Polygon(\r\n        np.vstack([\r\n            (x_min, 0),\r\n            tstack([wavelengths, values]),\r\n            (x_max, 0),\r\n        ]),\r\n        facecolor='none',\r\n        edgecolor='none')\r\n    axes.add_patch(polygon)\r\n\r\n    padding = 0.1\r\n    axes.bar(\r\n        x=wavelengths - padding,\r\n        height=max(values),\r\n        width=1 + padding,\r\n        color=colours,\r\n        align='edge',\r\n        clip_path=polygon)\r\n```\r\n\r\nThere are roughly 450 wavelengths and each one of them generates a coloured bar.\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 10.15.5\r\n  * Matplotlib version: 3.3.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): MacOSX\r\n  * Python version: 3.8\r\n  * Jupyter version (if applicable):\r\n  * Other libraries: \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\nMatplotlib was installed with Pip.\r\n\r\n",
      "issue_closed_at": "2020-08-05T20:39:02Z",
      "base_commit": "57c8baaf85f0cfd44a27ef834dc971128b7f8ee4",
      "changes": [
        {
          "file": "lib/matplotlib/bezier.py",
          "type": "function",
          "name": "axis_aligned_extrema",
          "class_name": "BezierSegment",
          "code": "def axis_aligned_extrema(self):\n        \"\"\"\n        Return the dimension and location of the curve's interior extrema.\n\n        The extrema are the points along the curve where one of its partial\n        derivatives is zero.\n\n        Returns\n        -------\n        dims : int, array_like\n            Index :math:`i` of the partial derivative which is zero at each\n            interior extrema.\n        dzeros : float, array_like\n            Of same size as dims. The :math:`t` such that :math:`d/dx_i B(t) =\n            0`\n        \"\"\"\n        n = self.degree\n        Cj = self.polynomial_coefficients\n        dCj = np.arange(1, n+1)[:, None] * Cj[1:]\n        if len(dCj) == 0:\n            return np.array([]), np.array([])\n        dims = []\n        roots = []\n        for i, pi in enumerate(dCj.T):\n            r = np.roots(pi[::-1])\n            roots.append(r)\n            dims.append(np.full_like(r, i))\n        roots = np.concatenate(roots)\n        dims = np.concatenate(dims)\n        in_range = np.isreal(roots) & (roots >= 0) & (roots <= 1)\n        return dims[in_range], np.real(roots)[in_range]"
        }
      ]
    }
  ]
}