0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

複数のModelFormを一括登録する

Last updated at Posted at 2023-03-16

はじめに

おはこんばんちは。@cosumi77と申します。

本記事はdjango初心者が記載しました。
誤りがある場合などは是非ご指摘頂けるとありがたいです。

本記事の目的

下図のように異なるテーブルの複数のModelFormを一括で登録したい場合があるかと思います。
(同じテーブルのModelFormを複数登録する場合はModelFormSetを使えば良い)
1.png

これを実装しようと色々調べてみたのですがヒットしなかったため、情報共有しようと思いました。
また、この実装を行う際に注意すべき事がありますので、そちらについても記載します。

実装

まずはモデルを作成します。
(何故かカラム名にナンバリングしてある理由は後述)

models.py
from django.db import models

class test1(models.Model):
    name1 = models.CharField(max_length=10)
    title1 = models.CharField(max_length=100)
    memo1 = models.CharField(max_length=100)

class test2(models.Model):
    name2 = models.CharField(max_length=10)
    title2 = models.CharField(max_length=100)
    memo2 = models.CharField(max_length=100)

続いて、ModelFormを実装します。

forms.py
from django import forms
from .models import test1, test2

class Test1ModelForm(forms.ModelForm):

    class Meta:
        model = test1
        fields = '__all__' 

class Test2ModelForm(forms.ModelForm):

    class Meta:
        model = test2
        fields = '__all__' 

Viewは書き方が怪しいですが、一応動くので載せておきます。
(もっとこうしたら良い等ありましたら、是非ともコメントください!)

複数のFormをcontextとして渡します。

from django.views.generic.edit import View
from django.template.response import TemplateResponse
from django.http.response import HttpResponseRedirect
from .forms import (
    Test1ModelForm, Test2ModelForm,
)
from django.urls import reverse

class testView(View):

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

        #複数のフォームをコンテキストとして渡す
        context = {
            'main_form' : Test1ModelForm(),
            'detail_form' : Test2ModelForm(),
        }
    
        return TemplateResponse(request, 'test.html', context)

    def post(self, request, *args, **kwargs):
        form1 = Test1ModelForm(request.POST)
        form2 = Test2ModelForm(request.POST)

        #全てのフォームのvalidationが通ったらsave
        if form1.is_valid() and form2.is_valid():
            form1.save()
            form2.save()
        
        return HttpResponseRedirect(reverse('test'))

templateでは各々のformを表示するように記述します。

template.html
<form method="post">
    {% csrf_token %}
        <table>
            {{ main_form.as_table }}
        </table>
        <br>

        <table>
            {{ detail_form.as_table }}
        </table>

    <input type="submit" value="登録">
</form>

最後にpathを通します。

urls.py
from django.contrib import admin
from django.urls import path

from test1.views import testView

urlpatterns = [
    path('admin/', admin.site.urls),
    path('test/', testView.as_view(),name='test')
]

後はsettings.pyにテンプレートの記述を追加すれば完了です(説明は省略)。
disp2.png

動作検証

実際にフォームに入力して、各々のテーブルに登録されるか確認してみます。
disp3.png

登録ボタンを押下すると、きちんと登録出来てました。
4.png
5.png

実装にあたっての注意点

以下のようにカラム名が重複している場合、一番最後に定義したmodelを対象したmodelFormに入力された値で更新がかかります。
要は、同じ名前だと値が上書きされてしまうようです。

models.py
from django.db import models

class test1(models.Model):
    name = models.CharField(max_length=10)
    title = models.CharField(max_length=100)
    memo = models.CharField(max_length=100)

#各カラム名が全く同じ
class test2(models.Model):
    name = models.CharField(max_length=10)
    title = models.CharField(max_length=100)
    memo = models.CharField(max_length=100)

具体的には、例えば以下のように入力した後に登録してみると、どちらのテーブルにもddd、eee、fffが登録されます。

6.png

これは以下のようにform側でlabelの文言を変えるだけでは解消しません。

forms.py
from django import forms
from .models import test1, test2

class Test1ModelForm(forms.ModelForm):

    class Meta:
        model = test1
        fields = '__all__' 

class Test2ModelForm(forms.ModelForm):


    class Meta:
        model = test2
        fields = '__all__' 
        
        #★ラベルで別名を付与
        labels={
            'name':'name2',
            'title':'title2',
            'memo':'memo2',
        }

色々調べてみたのですが、このあたりの良い解決策は見当たりませんでした。
というわけで、model上での定義名を工夫し、カラム名が一意になるように定義する必要があるというのが私の結論です。
(良い方法がありましたら教えてくださいm(_ _)m)

0
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?