【问题标题】:Is it possible to prepopulate Django FormSets with FileFields?是否可以使用 FileFields 预填充 Django FormSets?
【发布时间】:2021-11-20 21:37:33
【问题描述】:

感谢您花时间阅读我的问题。这是我的问题。我正在表单上创建附件。多种的。一切都很好。这就是问题所在...我想在更新表单上“获取”这些附件,以便在表单获得批准后显示并删除它们。这被证明具有挑战性。在某些情况下,我通过使用字典来获取我需要作为初始数据的数据来预填充表单。除了 Django 引用的 FileFields 或 FieldFile 之外,一切都按预期工作。我已经阅读了一些关于 SO 的类似文章......但没有任何帮助。我了解安全问题,我不想“强制”上传..我只是想获取附件名称并将其复制到另一个模型。我的表单已提交,但未处理附件。

这是我的代码....

HTML...

<form method="POST" enctype="multipart/form-data" id="forms">

    {{ procedure_attachment_form.management_form }}

    {{ procedure_attachment_form.non_form_errors }}

    {% for fileform in procedure_attachment_form.forms %}

    {{ fileform.id }}

      <div class="inline {{ procedure_attachment_form.prefix }}">

          {{ fileform.attachments }}

            {% if procedure_attachment_form.non_form_errors %}

              <h3 class="spacer174">
                {{ procedure_attachment_form.non_form_errors }}
              </h3>

            {% endif %}

            {% if fileform.attachments.errors %}

              <h3 class="spacer174">
                {{ fileform.attachments.errors }}
              </h3>

            {% endif %}

          {{ fileform.procedure.as_hidden }}

          </div>

    {% endfor %}

我的表格...

class UpdateProcedureFilesForm(forms.ModelForm):

class Meta:
    model = UpdateProcedureFiles
    fields = ['attachments']
    widgets = {
        'attachments': ClearableFileInput(attrs={'multiple': True}),
    }

我的视图(创建视图)

class UpdateProcedureView(LoginRequiredMixin,CreateView):
    model = UpdateProcedure
    form_class = UpdateProcedureForm
    template_name = 'update_procedure.html'

def get(self, request, *args, **kwargs):
    self.object = self.get_object()
    context = self.get_context_data()
    form_class = self.get_form_class()
    form = self.get_form(form_class)

    dropdown = self.kwargs["pk"]
    attachments = ProcedureFiles.objects.filter(procedure_id=dropdown)

    attachment_listofdicts = []
    for attachment in attachments:
        attachment_dict = model_to_dict(attachment)
        del attachment_dict['id']
        del attachment_dict['procedure']
        del attachment_dict['archive_procedure']
        del attachment_dict['new_procedure']
        del attachment_dict['update_procedure']
        print(attachment_dict)
        attachment_listofdicts.append(attachment_dict)

    UpdateProcedureFileFormSet = inlineformset_factory(UpdateProcedure,
                                                       UpdateProcedureFiles,
                                                       form=UpdateProcedureFilesForm,
                                                       extra=len(attachment_listofdicts),
                                                       can_order=True,
                                                       min_num=0,
                                                       validate_min=True)

    procedure_attachment_form = UpdateProcedureFileFormSet(initial=attachment_listofdicts)
    # print(procedure_attachment_form)

    return self.render_to_response(
        self.get_context_data(
            form=form,
            procedure_attachment_form=procedure_attachment_form,
        )
    )


def get_object(self, queryset=None):
    return get_object_or_404(Procedure, id=self.kwargs['pk'])

def get_initial(self):
    initial = super(UpdateProcedureView, self).get_initial()
    procedure = Procedure.objects.get(pk=self.kwargs["pk"])
    initial = procedure.__dict__.copy()
    department = self.request.user.userprofile.department_access.all()

    initial.update({
                      "name": procedure.name,  
    })

    if procedure.department in self.request.user.userprofile.department_access.all() and procedure.access_level == "Default" :
        return initial
    else:
        raise Http404

def get_context_data(self, **kwargs):
    context = super(UpdateProcedureView, self).get_context_data(**kwargs)
    pk=self.kwargs["pk"]
    if self.request.POST:
        context["attachments"] = UpdateProcedureFileFormSet(self.request.POST,self.request.FILES)
    else:
        context["attachments"] = UpdateProcedureFileFormSet()
    return context

def form_valid(self, form, procedure_attachment_form):
    self.object = form.save()
    procedure_attachment_form.instance = self.object

    instance = form.save()
    return super(UpdateProcedureView, self).form_valid(form)

def form_invalid(self, form, procedure_attachment_form):
    return self.render_to_response(
        self.get_context_data(form=form,
                              procedure_attachment_form=procedure_attachment_form,
                              ))

def post(self, request, *args, **kwargs):
    print(request.POST)
    if "cancel" in request.POST:
        return HttpResponseRedirect(reverse('Procedures:procedure_main_menu'))
    else:
        self.object = None
        form_class = self.get_form_class()
        form = self.get_form(form_class)
        user = request.user
        userprofile = request.user
        procedure_attachment_form = UpdateProcedureFileFormSet(self.request.POST,self.request.FILES)
        files = request.FILES.getlist('attachments') #field name in model
        if (form.is_valid() and procedure_attachment_form.is_valid()):
            procedure_instance = form.save(commit=False)
            procedure_instance.user = user
            procedure_instance.save()
            for f in files:
                file_instance = UpdateProcedureFiles(attachments=f, update_procedure=procedure_instance)
                file_instance.save()
            return self.form_valid(form, procedure_attachment_form)
        else:
            return self.form_invalid(form, procedure_attachment_form)

同样,这一切都有效。唯一的例外是当涉及 FileFields 时......然后 nada。提前感谢您的任何想法。

【问题讨论】:

标签: django django-models django-views django-forms django-templates


【解决方案1】:

来自 Facebook 群组的一位名叫 Matt Hoskins 的好心人向我提供了这个解释,经过 3 天的研究,我倾向于相信他。以基本前提结束这一点,即如果可能的话,这并不容易。转向不同的方法。这是他更有说服力的总结……啊,我想我明白了这个问题。这根本不是表单集的问题,因为它发生了 - 关键是您正在尝试为未绑定到实例的表单上的文件字段设置初始值。 HTML 文件输入不能采用初始值(这不是 django 的事情,这正是 HTML/浏览器的工作方式),因此当 django 呈现文件字段小部件时,即使底层数据有初始值,也永远不会有初始值。 django 如何使用文件字段编辑模型实例是,如果用户在表单上选择一个文件并提交,那么浏览器将提交该文件作为字段的值,django 将更新实例上的字段,但是如果用户不会在表单上选择文件,那么浏览器根本不会提交该字段(并不是它会为该字段提交一个空值,只是在 request.FILES 中没有该字段的条目)和当这种情况发生时,django 不会更新实例上的字段(即它将保留其现有值)。 ClearableFileInput 小部件向纯 HTML 文件输入添加了一个额外的 HTML 复选框字段,以允许清除要请求的现有值并显示任何现有值的名称,但文件输入本身仍然不能存储任何初始值.因此,当用户在没有选择新文件的情况下提交带有 ClearableFileInput 小部件的表单时,该字段的 request.FILES 中不会出现任何内容(关联的清除复选框字段的值将被提交,但这纯粹是告诉 django 是否清除文件实例上的字段)。 因此,如果您的内联表单集预先填充了实际实例而不是初始实例,那么它会起作用,但是因为您正在尝试基于现有数据创建一组新实例,并且现有数据仅由 HTML 带来表单,并且因为 html 输入文件字段不能具有初始值,所以对于用户未触摸的字段,您最终将一无所获(您可以看到 ClearableFileInput 小部件显示的文件值从初始值中提取是误导性的 - 那些值不是由表单提交的)。 希望这是有道理的......我很快就会写更多的思考?

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-01-27
    • 2021-11-25
    • 1970-01-01
    • 2012-08-22
    • 1970-01-01
    • 2018-11-04
    相关资源
    最近更新 更多