自動化無しに生活無し

WEB開発関係を中心に備忘録をまとめています

DjangoのMessageFrameworkで投稿とエラーをフロント側に表示する

thumbnail

MessageFrameworkを使ってHelloWorldを表示させる。

元になるコードは40分Djangoにforms.pyを追加した状態から流用している。

まず、views.pyにて、公式からコードを拝借して追加した。

from django.shortcuts import render,redirect

from django.views import View

from .models import Topic
from .forms import TopicForm

#↓追加
from django.contrib import messages

class IndexView(View):

    def get(self, request, *args, **kwargs):

        #↓追加    
        messages.add_message(request, messages.INFO, 'Hello world.')

        topics  = Topic.objects.all()
        context = { "topics":topics }

        return render(request,"bbs/index.html",context)

    def post(self, request, *args, **kwargs):

        form    = TopicForm(request.POST)

        if form.is_valid():
            form.save()

        return redirect("bbs:index")

index   = IndexView.as_view()

HTML側でメッセージを表示させるにはこうする。

<!DOCTYPE html>
<html lang="ja">
<head>
	<meta charset="UTF-8">
    <meta name="viewport" content="width=device-width,initial-scale=1">
	<title>簡易掲示板</title>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.4.1/css/bootstrap.min.css" integrity="sha384-Vkoo8x4CGsO3+Hhxv8T/Q5PaXtkKtu6ug5TOeNV6gBiFeWPGFN9MuhOf23Q9Ifjh" crossorigin="anonymous">
</head>
<body>

    <main class="container">

        <!--↓追加-->
        {% if messages %}
        <ul class="messages">
            {% for message in messages %}
            <li{% if message.tags %} class="{{ message.tags }}"{% endif %}>{{ message }}</li>
            {% endfor %}
        </ul>
        {% endif %}


        <form method="POST">
            {% csrf_token %}
            <textarea class="form-control" name="comment"></textarea>
            <input type="submit" value="送信">
        </form>

        {% for topic in topics %}
        <div class="border">
            {{ topic.comment }}
        </div>
        {% endfor %}

    </main>
</body>
</html>

つまり、仕組みは

messages.add_message(request, messages.INFO, 'Hello world.')

これでメッセージを追加。settings.pyの設定によりテンプレート側に引き渡される変数messagesを参照し、HTML上で表示させている。

【応用】投稿成功時、投稿エラー時にそれぞれメッセージを表示させる

テンプレートはそのままにビューを下記のようにする。

from django.shortcuts import render,redirect

from django.views import View
from .models import Topic
from .forms import TopicForm


from django.contrib import messages


class IndexView(View):

    def get(self, request, *args, **kwargs):

        ##messages.add_message(request, messages.INFO, 'Hello world.')

        topics  = Topic.objects.all()
        context = { "topics":topics }

        return render(request,"bbs/index.html",context)

    def post(self, request, *args, **kwargs):

        form    = TopicForm(request.POST)

        if form.is_valid():
            form.save()
            messages.add_message(request, messages.INFO, '投稿内容を保存しました')
        else:
            messages.add_message(request, messages.INFO, '投稿エラー')
            

        return redirect("bbs:index")

index   = IndexView.as_view()

バリデーションに失敗したら投稿エラーの旨を、バリデーション成功して保存を終えたら投稿内容を保存しましたと言う。リダイレクトを経由したとしても、問題なくHTML上で表示される。

メッセージが表示される。

結論

これを利用すれば、わざわざcontextにデータを渡さなくても1行でメッセージの追加ができる。

このようにリダイレクトする場合でもメッセージは共有されるので、まとめて表示される。

Ajaxを使用している場合でも有効。

参照元:https://docs.djangoproject.com/en/3.2/ref/contrib/messages/

【補足1】もっとシンプルに書きたい

毎度毎度、.add_message()などと書くのは時間がかかる。もっとシンプルに書くには下記のようにする。

messages.add_message(request, messages.INFO, "Hello world.")

#↑と↓は等価

messages.info(request, "Hello world.")

他にも下記のようにして呼び出しできる

messages.debug(request, '%s SQL statements were executed.' % count)
messages.info(request, 'Three credits remain in your account.')
messages.success(request, 'Profile details updated.')
messages.warning(request, 'Your account expires in three days.')
messages.error(request, 'Document deleted.')

詳しくは下記を参照。

https://docs.djangoproject.com/en/3.2/ref/contrib/messages/#using-messages-in-views-and-templates

【補足2】DjangoMessageFrameworkが動作しない時は?

まず、settings.pyINSTALLED_APPS

'django.contrib.messages'

が含まれているかどうかをチェック。Django 3.2であればデフォルトで書かれてある。

同じくMIDDLEWARE

'django.contrib.sessions.middleware.SessionMiddleware',
'django.contrib.messages.middleware.MessageMiddleware'

上記2つが含まれている事を確認。SessionMiddlewareMessageMiddlewareよりも前に書かれてある必要がある。これもデフォルトで書かれてある。

また同じくsettings.pyTEMPLATESOPTIONScontext_processorsにて、

'django.contrib.messages.context_processors.messages',

が含まれていることを確認。これもデフォルト。

つまり、とりわけsettings.pyを下手にいじっていなければデフォルトでMessageFrameworkは有効化されている。

【補足3】任意のエラーメッセージを表示させるには?

form.errorsをこのMessageFrameworkでそのまま表示させても、一般人には何のことかわからない。

だから、任意のエラーメッセージをフォームクラスに用意しておき、それをMessageFrameworkに表示させる。下記記事にて解説されてある。

【Django】任意のエラーメッセージを表示させる【forms.pyでerror_messagesを指定】

スポンサーリンク