2017-04-21 16:14:40 +00:00
|
|
|
import datetime
|
|
|
|
|
2016-10-28 17:21:45 +00:00
|
|
|
from django import forms
|
2015-08-31 02:13:42 +00:00
|
|
|
from django.forms import CheckboxSelectMultiple
|
2017-04-21 16:14:40 +00:00
|
|
|
from django.test import override_settings
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
from .base import WidgetTest
|
|
|
|
|
|
|
|
|
|
|
|
class CheckboxSelectMultipleTest(WidgetTest):
|
2016-04-28 22:48:52 +00:00
|
|
|
widget = CheckboxSelectMultiple
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_render_value(self):
|
2021-06-11 06:39:12 +00:00
|
|
|
self.check_html(self.widget(choices=self.beatles), 'beatles', ['J'], html="""
|
|
|
|
<div>
|
|
|
|
<div><label><input checked type="checkbox" name="beatles" value="J"> John</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="P"> Paul</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="G"> George</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="R"> Ringo</label></div>
|
|
|
|
</div>
|
|
|
|
""")
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_render_value_multiple(self):
|
2021-06-11 06:39:12 +00:00
|
|
|
self.check_html(self.widget(choices=self.beatles), 'beatles', ['J', 'P'], html="""
|
|
|
|
<div>
|
|
|
|
<div><label><input checked type="checkbox" name="beatles" value="J"> John</label></div>
|
|
|
|
<div><label><input checked type="checkbox" name="beatles" value="P"> Paul</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="G"> George</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="R"> Ringo</label></div>
|
|
|
|
</div>
|
|
|
|
""")
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_render_none(self):
|
|
|
|
"""
|
2018-03-30 09:55:33 +00:00
|
|
|
If the value is None, none of the options are selected, even if the
|
|
|
|
choices have an empty option.
|
2015-08-31 02:13:42 +00:00
|
|
|
"""
|
2021-06-11 06:39:12 +00:00
|
|
|
self.check_html(self.widget(choices=(('', 'Unknown'),) + self.beatles), 'beatles', None, html="""
|
|
|
|
<div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value=""> Unknown</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="J"> John</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="P"> Paul</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="G"> George</label></div>
|
|
|
|
<div><label><input type="checkbox" name="beatles" value="R"> Ringo</label></div>
|
|
|
|
</div>
|
|
|
|
""")
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_nested_choices(self):
|
|
|
|
nested_choices = (
|
|
|
|
('unknown', 'Unknown'),
|
|
|
|
('Audio', (('vinyl', 'Vinyl'), ('cd', 'CD'))),
|
|
|
|
('Video', (('vhs', 'VHS'), ('dvd', 'DVD'))),
|
|
|
|
)
|
|
|
|
html = """
|
2021-06-11 06:39:12 +00:00
|
|
|
<div id="media">
|
|
|
|
<div> <label for="media_0">
|
|
|
|
<input type="checkbox" name="nestchoice" value="unknown" id="media_0"> Unknown</label></div>
|
|
|
|
<div>
|
|
|
|
<label>Audio</label>
|
|
|
|
<div> <label for="media_1_0">
|
|
|
|
<input checked type="checkbox" name="nestchoice" value="vinyl" id="media_1_0"> Vinyl</label></div>
|
|
|
|
<div> <label for="media_1_1">
|
|
|
|
<input type="checkbox" name="nestchoice" value="cd" id="media_1_1"> CD</label></div>
|
|
|
|
</div><div>
|
|
|
|
<label>Video</label>
|
|
|
|
<div> <label for="media_2_0">
|
|
|
|
<input type="checkbox" name="nestchoice" value="vhs" id="media_2_0"> VHS</label></div>
|
|
|
|
<div> <label for="media_2_1">
|
|
|
|
<input type="checkbox" name="nestchoice" value="dvd" id="media_2_1" checked> DVD</label></div>
|
|
|
|
</div>
|
|
|
|
</div>
|
2015-08-31 02:13:42 +00:00
|
|
|
"""
|
|
|
|
self.check_html(
|
2016-04-28 22:48:52 +00:00
|
|
|
self.widget(choices=nested_choices), 'nestchoice', ('vinyl', 'dvd'),
|
|
|
|
attrs={'id': 'media'}, html=html,
|
2015-08-31 02:13:42 +00:00
|
|
|
)
|
|
|
|
|
2017-01-23 12:44:57 +00:00
|
|
|
def test_nested_choices_without_id(self):
|
|
|
|
nested_choices = (
|
|
|
|
('unknown', 'Unknown'),
|
|
|
|
('Audio', (('vinyl', 'Vinyl'), ('cd', 'CD'))),
|
|
|
|
('Video', (('vhs', 'VHS'), ('dvd', 'DVD'))),
|
|
|
|
)
|
|
|
|
html = """
|
2021-06-11 06:39:12 +00:00
|
|
|
<div>
|
|
|
|
<div> <label><input type="checkbox" name="nestchoice" value="unknown"> Unknown</label></div>
|
|
|
|
<div>
|
|
|
|
<label>Audio</label>
|
|
|
|
<div> <label><input checked type="checkbox" name="nestchoice" value="vinyl"> Vinyl</label></div>
|
|
|
|
<div> <label><input type="checkbox" name="nestchoice" value="cd"> CD</label></div>
|
|
|
|
</div><div>
|
|
|
|
<label>Video</label>
|
|
|
|
<div> <label><input type="checkbox" name="nestchoice" value="vhs"> VHS</label></div>
|
|
|
|
<div> <label><input type="checkbox" name="nestchoice" value="dvd"checked> DVD</label></div>
|
|
|
|
</div>
|
|
|
|
</div>
|
2017-01-23 12:44:57 +00:00
|
|
|
"""
|
|
|
|
self.check_html(self.widget(choices=nested_choices), 'nestchoice', ('vinyl', 'dvd'), html=html)
|
|
|
|
|
2015-08-31 02:13:42 +00:00
|
|
|
def test_separate_ids(self):
|
|
|
|
"""
|
|
|
|
Each input gets a separate ID.
|
|
|
|
"""
|
|
|
|
choices = [('a', 'A'), ('b', 'B'), ('c', 'C')]
|
|
|
|
html = """
|
2021-06-11 06:39:12 +00:00
|
|
|
<div id="abc">
|
|
|
|
<div>
|
2018-01-21 07:09:10 +00:00
|
|
|
<label for="abc_0"><input checked type="checkbox" name="letters" value="a" id="abc_0"> A</label>
|
2021-06-11 06:39:12 +00:00
|
|
|
</div>
|
|
|
|
<div><label for="abc_1"><input type="checkbox" name="letters" value="b" id="abc_1"> B</label></div>
|
|
|
|
<div>
|
2018-01-21 07:09:10 +00:00
|
|
|
<label for="abc_2"><input checked type="checkbox" name="letters" value="c" id="abc_2"> C</label>
|
2021-06-11 06:39:12 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
2015-08-31 02:13:42 +00:00
|
|
|
"""
|
2016-04-28 22:48:52 +00:00
|
|
|
self.check_html(self.widget(choices=choices), 'letters', ['a', 'c'], attrs={'id': 'abc'}, html=html)
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_separate_ids_constructor(self):
|
|
|
|
"""
|
|
|
|
Each input gets a separate ID when the ID is passed to the constructor.
|
|
|
|
"""
|
2016-04-28 22:48:52 +00:00
|
|
|
widget = CheckboxSelectMultiple(attrs={'id': 'abc'}, choices=[('a', 'A'), ('b', 'B'), ('c', 'C')])
|
2015-08-31 02:13:42 +00:00
|
|
|
html = """
|
2021-06-11 06:39:12 +00:00
|
|
|
<div id="abc">
|
|
|
|
<div>
|
2018-01-21 07:09:10 +00:00
|
|
|
<label for="abc_0"><input checked type="checkbox" name="letters" value="a" id="abc_0"> A</label>
|
2021-06-11 06:39:12 +00:00
|
|
|
</div>
|
|
|
|
<div><label for="abc_1"><input type="checkbox" name="letters" value="b" id="abc_1"> B</label></div>
|
|
|
|
<div>
|
2018-01-21 07:09:10 +00:00
|
|
|
<label for="abc_2"><input checked type="checkbox" name="letters" value="c" id="abc_2"> C</label>
|
2021-06-11 06:39:12 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
2015-08-31 02:13:42 +00:00
|
|
|
"""
|
2016-04-28 22:48:52 +00:00
|
|
|
self.check_html(widget, 'letters', ['a', 'c'], html=html)
|
2016-08-12 17:59:01 +00:00
|
|
|
|
2021-09-09 05:42:05 +00:00
|
|
|
@override_settings(USE_THOUSAND_SEPARATOR=True)
|
2017-04-21 16:14:40 +00:00
|
|
|
def test_doesnt_localize_input_value(self):
|
|
|
|
choices = [
|
|
|
|
(1, 'One'),
|
|
|
|
(1000, 'One thousand'),
|
|
|
|
(1000000, 'One million'),
|
|
|
|
]
|
|
|
|
html = """
|
2021-06-11 06:39:12 +00:00
|
|
|
<div>
|
|
|
|
<div><label><input type="checkbox" name="numbers" value="1"> One</label></div>
|
|
|
|
<div><label><input type="checkbox" name="numbers" value="1000"> One thousand</label></div>
|
|
|
|
<div><label><input type="checkbox" name="numbers" value="1000000"> One million</label></div>
|
|
|
|
</div>
|
2017-04-21 16:14:40 +00:00
|
|
|
"""
|
|
|
|
self.check_html(self.widget(choices=choices), 'numbers', None, html=html)
|
|
|
|
|
|
|
|
choices = [
|
|
|
|
(datetime.time(0, 0), 'midnight'),
|
|
|
|
(datetime.time(12, 0), 'noon'),
|
|
|
|
]
|
|
|
|
html = """
|
2021-06-11 06:39:12 +00:00
|
|
|
<div>
|
|
|
|
<div><label><input type="checkbox" name="times" value="00:00:00"> midnight</label></div>
|
|
|
|
<div><label><input type="checkbox" name="times" value="12:00:00"> noon</label></div>
|
|
|
|
</div>
|
2017-04-21 16:14:40 +00:00
|
|
|
"""
|
|
|
|
self.check_html(self.widget(choices=choices), 'times', None, html=html)
|
|
|
|
|
2016-08-12 17:59:01 +00:00
|
|
|
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)
|
2016-09-30 18:49:50 +00:00
|
|
|
|
|
|
|
def test_value_omitted_from_data(self):
|
|
|
|
widget = self.widget(choices=self.beatles)
|
|
|
|
self.assertIs(widget.value_omitted_from_data({}, {}, 'field'), False)
|
|
|
|
self.assertIs(widget.value_omitted_from_data({'field': 'value'}, {}, 'field'), False)
|
2016-10-28 17:21:45 +00:00
|
|
|
|
|
|
|
def test_label(self):
|
2020-09-22 08:30:53 +00:00
|
|
|
"""
|
2016-10-28 17:21:45 +00:00
|
|
|
CheckboxSelectMultiple doesn't contain 'for="field_0"' in the <label>
|
|
|
|
because clicking that would toggle the first checkbox.
|
|
|
|
"""
|
|
|
|
class TestForm(forms.Form):
|
|
|
|
f = forms.MultipleChoiceField(widget=CheckboxSelectMultiple)
|
|
|
|
|
|
|
|
bound_field = TestForm()['f']
|
|
|
|
self.assertEqual(bound_field.field.widget.id_for_label('id'), '')
|
|
|
|
self.assertEqual(bound_field.label_tag(), '<label>F:</label>')
|