Поле формы файла Django исчезает при ошибке формы - PullRequest
15 голосов
/ 25 октября 2011

Вот проблема, у меня есть форма Django, содержащая поле файла, а именно:

photo = forms.FileField(help_text="Please attach a photo", required=False)

Если форма проверена, поле «Файл» ограничено и сохранено правильно. Проблема в том, что пользователь заполняет всю форму и не проверяет: путь к выбранному файлу исчезает .

Итак, если пользователь не осознает этого, он / она исправляет ошибки других полей и отправляет их снова - на этот раз без фотографии.

На всякий случай, форма создается в виде, подобном этому:

ProfileForm(request.POST or None, request.FILES or None)

и HTML-код:

<div id="uniform-id_photo" class="uploader">
  <input id="id_photo" class="clearablefileinput" type="file" name="photo" size="19" style="opacity: 0;">
  <span class="filename" style="-moz-user-select: none;">No file selected</span>
  <span class="action" style="-moz-user-select: none;">Choose File</span>
</div>

У кого-нибудь была такая же проблема раньше? Есть мысли по поводу решения? :)

Спасибо!

Ответы [ 2 ]

26 голосов
/ 25 октября 2011

К сожалению, это проблема (на самом деле функция безопасности), возникающая в браузерах, и она не решаема как таковая. Браузеры не позволят вам указать начальное значение для входных файлов, и вы ничего не сможете сделать, чтобы обойти это.

Причина в том, что если веб-сайт может сделать это, он откроет вектор, который позволит любому веб-сайту украсть любой файл на вашем компьютере, угадав пути к файлам - они могут просто запустить скрипт в фоновом режиме, который пытался отправлять интересные файлы обратно на сервер.

Единственный обходной путь - фактически сохранить загруженный файл на сервере независимо от того, проверяет ли форма, а затем, когда вы возвращаете форму и ошибки пользователю, указываете, что вы получили файл и что он должен заполнять только это поле, чтобы заменить его.

0 голосов
/ 16 декабря 2014

Я пишу какое-то решение:

class CustomClearableFileInput(ClearableFileInput):

def render(self, name, value, attrs=None):
    if len(<YourModel>.objects.filter(id=self.form_instance_id))>0:
        file = <YourModel>.objects.get(id=self.form_instance_id).<yourField>
    else:
        file = ''
    substitutions = {
        'initial_text': self.initial_text,
        'input_text': self.input_text,
        'clear_template': '',
        'clear_checkbox_label': self.clear_checkbox_label,
    }
    template = '%(input)s'
    substitutions['input'] = super(ClearableFileInput, self).render(name, value, attrs)
    self.template_with_initial = ('<p class="file-upload">%s</p>'
                        % self.template_with_initial)
    self.template_with_clear = ('<span class="clearable-file-input">%s</span>'
                       % self.template_with_clear)

    if value and hasattr(value, "url"):
        template = self.template_with_initial
        substitutions['initial'] = format_html(self.url_markup_template,
                                               value.url,
                                               force_text(value))
        if not self.is_required:
            checkbox_name = self.clear_checkbox_name(name)
            checkbox_id = self.clear_checkbox_id(checkbox_name)
            substitutions['clear_checkbox_name'] = conditional_escape(checkbox_name)
            substitutions['clear_checkbox_id'] = conditional_escape(checkbox_id)
            substitutions['clear'] = CheckboxInput().render(checkbox_name, False, attrs={'id': checkbox_id})
            substitutions['clear_template'] = self.template_with_clear % substitutions
            url = '' if file == '' else file.url
    else:
        template = self.template_with_initial

        substitutions['initial'] = format_html(self.url_markup_template,
                                               url,
                                               force_text(file))
        if not self.is_required:
            checkbox_name = self.clear_checkbox_name(name)
            checkbox_id = self.clear_checkbox_id(checkbox_name)
            substitutions['clear_checkbox_name'] = conditional_escape(checkbox_name)
            substitutions['clear_checkbox_id'] = conditional_escape(checkbox_id)
            if fav == '':
                substitutions['clear'] = CheckboxInput().render(checkbox_name, False, attrs={'id': checkbox_id, 'disabled': 'disabled'})
            else:
                substitutions['clear'] = CheckboxInput().render(checkbox_name, False, attrs={'id': checkbox_id})
            substitutions['clear_template'] = self.template_with_clear % substitutions



    return mark_safe(template % substitutions)

И тогда в своей форме вы должны написать:

class <YourModel>Form(ModelForm):
    class Meta:
        model = <YourModel>
        fields = '__all__'
        widgets= {'<YourField>': CustomClearableFileInput}

    def __init__(self, *args, **kwargs):
        super(OperatorSettingsForm, self).__init__(*args, **kwargs)
        self.fields['<YourField>'].widget.form_instance_id = self.instance.id

Это работает для меня. Я думаю, у вас тоже не будет проблем:)

...