簡體   English   中英

如何使用 django formset_factories?

[英]How do I use django formset_factories?

我正在嘗試在一個基於 django 的項目/網站上工作,該項目/網站記錄了最后帶走我們狗的人。 我對 django 很陌生,所以請忍受缺乏該領域的知識。 我目前設置了我的項目,以便我有基於用戶/家庭成員、家庭的狗、帖子(何時和誰帶狗的日志)和動作的模型,這是一個使用狗的子 class 和將模型作為外鍵發布,以查看狗在被帶出時是否撒尿或拉便。 我堅持的部分是試圖弄清楚我如何創建帖子表單。 由於我們有兩只狗,我需要表單/發布頁面來為存在的每只狗 model 顯示一組復選框(用於小便和便便操作)。 雖然我可以成功顯示一個動作集,但我遇到了嘗試顯示正確數量的動作集以及發布動作集和帖子本身的困難。 我一直在嘗試使用 formset_factory function 但我對如何正確地將其轉換為 function 感到困惑。 任何人都可以幫忙嗎? 我已經被這個問題困擾了很長一段時間,任何支持都將不勝感激。

模型.py:

from django.db import models
from django.utils import timezone
from django.contrib.auth.models import User
from django.urls import reverse

class Dog(models.Model):
    name = models.CharField(max_length = 20)

class Post(models.Model):
    walker = models.ForeignKey(User, on_delete = models.CASCADE)
    time_posted = models.DateTimeField(default = timezone.now)
    issues = models.TextField(max_length = 300)

class Action(models.Model):
    post = models.ForeignKey(Post, on_delete = models.CASCADE)
    dog = models.ForeignKey(Dog, on_delete = models.CASCADE)
    peed = models.BooleanField(default = False)
    pooped = models.BooleanField(default = False)

forms.py:

from django import forms
from django.forms import formset_factory
from dog_log_app.models import *

class Log_Form(forms.ModelForm):  
    class Meta:  
        model = Post
        fields = ("issues",)

class Action_Form(forms.Form):
    peed = forms.BooleanField(initial = False, required = False)
    pooped = forms.BooleanField(initial = False, required = False)

視圖.py:

from django.shortcuts import render
from django.forms import formset_factory, modelformset_factory
from .models import Post, Dog, Action
from dog_log_app.forms import *
from django.contrib.auth.models import User
from django.http import HttpResponse, HttpResponseRedirect

def home(request):
    context = {
        'posts': Post.objects.all().order_by('-time_posted'),
        'actions': Action.objects.all(),
        'dogs': Dog.objects.all()
    }
    return render(request, 'dog_log_app/home.html', context)

def post(request):
    form_post = Log_Form(request.POST or None)
    form_actions = modelformset_factory(Action, fields = ('peed', 'pooped'), extra = Dog.objects.count())

    if request.method == 'POST':
        form_post = Log_Form(request.POST)


        if form_post.is_valid() and form_actions.is_valid():
            post_save = form_post.save(commit = False)
            post_save.walker = request.user
            post_save.save()
            form_actions.save()

            return HttpResponseRedirect('..')
    
    context = {
        'post': form_post,
        'action': formset_factory(Action_Form, extra = Dog.objects.count()),
        'dogs': Dog.objects.all()
    }
    return render(request, 'dog_log_app/post_form.html', context)

post_form.html:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Post</title>
</head>
<body>
    <div>
        <form method="POST">
            {% csrf_token %}
            <fieldset>
                <legend>Create Post</legend>
                <h3>{{dogs.name}}</h3>
                <p>{{action.as_p}}</p>
                <p>{{post.as_p}}</p>
            </fieldset>
            <div>
                <button type="submit" value="Ok">Post</button>
            </div>
        </form>
    </div>
</body>
</html>

您可以嘗試使用 ajax 添加您想要的金額,我通常使用以下表格:

from django.forms import ModelForm
from django.forms.models import inlineformset_factory

class PostForm(ModelForm):
    class Meta:
        model = Post
        fields = (
            'issues',
        )

class ActionForm(ModelForm):
    class Meta:
        model = Action
        fields = (
            'peed', 'pooped'
        )

ActionFormSet = inlineformset_factory(
    Post, Action, form=ActionForm,
    fields = ['peed', 'pooped'], extra=1, can_delete=True,
)

創建 forms 后,我們繼續查看視圖,為方便起見,這些視圖將基於類:

from django.db import transaction
from django.views.generic import CreateView, UpdateView

from .models import Post
from .forms import PostForm, ActionFormSet

class PostCreateView(CreateView):
    template_name = 'your template'
    form_class = PostForm
    success_url = reverse_lazy('your url')

    def get_context_data(self, **kwargs):
        data = super().get_context_data(**kwargs)
        if self.request.POST:
            data['formset'] = ActionFormSet(self.request.POST)
        else:
            data['formset'] = ActionFormSet()

        return data

    def form_valid(self, form):
        context = self.get_context_data()
        formset = context['formset']

        with transaction.atomic():
            self.object = form.save()

            if formset.is_valid():
                formset.instance = self.object
                formset.instance.walker = self.request.user
                formset.save()

        return super().form_valid(form)


class PostUpdateView(UpdateView):
    template_name = 'your template'
    form_class = PostForm
    model = Post
    success_url = reverse_lazy('your url')

    def get_context_data(self, **kwargs):
        data = super().get_context_data(**kwargs)
        if self.request.POST:
            data['formset'] = ActionFormSet(
                self.request.POST, instance=self.object)
        else:
            data['formset'] = ActionFormSet(instance=self.object)

        return data

    def form_valid(self, form):
        context = self.get_context_data()
        formset = context['formset']

        with transaction.atomic():
            self.object = form.save()

            if formset.is_valid():
                formset.instance = self.object
                formset.save()

        return super().form_valid(form)

並顯示信息,您可以使用 jquery 庫獲取有關保存的動態信息:在此處輸入鏈接描述這樣配置就容易多了

一個示例模板是:

{% extends "base.html" %}
{% load static %}

{% block content %}
    <div class="container">
        <div class="card">
            <div class="card-content">
                <h2>Agregar Proyecto</h2>
                <div class="col-md-4">
                    <form id="form-container" method="post">
                        {% csrf_token %}
                        {{ form }}
                        <h2>Actions</h2>
                        <table>
                            {{ formset.management_form }}
                            {% for form in formset %}
                                {% if forloop.first %}
                                    <thead>
                                        <tr>
                                            {% for field in form.visible_fields %}
                                                <th>{{ field.label|capfirst }}</th>
                                            {% endfor %}
                                        </tr>
                                    </thead>
                                {% endif %}
                                <tr id="projects_data">
                                    {% for field in form.visible_fields %}
                                        <td>
                                            {# Include the hidden fields in the form #}
                                            {% if forloop.first %}
                                                {% for hidden in form.hidden_fields %}
                                                    {{ hidden }}
                                                {% endfor %}
                                            {% endif %}
                                            {{ field.errors.as_ul }}
                                            {{ field }}
                                        </td>
                                    {% endfor %}
                                </tr>
                            {% endfor %}
                        </table>
                        <button type="submit" class="btn btn-success">{{ message }}</button>
                        <a href="{{ request.META.HTTP_REFERER }}" class="btn btn-danger">Cancel</a>
                    </form>
                </div>
            </div>
        </div>
    </div>
{% endblock %}

{% block script %}
    <script type="text/javascript">
        $(function () {
            $('#projects_data').formset({
                prefix: '{{ formset.prefix }}',
                addText: 'create',
                deleteText: 'remove',
            });
        })
    </script>
{% endblock %}

並以這種方式動態添加數據。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM