Python 使用formset将许多文件上传到一个记录中,这不是';不太管用

Python 使用formset将许多文件上传到一个记录中,这不是';不太管用,python,django,django-forms,formset,inline-formset,Python,Django,Django Forms,Formset,Inline Formset,我在网上使用了尽可能多的例子,试图得到我的两个简单模型: class Technical_Entry(models.Model): category = models.ForeignKey(Category, on_delete=models.CASCADE) ema = models.ForeignKey(EMA, on_delete=models.CASCADE) system = models.ForeignKey('System', on_delete=model

我在网上使用了尽可能多的例子,试图得到我的两个简单模型:

class Technical_Entry(models.Model):
    category = models.ForeignKey(Category, on_delete=models.CASCADE)
    ema = models.ForeignKey(EMA, on_delete=models.CASCADE)
    system = models.ForeignKey('System', on_delete=models.CASCADE) # are SYSTEMS RELATED TO SUBSYSTEMS OR JUST TWO GROUPS?  
    sub_system = models.ForeignKey(SubSystem, on_delete=models.CASCADE)

    drawing_number = models.CharField(max_length=200)
    drawing_title = models.CharField(max_length=255)
    engineer = models.CharField(max_length=200)
    vendor = models.ForeignKey(Vendor, on_delete=models.CASCADE)

    date_drawn = models.DateField()
    ab = models.BooleanField()



class Technical_Entry_Files(models.Model):
    tech_entry = models.ForeignKey(Technical_Entry, on_delete=models.CASCADE)
    file = models.FileField(upload_to='techdb/files/')

    def __str__(self):
        return self.tech_entry.drawing_number
使用表单集上载。虽然页面“显示”基本正确,但它不会在技术输入文件模型中创建记录

相关表格.py:

class FileUploadForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(FileUploadForm, self).__init__(*args, **kwargs)
        self.helper = FormHelper()  
        self.helper.form_class = 'form-horizontal'
        self.helper.label_class = 'col-lg-4'
        self.helper.field_class = 'col-lg-8'
    
    class Meta:
        model = Technical_Entry_Files
        fields = ('file',)


TechFileFormSet  = inlineformset_factory(Technical_Entry, Technical_Entry_Files, form=FileUploadForm, extra=1)



class Technical_EntryForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(Technical_EntryForm, self).__init__(*args, **kwargs)
        self.helper = FormHelper()  
        self.helper.form_class = 'form-horizontal'
        self.helper.label_class = 'col-lg-4'
        self.helper.field_class = 'col-lg-8'
        self.helper.add_input(Submit('submit', 'Submit'))

    class Meta:
        model = Technical_Entry
        fields = ('category', 'ema', 'system', 'sub_system', 'drawing_number', 'drawing_title', 'engineer', 'vendor', 'date_drawn', 'ab')


        widgets = {
            'date_drawn':DateInput(attrs={
            'class':'datepicker form-control',        
            'id' : 'datetimepicker2',
            'tabindex' : '1',
            'placeholder' : 'MM/DD/YYYY hh:mm',          
            'autocomplete':'off',
            }, format='%m/%d/%Y'),

            'system' : Select(attrs={'tabindex':'2'}),
        }
相关视图.py:

class TechEntryUpdateView(LoginRequiredMixin, UpdateView):                                                                                             

    model = Technical_Entry
    form_class = Technical_EntryForm
    template_name = 'techdb/tech_entry_form.html'
    success_url = '/'
                                                                                             
    log_entry_class = Technical_EntryForm(Technical_Entry)
                                                                                         

    def get_context_data(self, **kwargs):
        context = super(TechEntryUpdateView, self).get_context_data(**kwargs)

        if self.request.POST:
            context["file_upload"] = TechFileFormSet(self.request.POST, self.request.FILES,instance=self.object)
        else:
            context["file_upload"] = TechFileFormSet(instance=self.object)

#        entry = context['object'] 
 #       context['entry_id'] = entry.id
  #      theEntry = Technical_Entry.objects.get(pk=entry.id) 
#        entry_form = Technical_EntryForm(instance=theEntry)
#        context['entry_form'] = entry_form         
        return context

    def form_valid(self, form):
        context = self.get_context_data()
        file_upload = context["file_upload"]
        self.object = form.save()
        if file_upload.is_valid():
            file_upload.instance =self.object
            file_upload.save()
        return super().form_valid(form)

