diff --git a/django/forms/widgets.py b/django/forms/widgets.py index 8c5122ad1d3..208464b60a0 100644 --- a/django/forms/widgets.py +++ b/django/forms/widgets.py @@ -101,7 +101,9 @@ class Media: def render_js(self): return [ - format_html('', self.absolute_path(path)) + path.__html__() + if hasattr(path, "__html__") + else format_html('', self.absolute_path(path)) for path in self._js ] @@ -111,7 +113,9 @@ class Media: media = sorted(self._css) return chain.from_iterable( [ - format_html( + path.__html__() + if hasattr(path, "__html__") + else format_html( '', self.absolute_path(path), medium, diff --git a/docs/releases/4.1.txt b/docs/releases/4.1.txt index 02d51ed2a5f..4b62cf09cfb 100644 --- a/docs/releases/4.1.txt +++ b/docs/releases/4.1.txt @@ -192,6 +192,11 @@ Forms * The new ``edit_only`` argument for :func:`.modelformset_factory` and :func:`.inlineformset_factory` allows preventing new objects creation. +* The ``js`` and ``css`` class attributes of :doc:`Media ` + now allow using hashable objects, not only path strings, as long as those + objects implement the ``__html__()`` method (typically when decorated with + the :func:`~django.utils.html.html_safe` decorator). + Generic Views ~~~~~~~~~~~~~ diff --git a/docs/topics/forms/media.txt b/docs/topics/forms/media.txt index 6ca7c66fde5..7e5a04e3d9c 100644 --- a/docs/topics/forms/media.txt +++ b/docs/topics/forms/media.txt @@ -206,7 +206,10 @@ return values for dynamic ``media`` properties. Paths in asset definitions ========================== -Paths used to specify assets can be either relative or absolute. If a +Paths as strings +---------------- + +String paths used to specify assets can be either relative or absolute. If a path starts with ``/``, ``http://`` or ``https://``, it will be interpreted as an absolute path, and left as-is. All other paths will be prepended with the value of the appropriate prefix. If the @@ -254,6 +257,28 @@ Or if :mod:`~django.contrib.staticfiles` is configured using the +Paths as objects +---------------- + +.. versionadded:: 4.1 + +Asset paths may also be given as hashable objects implementing an +``__html__()`` method. The ``__html__()`` method is typically added using the +:func:`~django.utils.html.html_safe` decorator. The object is responsible for +outputting the complete HTML ``' % ( + ' integrity="{}"' if self.integrity else "{}" + ) + return format_html(template, self.absolute_path(path), self.integrity) + + +@override_settings( + STATIC_URL="http://media.example.com/static/", +) +class FormsMediaObjectTestCase(SimpleTestCase): + """Media handling when media are objects instead of raw strings.""" + + def test_construction(self): + m = Media( + css={"all": (CSS("path/to/css1", "all"), CSS("/path/to/css2", "all"))}, + js=( + JS("/path/to/js1"), + JS("http://media.other.com/path/to/js2"), + JS( + "https://secure.other.com/path/to/js3", + integrity="9d947b87fdeb25030d56d01f7aa75800", + ), + ), + ) + self.assertEqual( + str(m), + '\n' + '\n' + '\n' + '\n' + '', + ) + self.assertEqual( + repr(m), + "Media(css={'all': ['path/to/css1', '/path/to/css2']}, " + "js=['/path/to/js1', 'http://media.other.com/path/to/js2', " + "'https://secure.other.com/path/to/js3'])", + ) + + def test_simplest_class(self): + @html_safe + class SimpleJS: + """The simplest possible asset class.""" + + def __str__(self): + return '\n' + '\n' + '\n' + '", + ) + + def test_media_deduplication(self): + # The deduplication doesn't only happen at the point of merging two or + # more media objects. + media = Media( + css={ + "all": ( + CSS("/path/to/css1", "all"), + CSS("/path/to/css1", "all"), + "/path/to/css1", + ) + }, + js=(JS("/path/to/js1"), JS("/path/to/js1"), "/path/to/js1"), + ) + self.assertEqual( + str(media), + '\n' + '', + )