質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

89.49%

Django のテンプレートで継承したテンプレートに値を渡す方法

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 390

fa5

score 7

Django でwebアプリを作成しています。

クリックで開閉するドロップダウンリストを作成しようと思い、ベースとなるテンプレートを用意し、
それを継承したテンプレートにリストの中身を記述しようと考えています。

<div>
  <input id="{% block name %}{% endblock %}" type="checkbox">
  <label for="{% block name %}{% endblock %}">
    {% block label %}{% endblock %}
  </label>

  {% block content %}{% endblock %}
</div>
{% extends dropdown_base.html %}

{% block name %}menu{% endblock %}
{% block label %}label{% endblock %}
{% block content%}
  <ul>
    ~~~~
  </ul>
{% endblock %}

上記のように継承側からドロップダウンの開閉を制御するチェックボックスの id を指定して、
その id と同じ for 要素を持った label を設置する形で作成しようと試してみましたが、
どうやら同じ名称の block は複数存在してはいけないようでエラーになってしまいました。

そこで、継承側からベースとなるテンプレートへ id にあたる値を渡し、

<div>
  <input id="{{ name }}" type="checkbox">
  <label for="{{ name }}">
    {% block label %}{% endblock %}
  </label>

  {% block content %}{% endblock %}
</div>


のように書けないかとも考えましたが、ベース側にうまく値を渡す方法がわかりませんでした。

イメージとしては、

{% extends dropdown_base.html with name="menu" %}


の様にかければ最高なのですが、、、

同じような機能が実装できれば、値を渡す以外の方法でも構いません。
どうぞ、よろしくお願いします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

0

include を使うのはダメなんでしょうか。(意図的に避けてるようにも見えますが) 
私であれば、部分的なテンプレートに対して include と with を使って作るような気がします。

# dropdown_block.html
<div>
  <input id="{{ name }}" type="checkbox">
  <label for="{{ label }}">
    {{ label }}
  </label>

  <ul>
  {% for c in contents %}
    <li>{{ c }}</li>
  {% endfor %}
  </ul>
</div>
# 使う方法
{% include "dropdown_block.html" with name="name1" label="label1" contents=contents1  %}

参考: https://docs.djangoproject.com/ja/2.1/ref/templates/builtins/#include

追記

実際に試してないので、動くかわかりませんが、似たような回答が Stackoverflowにありました。 多分下記のように block.super をうまく使うと name変数を継承先でもセットできそうです。

https://stackoverflow.com/a/46581444/7724457

継承元テンプレート

{% block dropdown_wrapper %}
<div>
  <input id="{{ name }}" type="checkbox">
  <label for="{{ name }}">
    {% block label %}{% endblock %}
  </label>

  {% block content %}{% endblock %}
</div>
{% endblock %}

継承先テンプレート

{% extends dropdown_base.html %}

{% block dropdown_wrapper %}
  {% with name="name1" %}
    {{ block.super }}
  {% endwith%}
{% endblock %}

{% block label %}label{% endblock %}
{% block content%}
  <ul>
    ~~~~
  </ul>
{% endblock %}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/01/08 22:46

    ご回答ありがとうございます。
    私自身も include による方法を最初は考えていたのですが、extends にしたほうがコンテンツ部分を自由に設定できるのではないかと考え、今回の質問に至りました。
    extends の方はコンテンツ部分のHTMLを継承側で記述するので、例えば <li> の中に <div> で囲まれたブロック要素を入れるなどが可能と考えました。
    頂いた回答をもとに、引き続き何かいい方法がないか自分でも探してみたいと思います。

    キャンセル

  • 2019/01/09 01:02

    なるほど理解いたしました。別案を回答してみました。もしよかったら参考にしてみてください。

    キャンセル

  • 2019/01/13 23:06

    先ほど試してみたところ、追記していただいた方法で想定していた機能が実装できました。
    どうもありがとうございます。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 89.49%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる