web-dev-qa-db-ja.com

Jinja2:ループ内の変数の値を変更する

ループ内でループ外で宣言された変数の値を変更したい。しかし、常に変化し、初期値をループ外に保ちます。

{% set foo = False %}

{% for item in items %}
  {% set foo = True %}
  {% if foo %} Ok(1)! {% endif %}
{% endfor %}

{% if foo %} Ok(2)! {% endif %}

これはレンダリングします:

Ok(1)!

そのため、これまでに見つかった唯一の(悪い)解決策は次のとおりです。

{% set foo = [] %}

{% for item in items %}
  {% if foo.append(True) %} {% endif %}
  {% if foo %} Ok(1)! {% endif %}
{% endfor %}

{% if foo %} Ok(2)! {% endif %}

これはレンダリングします:

Ok(1)!
Ok(2)!

しかし、それは非常にいです!別のよりエレガントなソリューションはありますか?

57
Shankar Cabus

辞書ベースのアプローチも試してください。見苦しくないようです。

{% set vars = {'foo': False} %}

{% for item in items %}
  {% if vars.update({'foo': True}) %} {% endif %}
  {% if vars.foo %} Ok(1)! {% endif %}
{% endfor %}

{% if vars.foo %} Ok(2)! {% endif %}

これもレンダリングします:

Ok(1)!
Ok(2)!
57
Pashka

ドキュメントに記載されているとおり:

ループ内の割り当ては、反復の終了時にクリアされ、ループスコープを超えて存続できないことに注意してください。

ただし、バージョン2.10以降では、名前空間を使用できます。

    {% set ns = namespace(foo=false) %}      
    {% for item in items %}
      {% set ns.foo = True %}
      {% if ns.foo %} Ok(1)! {% endif %}
    {% endfor %}

    {% if ns.foo %} Ok(2)! {% endif %}
22
Omer

これを行うと、テンプレートコードをクリーンアップできます。

{% for item in items %}
  {{ set_foo_is_true(local_vars) }}
  {% if local_vars.foo %} Ok(1)! {% endif %}
{% endfor %}
{% if local_vars.foo %} Ok(2)! {% endif %}

そして、サーバーコードで使用

items = ['item1', 'item2', 'item3']
#---------------------------------------------
local_vars = { 'foo': False }
def set_foo_is_true(local_vars):
  local_vars['foo'] = True
  return ''
env.globals['set_foo_is_true'] = set_foo_is_true    
#---------------------------------------------
return env.get_template('template.html').render(items=items, local_vars=local_vars)

これは次のように一般化できます

{{ set_local_var(local_vars, "foo", False) }}
{% for item in items %}
  {{ set_local_var(local_vars, "foo", True) }}
  {% if local_vars.foo %} Ok(1)! {% endif %}
{% endfor %}
{% if local_vars.foo %} Ok(2)! {% endif %}

そして、サーバーコードで使用

items = ['item1', 'item2', 'item3']
#---------------------------------------------
local_vars = { 'foo': False }
def set_local_var(local_vars, name, value):
  local_vars[name] = value
  return ''
env.globals['set_local_var'] = set_local_var
#---------------------------------------------
return env.get_template('template.html').render(items=items, local_vars=local_vars)
1
Daniel F