mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #27037 -- Prevented required attribute on ClearableFileInput when initial data exists.
This commit is contained in:
		| @@ -134,18 +134,7 @@ class BoundField(object): | ||||
|         the form is not bound or the data otherwise. | ||||
|         """ | ||||
|         if not self.form.is_bound: | ||||
|             data = self.form.initial.get(self.name, self.field.initial) | ||||
|             if callable(data): | ||||
|                 if self._initial_value is not UNSET: | ||||
|                     data = self._initial_value | ||||
|                 else: | ||||
|                     data = data() | ||||
|                     # If this is an auto-generated default date, nix the | ||||
|                     # microseconds for standardized handling. See #22502. | ||||
|                     if (isinstance(data, (datetime.datetime, datetime.time)) and | ||||
|                             not self.field.widget.supports_microseconds): | ||||
|                         data = data.replace(microsecond=0) | ||||
|                     self._initial_value = data | ||||
|             data = self.initial | ||||
|         else: | ||||
|             data = self.field.bound_data( | ||||
|                 self.data, self.form.initial.get(self.name, self.field.initial) | ||||
| @@ -231,11 +220,27 @@ class BoundField(object): | ||||
|         id_ = widget.attrs.get('id') or self.auto_id | ||||
|         return widget.id_for_label(id_) | ||||
|  | ||||
|     @property | ||||
|     def initial(self): | ||||
|         data = self.form.initial.get(self.name, self.field.initial) | ||||
|         if callable(data): | ||||
|             if self._initial_value is not UNSET: | ||||
|                 data = self._initial_value | ||||
|             else: | ||||
|                 data = data() | ||||
|                 # If this is an auto-generated default date, nix the | ||||
|                 # microseconds for standardized handling. See #22502. | ||||
|                 if (isinstance(data, (datetime.datetime, datetime.time)) and | ||||
|                         not self.field.widget.supports_microseconds): | ||||
|                     data = data.replace(microsecond=0) | ||||
|                 self._initial_value = data | ||||
|         return data | ||||
|  | ||||
|     def build_widget_attrs(self, attrs, widget=None): | ||||
|         if not widget: | ||||
|             widget = self.field.widget | ||||
|         attrs = dict(attrs)  # Copy attrs to avoid modifying the argument. | ||||
|         if not widget.is_hidden and self.field.required and self.form.use_required_attribute: | ||||
|         if widget.use_required_attribute(self.initial) and self.field.required and self.form.use_required_attribute: | ||||
|             attrs['required'] = True | ||||
|         if self.field.disabled: | ||||
|             attrs['disabled'] = True | ||||
|   | ||||
| @@ -248,6 +248,9 @@ class Widget(six.with_metaclass(RenameWidgetMethods)): | ||||
|         """ | ||||
|         return id_ | ||||
|  | ||||
|     def use_required_attribute(self, initial): | ||||
|         return not self.is_hidden | ||||
|  | ||||
|  | ||||
| class Input(Widget): | ||||
|     """ | ||||
| @@ -429,6 +432,9 @@ class ClearableFileInput(FileInput): | ||||
|             return False | ||||
|         return upload | ||||
|  | ||||
|     def use_required_attribute(self, initial): | ||||
|         return super(ClearableFileInput, self).use_required_attribute(initial) and not initial | ||||
|  | ||||
|  | ||||
| class Textarea(Widget): | ||||
|     def __init__(self, attrs=None): | ||||
| @@ -795,12 +801,10 @@ class CheckboxSelectMultiple(RendererMixin, SelectMultiple): | ||||
|     renderer = CheckboxFieldRenderer | ||||
|     _empty_value = [] | ||||
|  | ||||
|     def build_attrs(self, extra_attrs=None, **kwargs): | ||||
|         attrs = super(CheckboxSelectMultiple, self).build_attrs(extra_attrs, **kwargs) | ||||
|         # Remove the 'required' attribute because browser validation would | ||||
|     def use_required_attribute(self, initial): | ||||
|         # Don't use the 'required' attribute because browser validation would | ||||
|         # require all checkboxes to be checked instead of at least one. | ||||
|         attrs.pop('required', None) | ||||
|         return attrs | ||||
|         return False | ||||
|  | ||||
|  | ||||
| class MultiWidget(Widget): | ||||
|   | ||||
| @@ -45,3 +45,6 @@ Bugfixes | ||||
|  | ||||
| * Fixed crash of ``django.views.static.serve()`` with ``show_indexes`` enabled | ||||
|   (:ticket:`26973`). | ||||
|  | ||||
| * Fixed ``ClearableFileInput`` to avoid the ``required`` HTML attribute when | ||||
|   initial data exists (:ticket:`27037`). | ||||
|   | ||||
| @@ -2360,6 +2360,15 @@ Password: <input type="password" name="password" required /> | ||||
|             '<tr><th>File1:</th><td><input type="file" name="file1" required /></td></tr>', | ||||
|         ) | ||||
|  | ||||
|         # A required file field with initial data should not contain the | ||||
|         # required HTML attribute. The file input is left blank by the user to | ||||
|         # keep the existing, initial value. | ||||
|         f = FileForm(initial={'file1': 'resume.txt'}, auto_id=False) | ||||
|         self.assertHTMLEqual( | ||||
|             f.as_table(), | ||||
|             '<tr><th>File1:</th><td><input type="file" name="file1" /></td></tr>', | ||||
|         ) | ||||
|  | ||||
|     def test_basic_processing_in_view(self): | ||||
|         class UserRegistration(Form): | ||||
|             username = CharField(max_length=10) | ||||
|   | ||||
| @@ -112,3 +112,11 @@ class CheckboxSelectMultipleTest(WidgetTest): | ||||
|         </ul> | ||||
|         """ | ||||
|         self.check_html(widget, 'letters', ['a', 'c'], html=html) | ||||
|  | ||||
|     def test_use_required_attribute(self): | ||||
|         widget = self.widget(choices=self.beatles) | ||||
|         # Always False because browser validation would require all checkboxes | ||||
|         # to be checked instead of at least one. | ||||
|         self.assertIs(widget.use_required_attribute(None), False) | ||||
|         self.assertIs(widget.use_required_attribute([]), False) | ||||
|         self.assertIs(widget.use_required_attribute(['J', 'P']), False) | ||||
|   | ||||
| @@ -143,3 +143,9 @@ class ClearableFileInputTest(WidgetTest): | ||||
|  | ||||
|         html = self.widget.render('myfile', NoURLFieldFile()) | ||||
|         self.assertHTMLEqual(html, '<input name="myfile" type="file" />') | ||||
|  | ||||
|     def test_use_required_attribute(self): | ||||
|         # False when initial data exists. The file input is left blank by the | ||||
|         # user to keep the existing, initial value. | ||||
|         self.assertIs(self.widget.use_required_attribute(None), True) | ||||
|         self.assertIs(self.widget.use_required_attribute('resume.txt'), False) | ||||
|   | ||||
| @@ -8,3 +8,10 @@ class HiddenInputTest(WidgetTest): | ||||
|  | ||||
|     def test_render(self): | ||||
|         self.check_html(self.widget, 'email', '', html='<input type="hidden" name="email" />') | ||||
|  | ||||
|     def test_use_required_attribute(self): | ||||
|         # Always False to avoid browser validation on inputs hidden from the | ||||
|         # user. | ||||
|         self.assertIs(self.widget.use_required_attribute(None), False) | ||||
|         self.assertIs(self.widget.use_required_attribute(''), False) | ||||
|         self.assertIs(self.widget.use_required_attribute('foo'), False) | ||||
|   | ||||
| @@ -76,3 +76,9 @@ class TextInputTest(WidgetTest): | ||||
|     def test_attrs_safestring(self): | ||||
|         widget = TextInput(attrs={'onBlur': mark_safe("function('foo')")}) | ||||
|         self.check_html(widget, 'email', '', html='<input onBlur="function(\'foo\')" type="text" name="email" />') | ||||
|  | ||||
|     def test_use_required_attribute(self): | ||||
|         # Text inputs can safely trigger the browser validation. | ||||
|         self.assertIs(self.widget.use_required_attribute(None), True) | ||||
|         self.assertIs(self.widget.use_required_attribute(''), True) | ||||
|         self.assertIs(self.widget.use_required_attribute('resume.txt'), True) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user