Python 如何在Jinja扩展中解析和注入其他节点?
我正在尝试修改Jinja2,以生成用于包装块的通用扩展(后面是一些更复杂的扩展) 我的目标是在模板中支持以下内容:Python 如何在Jinja扩展中解析和注入其他节点?,python,django,python-2.7,jinja2,django-1.6,Python,Django,Python 2.7,Jinja2,Django 1.6,我正在尝试修改Jinja2,以生成用于包装块的通用扩展(后面是一些更复杂的扩展) 我的目标是在模板中支持以下内容: {% wrap template='wrapper.html.j2' ... %} <img src="{{ url('image:thumbnail' ... }}"> {% endwrap %} 但是,它落在我的节点上。Include行,我只得到断言帧为None,“不允许根帧”。我认为我需要将AST传递给节点。Template而不是模板名称,但我真的不知道
{% wrap template='wrapper.html.j2' ... %}
<img src="{{ url('image:thumbnail' ... }}">
{% endwrap %}
但是,它落在我的
节点上。Include
行,我只得到断言帧为None,“不允许根帧”
。我认为我需要将AST传递给节点。Template
而不是模板名称,但我真的不知道如何解析其他节点以获得AST而不是字符串输出(即渲染),也不知道这是否是正确的方法。我说的对吗,我该怎么做呢?templatetags/wrap.py
class WrapExtension(jinja2.ext.Extension):
tags = set(['wrap'])
template = None
def parse(self, parser):
tag = parser.stream.current.value
lineno = parser.stream.next().lineno
args, kwargs = self.parse_args(parser)
body = parser.parse_statements(['name:end{}'.format(tag)], drop_needle=True)
return nodes.CallBlock(self.call_method('wrap', args, kwargs), [], [], body).set_lineno(lineno)
def parse_args(self, parser):
args = []
kwargs = []
require_comma = False
while parser.stream.current.type != 'block_end':
if require_comma:
parser.stream.expect('comma')
if parser.stream.current.type == 'name' and parser.stream.look().type == 'assign':
key = parser.stream.current.value
parser.stream.skip(2)
value = parser.parse_expression()
kwargs.append(nodes.Keyword(key, value, lineno=value.lineno))
else:
if kwargs:
parser.fail('Invalid argument syntax for WrapExtension tag',
parser.stream.current.lineno)
args.append(parser.parse_expression())
require_comma = True
return args, kwargs
@jinja2.contextfunction
def wrap(self, context, caller, template=None, *args, **kwargs):
return self.environment.get_template(template or self.template).render(dict(context, content=caller(), **kwargs))
class ExampleExtension(WrapExtension):
tags = set(['example'])
template = 'example.html.j2'
base.html.j2
<h1>dsd</h1>
{% wrap template='wrapper.html.j2' %}
{% for i in range(3) %}
im wrapped content {{ i }}<br>
{% endfor %}
{% endwrap %}
Hello im wrapper
<br>
<hr>
{{ content|safe }}
<hr>
{% example otherstuff=True, somethingelse=False %}
{% for i in range(3) %}
im wrapped content {{ i }}<br>
{% endfor %}
{% endexample %}
base.html.j2
<h1>dsd</h1>
{% wrap template='wrapper.html.j2' %}
{% for i in range(3) %}
im wrapped content {{ i }}<br>
{% endfor %}
{% endwrap %}
Hello im wrapper
<br>
<hr>
{{ content|safe }}
<hr>
{% example otherstuff=True, somethingelse=False %}
{% for i in range(3) %}
im wrapped content {{ i }}<br>
{% endfor %}
{% endexample %}
{%example otherstuff=True,somethingelse=False%}
{范围(3)%内的i的百分比}
im包装内容{{i}}
{%endfor%}
{%endexample%}
更好的处理方法是使用宏。定义为:
{% macro wrapper() -%}
<div>
some ifs and stuff
{{ caller() }}
more ifs and stuff
</div>
{%- endmacro %}
宏可以具有类似python函数的参数,并且可以导入:
{% from macros import wrapper %}
有关更多详细信息,请参阅
宏
、调用
和导入
标记的文档。能否添加有关所需结果的更多详细信息?您的意思是以包含在wrapper.html.j2
内容中的标记结束吗?你能给我一个wrapper.html.j2
的内容示例吗?哦,哎呀,为了简化我的示例,我让它变得毫无意义,我将更新示例。我将我的示例简化了一点,除了传递给hello_wrapper.js的模板
之外,我确实需要任何kwargs–否则看起来不错:我不确定我是否遵循了,如何将args和kwargs传递到parse
中的CallBlock
?我假设它们最终需要被传递到\u wrap
?内部调用的渲染
调用。Parse将文本编译为节点。解析后,渲染器调用节点(您可以看到CallBlock继承节点类)。因此,您需要准备CallBlack:set方法,args和kwargs将它们转换为文字类型。我已经更新了您提供的示例,包括传递args和kwargs,以及将它们合并到上下文中,以便我可以继续在包装模板中使用request和contextprocessor生成的变量。此外,我还举了一个例子,说明如何在其他具有预定义模板的自定义标记中使用它。谢谢很抱歉在3年后询问后续问题……但在Jinja2 2.10中,这给出了TypeError:ExtensionRegistry类型的对象没有len()