mirror of https://github.com/django/django.git
Fixed #34488 -- Made ClearableFileInput preserve "Clear" checked attribute when form is invalid.
This commit is contained in:
parent
fb535e0a90
commit
8a6c0203c4
|
@ -1,6 +1,6 @@
|
||||||
{% if widget.is_initial %}<p class="file-upload">{{ widget.initial_text }}: <a href="{{ widget.value.url }}">{{ widget.value }}</a>{% if not widget.required %}
|
{% if widget.is_initial %}<p class="file-upload">{{ widget.initial_text }}: <a href="{{ widget.value.url }}">{{ widget.value }}</a>{% if not widget.required %}
|
||||||
<span class="clearable-file-input">
|
<span class="clearable-file-input">
|
||||||
<input type="checkbox" name="{{ widget.checkbox_name }}" id="{{ widget.checkbox_id }}"{% if widget.attrs.disabled %} disabled{% endif %}>
|
<input type="checkbox" name="{{ widget.checkbox_name }}" id="{{ widget.checkbox_id }}"{% include "django/forms/widgets/attrs.html" %}>
|
||||||
<label for="{{ widget.checkbox_id }}">{{ widget.clear_checkbox_label }}</label></span>{% endif %}<br>
|
<label for="{{ widget.checkbox_id }}">{{ widget.clear_checkbox_label }}</label></span>{% endif %}<br>
|
||||||
{{ widget.input_text }}:{% endif %}
|
{{ widget.input_text }}:{% endif %}
|
||||||
<input type="{{ widget.type }}" name="{{ widget.name }}"{% include "django/forms/widgets/attrs.html" %}>{% if widget.is_initial %}</p>{% endif %}
|
<input type="{{ widget.type }}" name="{{ widget.name }}"{% include "django/forms/widgets/attrs.html" %}>{% if widget.is_initial %}</p>{% endif %}
|
||||||
|
|
|
@ -433,6 +433,7 @@ class ClearableFileInput(FileInput):
|
||||||
initial_text = _("Currently")
|
initial_text = _("Currently")
|
||||||
input_text = _("Change")
|
input_text = _("Change")
|
||||||
template_name = "django/forms/widgets/clearable_file_input.html"
|
template_name = "django/forms/widgets/clearable_file_input.html"
|
||||||
|
checked = False
|
||||||
|
|
||||||
def clear_checkbox_name(self, name):
|
def clear_checkbox_name(self, name):
|
||||||
"""
|
"""
|
||||||
|
@ -475,10 +476,12 @@ class ClearableFileInput(FileInput):
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
context["widget"]["attrs"].setdefault("disabled", False)
|
context["widget"]["attrs"].setdefault("disabled", False)
|
||||||
|
context["widget"]["attrs"]["checked"] = self.checked
|
||||||
return context
|
return context
|
||||||
|
|
||||||
def value_from_datadict(self, data, files, name):
|
def value_from_datadict(self, data, files, name):
|
||||||
upload = super().value_from_datadict(data, files, name)
|
upload = super().value_from_datadict(data, files, name)
|
||||||
|
self.checked = self.clear_checkbox_name(name) in data
|
||||||
if not self.is_required and CheckboxInput().value_from_datadict(
|
if not self.is_required and CheckboxInput().value_from_datadict(
|
||||||
data, files, self.clear_checkbox_name(name)
|
data, files, self.clear_checkbox_name(name)
|
||||||
):
|
):
|
||||||
|
|
|
@ -1772,39 +1772,59 @@ class RelatedFieldWidgetSeleniumTests(AdminWidgetSeleniumTestCase):
|
||||||
|
|
||||||
@skipUnless(Image, "Pillow not installed")
|
@skipUnless(Image, "Pillow not installed")
|
||||||
class ImageFieldWidgetsSeleniumTests(AdminWidgetSeleniumTestCase):
|
class ImageFieldWidgetsSeleniumTests(AdminWidgetSeleniumTestCase):
|
||||||
def test_clearablefileinput_widget(self):
|
name_input_id = "id_name"
|
||||||
|
photo_input_id = "id_photo"
|
||||||
|
tests_files_folder = "%s/files" % os.getcwd()
|
||||||
|
clear_checkbox_id = "photo-clear_id"
|
||||||
|
|
||||||
|
def _submit_and_wait(self):
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
|
with self.wait_page_loaded():
|
||||||
|
self.selenium.find_element(
|
||||||
|
By.CSS_SELECTOR, "input[value='Save and continue editing']"
|
||||||
|
).click()
|
||||||
|
|
||||||
|
def _run_image_upload_path(self):
|
||||||
from selenium.webdriver.common.by import By
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
self.admin_login(username="super", password="secret", login_url="/")
|
self.admin_login(username="super", password="secret", login_url="/")
|
||||||
self.selenium.get(
|
self.selenium.get(
|
||||||
self.live_server_url + reverse("admin:admin_widgets_student_add"),
|
self.live_server_url + reverse("admin:admin_widgets_student_add"),
|
||||||
)
|
)
|
||||||
|
|
||||||
photo_input_id = "id_photo"
|
|
||||||
save_and_edit_button_css_selector = "input[value='Save and continue editing']"
|
|
||||||
tests_files_folder = "%s/files" % os.getcwd()
|
|
||||||
clear_checkbox_id = "photo-clear_id"
|
|
||||||
|
|
||||||
def _submit_and_wait():
|
|
||||||
with self.wait_page_loaded():
|
|
||||||
self.selenium.find_element(
|
|
||||||
By.CSS_SELECTOR, save_and_edit_button_css_selector
|
|
||||||
).click()
|
|
||||||
|
|
||||||
# Add a student.
|
# Add a student.
|
||||||
title_input = self.selenium.find_element(By.ID, "id_name")
|
name_input = self.selenium.find_element(By.ID, self.name_input_id)
|
||||||
title_input.send_keys("Joe Doe")
|
name_input.send_keys("Joe Doe")
|
||||||
photo_input = self.selenium.find_element(By.ID, photo_input_id)
|
photo_input = self.selenium.find_element(By.ID, self.photo_input_id)
|
||||||
photo_input.send_keys(f"{tests_files_folder}/test.png")
|
photo_input.send_keys(f"{self.tests_files_folder}/test.png")
|
||||||
_submit_and_wait()
|
self._submit_and_wait()
|
||||||
student = Student.objects.last()
|
student = Student.objects.last()
|
||||||
self.assertEqual(student.name, "Joe Doe")
|
self.assertEqual(student.name, "Joe Doe")
|
||||||
self.assertEqual(student.photo.name, "photos/test.png")
|
self.assertRegex(student.photo.name, r"^photos\/(test|test_.+).png")
|
||||||
|
|
||||||
|
def test_clearablefileinput_widget(self):
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
|
self._run_image_upload_path()
|
||||||
|
self.selenium.find_element(By.ID, self.clear_checkbox_id).click()
|
||||||
|
self._submit_and_wait()
|
||||||
|
student = Student.objects.last()
|
||||||
|
self.assertEqual(student.name, "Joe Doe")
|
||||||
|
self.assertEqual(student.photo.name, "")
|
||||||
|
# "Currently" with "Clear" checkbox and "Change" are not shown.
|
||||||
|
photo_field_row = self.selenium.find_element(By.CSS_SELECTOR, ".field-photo")
|
||||||
|
self.assertNotIn("Currently", photo_field_row.text)
|
||||||
|
self.assertNotIn("Change", photo_field_row.text)
|
||||||
|
|
||||||
|
def test_clearablefileinput_widget_invalid_file(self):
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
|
self._run_image_upload_path()
|
||||||
# Uploading non-image files is not supported by Safari with Selenium,
|
# Uploading non-image files is not supported by Safari with Selenium,
|
||||||
# so upload a broken one instead.
|
# so upload a broken one instead.
|
||||||
photo_input = self.selenium.find_element(By.ID, photo_input_id)
|
photo_input = self.selenium.find_element(By.ID, self.photo_input_id)
|
||||||
photo_input.send_keys(f"{tests_files_folder}/brokenimg.png")
|
photo_input.send_keys(f"{self.tests_files_folder}/brokenimg.png")
|
||||||
_submit_and_wait()
|
self._submit_and_wait()
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
self.selenium.find_element(By.CSS_SELECTOR, ".errorlist li").text,
|
self.selenium.find_element(By.CSS_SELECTOR, ".errorlist li").text,
|
||||||
(
|
(
|
||||||
|
@ -1813,12 +1833,30 @@ class ImageFieldWidgetsSeleniumTests(AdminWidgetSeleniumTestCase):
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
# "Currently" with "Clear" checkbox and "Change" still shown.
|
# "Currently" with "Clear" checkbox and "Change" still shown.
|
||||||
cover_field_row = self.selenium.find_element(By.CSS_SELECTOR, ".field-photo")
|
photo_field_row = self.selenium.find_element(By.CSS_SELECTOR, ".field-photo")
|
||||||
self.assertIn("Currently", cover_field_row.text)
|
self.assertIn("Currently", photo_field_row.text)
|
||||||
self.assertIn("Change", cover_field_row.text)
|
self.assertIn("Change", photo_field_row.text)
|
||||||
# "Clear" box works.
|
|
||||||
self.selenium.find_element(By.ID, clear_checkbox_id).click()
|
def test_clearablefileinput_widget_preserve_clear_checkbox(self):
|
||||||
_submit_and_wait()
|
from selenium.webdriver.common.by import By
|
||||||
student.refresh_from_db()
|
|
||||||
self.assertEqual(student.name, "Joe Doe")
|
self._run_image_upload_path()
|
||||||
self.assertEqual(student.photo.name, "")
|
# "Clear" is not checked by default.
|
||||||
|
self.assertIs(
|
||||||
|
self.selenium.find_element(By.ID, self.clear_checkbox_id).is_selected(),
|
||||||
|
False,
|
||||||
|
)
|
||||||
|
# "Clear" was checked, but a validation error is raised.
|
||||||
|
name_input = self.selenium.find_element(By.ID, self.name_input_id)
|
||||||
|
name_input.clear()
|
||||||
|
self.selenium.find_element(By.ID, self.clear_checkbox_id).click()
|
||||||
|
self._submit_and_wait()
|
||||||
|
self.assertEqual(
|
||||||
|
self.selenium.find_element(By.CSS_SELECTOR, ".errorlist li").text,
|
||||||
|
"This field is required.",
|
||||||
|
)
|
||||||
|
# "Clear" persists checked.
|
||||||
|
self.assertIs(
|
||||||
|
self.selenium.find_element(By.ID, self.clear_checkbox_id).is_selected(),
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
|
|
@ -17,7 +17,8 @@ class FakeFieldFile:
|
||||||
|
|
||||||
|
|
||||||
class ClearableFileInputTest(WidgetTest):
|
class ClearableFileInputTest(WidgetTest):
|
||||||
widget = ClearableFileInput()
|
def setUp(self):
|
||||||
|
self.widget = ClearableFileInput()
|
||||||
|
|
||||||
def test_clear_input_renders(self):
|
def test_clear_input_renders(self):
|
||||||
"""
|
"""
|
||||||
|
@ -148,6 +149,7 @@ class ClearableFileInputTest(WidgetTest):
|
||||||
name="myfile",
|
name="myfile",
|
||||||
)
|
)
|
||||||
self.assertIs(value, False)
|
self.assertIs(value, False)
|
||||||
|
self.assertIs(self.widget.checked, True)
|
||||||
|
|
||||||
def test_clear_input_checked_returns_false_only_if_not_required(self):
|
def test_clear_input_checked_returns_false_only_if_not_required(self):
|
||||||
"""
|
"""
|
||||||
|
@ -164,6 +166,7 @@ class ClearableFileInputTest(WidgetTest):
|
||||||
name="myfile",
|
name="myfile",
|
||||||
)
|
)
|
||||||
self.assertEqual(value, field)
|
self.assertEqual(value, field)
|
||||||
|
self.assertIs(widget.checked, True)
|
||||||
|
|
||||||
def test_html_does_not_mask_exceptions(self):
|
def test_html_does_not_mask_exceptions(self):
|
||||||
"""
|
"""
|
||||||
|
|
Loading…
Reference in New Issue