我正在表单上创建附件。多个。一切都很好。问题是……我想"得到"在更新表单上显示这些附件,以便在表单获得批准时可以删除它们。事实证明,这是具有挑战性的。在某些情况下,我通过使用字典来获取作为初始数据所需的数据来预填充表单。所有的工作都如预期的那样,除了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}),
}
My View(CreateView)
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时…然后没有什么结果。
来自Facebook群组的一位名叫Matt Hoskins的绅士给了我这样的解释,经过3天的研究,我倾向于相信他。最后的基本前提是,这是不容易的,如果可能的话。接下来是另一种方法。下面是他更有说服力的总结....啊,我想我明白了。这根本不是表单集的问题,因为它发生了-关键是您试图为未绑定到实例的表单上的文件字段设置初始值。HTML文件输入不能使用初始值(这不是django的事情,这只是HTML/浏览器的工作方式),所以当django呈现文件字段小部件时,即使底层数据有初始值,也不会有初始值。django是如何处理带有文件字段的编辑模型实例的,如果用户在表单上选择了一个文件并提交,那么浏览器会将该文件作为字段的值提交,django会更新该实例上的字段,但是如果用户没有在表单上选择一个文件,那么浏览器根本不会提交该字段(这并不是说它会为字段提交一个空值)。在request.FILES中没有该字段的条目),当这种情况发生时,django不会更新实例上的字段(也就是说,它将保留其现有的值)。ClearableFileInput小部件向普通HTML文件输入添加了一个额外的HTML复选框字段,以允许清除要请求的现有值,并将显示任何现有值的名称,但文件输入本身仍然不能存储任何初始值。因此,当用户提交带有ClearableFileInput小部件的表单而不选择新文件时,请求中不会出现任何内容。该字段的FILES(关联的clear复选框字段的值将被提交,但这纯粹是告诉django是否清除该实例上的file字段)。所以如果你的内联表单集是用实际实例预先填充的,而不是初始化的,那么它就会工作,但是,因为你试图创建一个新的实例集基于现有的数据,现有的数据只是通过HTML表单带来的,因为HTML输入文件字段不能有一个初始值,然后你结束了什么字段用户不碰(事实上,你可以看到文件值显示由cleararablefileinput小部件从初始拾取是误导-这些值不是由表单提交)。希望这能说得通……稍后我会写更多的思考🙂