Django 如何使用 FormMixin 将表单添加到 DetailView

Django how to add a form to a DetailView with FormMixin

我正在尝试向 DetailView 添加一个用于评论的表单。 DetailView 显示特定项目的注释。所以评论有一个外键是特定的注释,注释有一个特定项目的外键。

我正在尝试将 FormMixin 与 DetailView 一起使用。到目前为止,我还没有蜜蜂成功。目前我可以显示表单但它没有保存并且在终端中我看到以下错误 Method Not Allowed (POST): /projects/project/1/note/1/

我可以让它们单独工作,但不能与 DetailView 中的表单一起工作。

这是我的模型:

class ProjectNotes(models.Model):
    title = models.CharField(max_length=200)
    body = tinymce_models.HTMLField()
    date = models.DateField(auto_now_add=True)
    project = models.ForeignKey(Project, default=0, blank=True, on_delete=models.CASCADE, related_name='notes')

    def __str__(self):
        return self.title
class ProjectNoteComments(models.Model):
    body = tinymce_models.HTMLField()
    date = models.DateField(auto_now_add=True)
    projectnote = models.ForeignKey(ProjectNotes, default=0, blank=True, on_delete=models.CASCADE, related_name='comments')

观点:

class ProjectNotesDetailView(DetailView, FormMixin):
    model = ProjectNotes
    id = ProjectNotes.objects.only('id')
    template_name = 'company_accounts/project_note_detail.html'
    comments = ProjectNotes.comments
    form_class = NoteCommentForm

    def form_valid(self, form):
        projectnote = get_object_or_404(ProjectNotes, id=self.kwargs.get('pk'))
        comment = form.save(commit=False)
        comment.projectnote = projectnote
        comment.save()
        return super().form_valid(form)

    def get_success_url(self):
        return reverse('project_detail', args=[self.kwargs.get('pk')])

形式:

class NoteCommentForm(forms.ModelForm):
    class Meta:
        model = ProjectNoteComments
        fields =['body',]

        widgets = {
            'body': forms.TextInput(attrs={'class': 'form-control'})
        }

模板:

% extends 'base.html' %}
{% load crispy_forms_tags %}

{% block content %}
<div class="section-container container">
  <div class="project-entry">
    <h2>{{ projectnotes.title }}</h2>
    <p>{{ projectnotes.body | safe }}</p>
  </div>
  <div><b>Comments on {{projectnotes.title}}</b></div>
  {% if projectnotes.comments.all %}
    {% for comment in projectnotes.comments.all %}
        <div class="notecomments" style="padding: 10px;">
          
            {{ comment.body | safe }}
         
            
        </div>
      {% endfor %}
    {% else %}
    <p>No comments have been have been added yet.</p>
    {% endif %}
  
  <h2><a href="">add note</a></h2>
  
  <h1>Add Comment</h1>
  
   <form action="" method="post">
    {% csrf_token %}
    {{ form.media }}
    {{ form|crispy }}
    <input type="submit" value="save">
   </form>
  
{% endblock content %}

尝试在 ProjectNotesDetailView 中更改 DetailViewFormMixin 之间的顺序,然后实施 post 方法(由 `FormMixin 启用):

class ProjectNotesDetailView(FormMixin, DetailView):
    model = ProjectNotes
    id = ProjectNotes.objects.only('id')
    template_name = 'company_accounts/project_note_detail.html'
    comments = ProjectNotes.comments
    form_class = NoteCommentForm

    def form_valid(self, form):
        projectnote = get_object_or_404(ProjectNotes, id=self.kwargs.get('pk'))
        comment = form.save(commit=False)
        comment.projectnote = projectnote
        comment.save()
        return super().form_valid(form)

    def get_success_url(self):
        return reverse('project_detail', args=[self.kwargs.get('pk')])



    def post(self, request, *args, **kwargs):
        if not request.user.is_authenticated:
            return HttpResponseForbidden()
        self.object = self.get_object()
        form = self.get_form()
        if form.is_valid():
            return self.form_valid(form)
        else:
            return self.form_invalid(form)

查看如何 use formmixin with detailview(文档)。