class TechEntryCreateView(LoginRequiredMixin, CreateView):
    print ("we are here")
    model = Technical_Entry
    form_class = Technical_EntryForm
    template_name = 'techdb/tech_entry_form.html'
    print(template_name)
    success_url = '/techentry_add'

    def get_context_data(self, **kwargs):
        data = super(TechEntryCreateView, self).get_context_data(**kwargs)
        if self.request.POST:
            data['file_upload'] = TechFileFormSet(self.request.POST, self.request.FILES)
        else:
            data['file_upload'] = TechFileFormSet()
        return data


    def form_valid(self, form):
        context =self.get_context_data()
        file_upload = context['file_upload']
        with transaction.atomic():
            self.object = form.save()

            if file_upload.is_valid():
                file_upload.instance =self.object
                file_upload.save()
        return super(TechEntryCreateView, self).form_valid(form)
以及tech_entry_form.html:

{% extends 'base.html' %}


{% load static %}

{% block page-js %}

<script>
    $('.link-formset').formset({
        addText: 'add file',
        deleteText: 'remove',        
    });
</script>
{% endblock %}

{% block content %}

<main role="main" class="container">

      <div class="starter-template">
        <h1>New Tech Entry</h1>
      </div>

    <h2> Details of Technical Entry </h2>
     <div class="row">
      <div class="col-sm">
       <form action="" method="post" enctype="multipart/form-data">{% csrf_token %}
                {{ form.as_p }}

                <h2> Files </h2>
                {{ file_upload.management_form }}
                {% for upload_form in file_upload.forms %}
                  <div class="link-formset">
                    {{ upload_form.file }}        
                  </div>
                 {% endfor %}

               
                <input type="submit" value="Save"/><a href="{% url 'tech_database:index' %}">back to the list</a>
            </form>
        </div>
       </div>
     </div>
    </main><!-- /.container -->


{% endblock %}
{%extends'base.html%}
{%load static%}
{%block page js%}
$('.link formset').formset({
addText:“添加文件”,
deleteText:“删除”,
});
{%endblock%}
{%block content%}
新技术进入
技术条目的详细信息
{%csrf_令牌%}
{{form.as_p}}
文件夹
{{file_upload.management_form}
{文件{u upload.forms%}中的上传表格为%
{{upload_form.file}
{%endfor%}
{%endblock%}
它会保存条目,但不会保存上传的文件,我也看不到任何错误

没有techdb/files文件夹(也许我必须创建它?),但它肯定不会在任何地方失败…创建技术输入记录,但不是技术输入文件记录,磁盘上也没有添加任何内容

此外,这真的是最大的一块…它只允许我上传一个文件,即使表格应该允许许多文件到一个技术条目?(也许我需要踢一些java脚本)


我发现的所有示例要么不使用基于模型的表单、基于类的视图,要么看起来有些过火。我只需要这些简单的模型,让我上传许多文件到一个技术条目。我原以为我已经很接近了,但恐怕我离实现这个目标还差一步:

1。我注意到一件事是你的
没有

<form action="" method="post" enctype="multipart/form-data">
不确定上述内容是否应纳入您的观点

以下内容来自文档:在基于函数的视图中

if request.method == 'POST':
        formset = ArticleFormSet(request.POST, request.FILES)
文件的另一部分:说明:

当Django处理文件上载时,文件数据最终会放在request.FILES中


添加了它,仍然没有骰子。更新了另一个点。谢谢你,额外的眼睛帮助了。这成功了!所以现在我必须弄清楚如何让我列出已经是条目一部分的文件,并允许我添加任意数量的文件…这我真的不知道从哪里开始。但至少我知道上传是在一个表单集中进行的!很酷,谢谢
formset=ImageFormSet(request.POST、request.FILES、instance=item)
。此处的
instance
应该是一个用于列出已作为条目一部分的文件的实例
TechFileFormSet=inlineformset\u工厂(Technical\u Entry,Technical\u Entry\u Files,form=FileUploadForm,extra=1,max\u num=5)
为了“添加任意数量的文件”,您可能需要签出
extra
max\u num
。需要注意的一点是,这些将为您提供更多表单。但是每个表单(我记得)只能包含一个文件。正因为如此,我选择了js库来上传图片。js库可能是我要走的路,我正在尝试一种不同的方法。我认为真正不起作用的是我对jquery表单集的链接表单集调用。我使用的是基于类的视图,所以我想我的UpdateView可能会关心instance=item(我的意思是它可能是一个已经为该技术条目上传的文件列表…所以仍然在这里查找要做的事情。这是我第一次尝试类似的事情,很遗憾。
if request.method == 'POST':
        formset = ArticleFormSet(request.POST, request.FILES)