form = AddItemForm(request.POST, request.FILES)
if form.is_valid()
do_stuff
return render_to_response(blah.html, {'form':form})
现已形成将与字段的原始值沿着错误信息后保留一个文件,但它不保留选定的文件 如何保持选定的文件如果表单验证失败?如何使一个Django表单验证失败
form = AddItemForm(request.POST, request.FILES)
if form.is_valid()
do_stuff
return render_to_response(blah.html, {'form':form})
现已形成将与字段的原始值沿着错误信息后保留一个文件,但它不保留选定的文件 如何保持选定的文件如果表单验证失败?如何使一个Django表单验证失败
如果您使用的是modelForm,并且您成功保存了具有fileField的该模型的新实例,则Django将仅将文件保存到磁盘上。
在你的情况下,你需要做的是从request.FILES字典中获取文件,并将其保存到磁盘上。它应该看起来像这样。
input_file = request.FILES['fileformfieldname']
new_file = open('/path/to/file.xxx')
new_file.write(input_file.read())
现在你已经保存到磁盘上的文件,你只需要记住的文件路径,这样当用户重新提交失败的形式,你可以再次打开它。
你想做什么的问题是,出于安全原因,浏览器将不允许文件输入框在页面加载时具有预选值。即使它仅保留来自同一页面的前一个实例的值,情况也是如此。 Django没有办法改变这一点。
如果您想避免要求用户重新选择并重新上传文件,即使验证失败,也需要保存上传的文件,然后用某些内容替换文件输入字段以表明您已经有数据。如果用户想要放入不同的文件,您可能还需要一个运行某个JavaScript的按钮来重新启用文件字段。我不认为Django为此提供了任何机制,因此您必须自己编写代码。
有点棘手,但这里是逻辑。
在第一次加载时,隐藏字段为空,因此没有任何反应。
在选择文件时,它会将路径+名称保存在隐藏字段中。
在第二负载,隐藏字段不为空,那么它将设置文件路径
这样一来,所有的脏活累活发生在浏览器,除非你拥有有效的形式,你不必保存文件上服务器。
这不会起作用。 Javascript无法更改输入类型=文件值,或者恶意用户可以将随机猜测的文件名从计算机下载。 – 2011-04-28 22:13:19
好的,所以我首先提出并实现了Narendra的解决方案,然后才意识到它无法工作,因为JavaScript无法设置输入类型=“文件”值。请参阅: http://www.w3schools.com/jsref/prop_fileupload_value.asp
但这里有一个可行的解决方案:形式
好的,这里是一个过程的草图,它适用于我试图用一个作者的电子邮件地址(选作电子邮件有时不验证)保存书籍pdf的例子。
models.py
class Book(models.Model):
pdf = models.FileField("PDF", upload_to="books/")
author_email = models.EmailField("Author Email")
class TempPDF(models.Model):
pdf = models.FileField("PDF", upload_to="books/")
forms.py
from project_name.app_name.models import Book, TempPDF
from django.forms import ModelForm
from django.contrib.admin.widgets import AdminFileWidget
class BookForm(ModelForm):
class Meta:
model = Book
exclude = ['pdf',]
class TempPDFForm(ModelForm):
class Meta:
model = TempPDF
widgets = dict(pdf = AdminFileWidget)
# The AdminFileWidget will give a link to the saved file as well as give a prompt to change the file.
# Note: be safe and don't let malicious users upload php/cgi scripts that your webserver may try running.
views.py
def new_book_form(request):
if request.method == 'POST':
## Always try saving file.
try:
temp_pdf_inst = TempPDF.objects.get(id=request.POST.has_key('temp_pdf_id'))
except: ## should really catch specific errors, but being quick
temp_pdf_inst = None
temp_pdf_form = TempPDFForm(request.POST, request.FILES, instance=temp_pdf_inst, prefix='temp_pdf')
if temp_pdf_form.is_valid() and len(request.FILES) > 0:
temp_pdf_inst = temp_pdf_form.save()
temp_pdf_id = temp_pdf_inst.id
book_form = BookForm(request.POST, prefix='book')
if book_form.is_valid(): # All validation rules pass
book = book_form.save(commit=False)
book.pdf = temp_pdf_inst.pdf
book.save()
if temp_pdf_inst != None:
temp_pdf_inst.delete()
return HttpResponseRedirect('/thanks/') # Redirect after POST
else:
book_form = BookForm() # An unbound form
temp_pdf_form = TempPDFForm()
temp_pdf_id = None
return render_to_response('bookform_template.html',
dict(book_form = book_form,
temp_pdf_form = temp_pdf_form,
temp_pdf_id = temp_pdf_id)
)
bookform_template.html
<table>
{{ book_form }}
{{ temp_pdf_form }}
<input type="hidden" name="temp_pdf_id" value="{{ temp_pdf_id }}">
</table>
insallation
pip install django-file-resubmit
settings.py
INSTALLED_APPS = {
...
'sorl.thumbnail',
'file_resubmit',
...
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
},
"file_resubmit": {
'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache',
"LOCATION": project_path('data/cache/file_resubmit')
},
}
使用
from django.contrib import admin
from file_resubmit.admin import AdminResubmitMixin
class ModelAdmin(AdminResubmitMixin, ModelAdmin):
pass
或
from django.forms import ModelForm
from file_resubmit.admin import AdminResubmitImageWidget, AdminResubmitFileWidget
class MyModelForm(forms.ModelForm)
class Meta:
model = MyModel
widgets = {
'picture': AdminResubmitImageWidget,
'file': AdminResubmitFileWidget,
}
我第一次尝试上传时,文件消失。第二次尝试是成功的。你有什么解释吗? – Benjamin 2016-11-08 10:24:21
我有同样的问题。如果我使用模型字段,当验证失败时是否真的必须保存文件,以便验证成功时可以上传它?然后,我必须做一些垃圾回收,以删除已上传的文件,但用户从未成功完成表单。我确信Django有这样做的好方法! – PhoebeB 2010-09-07 09:35:59