Использование view/form в Django для загрузки нескольких файлов
Класс ниже используется для создания формы в Django и получения необходимой информации. Проблема в том, что он предлагает только один файл для загрузки. Мне нужно загрузить несколько файлов. Я использую Crispy forms.
Мой упрощенный view.py выглядит так:
class PostCreateView(LoginRequiredMixin, CreateView):
model = Post
fields = ['title', 'file_1']
def form_valid(self, form):
form.instance.author = self.request.user
return super().form_valid(form)
HTML-код:
{% extends "blog/base.html" %}
{% load crispy_forms_tags %}
{% block content %}
<div class="content-section">
<form method="POST" enctype="multipart/form-data">
{% csrf_token %}
<fieldset class="form-group">
<legend class="border-bottom mb-4">Fill form</legend>
{{ form|crispy }}
</fieldset>
<div class="form-group">
<button class="btn btn-outline-info" type="submit">Submit</button>
</div>
</form>
</div>
{% endblock content %}
Когда я осматриваю страницу, объект выглядит следующим образом:
<input type="file" name="file_1" class="clearablefileinput form-control-file" id="id_file_1">
Я хочу, чтобы он содержал множество атрибутов. Как я могу этого добиться? Я не могу заставить его работать, используя документацию (https://docs.djangoproject.com/en/3.2/topics/http/file-uploads/).
Я пробовал:
widgets = {'file_1': form.ClearableFileInput(attrs={'multiple': True})}
form.instance.file_1 = form.FileField(widget=form.ClearableFileInput(attrs={'multiple':True}))
form.instance.file_1 = form.FileField(widget=form.FileInput(attrs={'multiple': True}))
Мой models.py
file_1 = models.FileField(blank=True, upload_to='PN_files/%Y/%m/%d/', verbose_name="File 1", validators=[validate_file_size], help_text="Allowed size is 50MB")
Я не могу найти пример реализации загрузки нескольких файлов, который можно было бы реализовать в моем классе.
Чтобы ответить на ваш вопрос, я попытаюсь создать внешний ключ к посту и буду использовать функцию views
app/models.py:
class Post(models.Model):
#add your all other fields here execept (files)
class File(models.Model):
post = models.ForeignKey(Post,on_delete=models.CASCADE)
file = models.FileField(blank=True, upload_to='PN_files/%Y/%m/%d/', verbose_name="Files", validators=[validate_file_size], help_text="Allowed size is 50MB")
app/forms.py:
from .models import Post,File
class PostForm(forms.ModelForm):
class Meta:
model = Post
fields = [#the field that you want to rendered]
class PostFileForm(PostForm): #extending form
file = forms.FileField(widget=forms.ClearableFileInput(attrs={'multiple': True}))
class Meta(PostForm.Meta):
fields = PostForm.Meta.fields + ['file',]
app/views.py:
from .forms import PostFileForm
from .models import File
def postcreate(request):
if request.method == 'POST':
form=PostFileForm(request.POST or None,request.FILES or None)
files = request.FILES.getlist('file')
if form.is_valid():
post = form.save(commit=False)
post.author = request.user
#add everything you want to add here
post.save()
if files:#check if user has uploaded some files
for f in files:
File.objects.create(post=post,file=f)
return redirect('the-name-of-the-view')
else:
form = PostFileForm()
return render(request,'the-template-you-want-to-rendered-',{'form':form})
С помощью этого вы можете загрузить много файлов, которые вы хотите. И если вы не хотите, чтобы файл был обязательным, вы можете добавить "required=False" внутри PostFileForm.