|
@@ -19,7 +19,7 @@ from django.urls import reverse
|
|
|
from django.urls.exceptions import NoReverseMatch
|
|
|
from django.utils import timezone
|
|
|
from django.utils.encoding import force_str
|
|
|
-from django.utils.html import avoid_wrapping, json_script
|
|
|
+from django.utils.html import avoid_wrapping, conditional_escape, json_script
|
|
|
from django.utils.http import urlencode
|
|
|
from django.utils.safestring import mark_safe
|
|
|
from django.utils.timesince import timesince
|
|
@@ -980,21 +980,107 @@ def resolve_url(url):
|
|
|
return ""
|
|
|
|
|
|
|
|
|
-@register.simple_tag(takes_context=True)
|
|
|
-def component(context, obj, fallback_render_method=False):
|
|
|
- # Render a component by calling its render_html method, passing request and context from the
|
|
|
- # calling template.
|
|
|
- # If fallback_render_method is true, objects without a render_html method will have render()
|
|
|
- # called instead (with no arguments) - this is to provide deprecation path for things that have
|
|
|
- # been newly upgraded to use the component pattern.
|
|
|
-
|
|
|
- has_render_html_method = hasattr(obj, "render_html")
|
|
|
- if fallback_render_method and not has_render_html_method and hasattr(obj, "render"):
|
|
|
- return obj.render()
|
|
|
- elif not has_render_html_method:
|
|
|
- raise ValueError(f"Cannot render {obj!r} as a component")
|
|
|
-
|
|
|
- return obj.render_html(context)
|
|
|
+class ComponentNode(template.Node):
|
|
|
+ def __init__(
|
|
|
+ self,
|
|
|
+ component,
|
|
|
+ extra_context=None,
|
|
|
+ isolated_context=False,
|
|
|
+ fallback_render_method=None,
|
|
|
+ target_var=None,
|
|
|
+ ):
|
|
|
+ self.component = component
|
|
|
+ self.extra_context = extra_context or {}
|
|
|
+ self.isolated_context = isolated_context
|
|
|
+ self.fallback_render_method = fallback_render_method
|
|
|
+ self.target_var = target_var
|
|
|
+
|
|
|
+ def render(self, context: Context) -> str:
|
|
|
+ # Render a component by calling its render_html method, passing request and context from the
|
|
|
+ # calling template.
|
|
|
+ # If fallback_render_method is true, objects without a render_html method will have render()
|
|
|
+ # called instead (with no arguments) - this is to provide deprecation path for things that have
|
|
|
+ # been newly upgraded to use the component pattern.
|
|
|
+
|
|
|
+ component = self.component.resolve(context)
|
|
|
+
|
|
|
+ if self.fallback_render_method:
|
|
|
+ fallback_render_method = self.fallback_render_method.resolve(context)
|
|
|
+ else:
|
|
|
+ fallback_render_method = False
|
|
|
+
|
|
|
+ values = {
|
|
|
+ name: var.resolve(context) for name, var in self.extra_context.items()
|
|
|
+ }
|
|
|
+
|
|
|
+ if hasattr(component, "render_html"):
|
|
|
+ if self.isolated_context:
|
|
|
+ html = component.render_html(context.new(values))
|
|
|
+ else:
|
|
|
+ with context.push(**values):
|
|
|
+ html = component.render_html(context)
|
|
|
+ elif fallback_render_method and hasattr(component, "render"):
|
|
|
+ html = component.render()
|
|
|
+ else:
|
|
|
+ raise ValueError(f"Cannot render {component!r} as a component")
|
|
|
+
|
|
|
+ if self.target_var:
|
|
|
+ context[self.target_var] = html
|
|
|
+ return ""
|
|
|
+ else:
|
|
|
+ if context.autoescape:
|
|
|
+ html = conditional_escape(html)
|
|
|
+ return html
|
|
|
+
|
|
|
+
|
|
|
+@register.tag(name="component")
|
|
|
+def component(parser, token):
|
|
|
+ bits = token.split_contents()[1:]
|
|
|
+ if not bits:
|
|
|
+ raise template.TemplateSyntaxError(
|
|
|
+ "'component' tag requires at least one argument, the component object"
|
|
|
+ )
|
|
|
+
|
|
|
+ component = parser.compile_filter(bits.pop(0))
|
|
|
+
|
|
|
+ # the only valid keyword argument immediately following the component
|
|
|
+ # is fallback_render_method
|
|
|
+ flags = token_kwargs(bits, parser)
|
|
|
+ fallback_render_method = flags.pop("fallback_render_method", None)
|
|
|
+ if flags:
|
|
|
+ raise template.TemplateSyntaxError(
|
|
|
+ "'component' tag only accepts 'fallback_render_method' as a keyword argument"
|
|
|
+ )
|
|
|
+
|
|
|
+ extra_context = {}
|
|
|
+ isolated_context = False
|
|
|
+ target_var = None
|
|
|
+
|
|
|
+ while bits:
|
|
|
+ bit = bits.pop(0)
|
|
|
+ if bit == "with":
|
|
|
+ extra_context = token_kwargs(bits, parser)
|
|
|
+ elif bit == "only":
|
|
|
+ isolated_context = True
|
|
|
+ elif bit == "as":
|
|
|
+ try:
|
|
|
+ target_var = bits.pop(0)
|
|
|
+ except IndexError:
|
|
|
+ raise template.TemplateSyntaxError(
|
|
|
+ "'component' tag with 'as' must be followed by a variable name"
|
|
|
+ )
|
|
|
+ else:
|
|
|
+ raise template.TemplateSyntaxError(
|
|
|
+ "'component' tag received an unknown argument: %r" % bit
|
|
|
+ )
|
|
|
+
|
|
|
+ return ComponentNode(
|
|
|
+ component,
|
|
|
+ extra_context=extra_context,
|
|
|
+ isolated_context=isolated_context,
|
|
|
+ fallback_render_method=fallback_render_method,
|
|
|
+ target_var=target_var,
|
|
|
+ )
|
|
|
|
|
|
|
|
|
class FragmentNode(template.Node):
|