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):
|
2022-02-03 19:24:19 +00:00
|
|
|
self.check_html(
|
|
|
|
self.widget(choices=self.beatles),
|
|
|
|
"beatles",
|
|
|
|
["J"],
|
|
|
|
html="""
|
2021-06-11 06:39:12 +00:00
|
|
|
<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>
|
2022-02-03 19:24:19 +00:00
|
|
|
""",
|
|
|
|
)
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_render_value_multiple(self):
|
2022-02-03 19:24:19 +00:00
|
|
|
self.check_html(
|
|
|
|
self.widget(choices=self.beatles),
|
|
|
|
"beatles",
|
|
|
|
["J", "P"],
|
|
|
|
html="""
|
2021-06-11 06:39:12 +00:00
|
|
|
<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>
|
2022-02-03 19:24:19 +00:00
|
|
|
""",
|
|
|
|
)
|
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
|
|
|
"""
|
2022-02-03 19:24:19 +00:00
|
|
|
self.check_html(
|
|
|
|
self.widget(choices=(("", "Unknown"),) + self.beatles),
|
|
|
|
"beatles",
|
|
|
|
None,
|
|
|
|
html="""
|
2021-06-11 06:39:12 +00:00
|
|
|
<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>
|
2022-02-03 19:24:19 +00:00
|
|
|
""",
|
|
|
|
)
|
2015-08-31 02:13:42 +00:00
|
|
|
|
|
|
|
def test_nested_choices(self):
|
|
|
|
nested_choices = (
|
2022-02-03 19:24:19 +00:00
|
|
|
("unknown", "Unknown"),
|
|
|
|
("Audio", (("vinyl", "Vinyl"), ("cd", "CD"))),
|
|
|
|
("Video", (("vhs", "VHS"), ("dvd", "DVD"))),
|
2015-08-31 02:13:42 +00:00
|
|
|
)
|
|
|
|
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(
|
2022-02-03 19:24:19 +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 = (
|
2022-02-03 19:24:19 +00:00
|
|
|
("unknown", "Unknown"),
|
|
|
|
("Audio", (("vinyl", "Vinyl"), ("cd", "CD"))),
|
|
|
|
("Video", (("vhs", "VHS"), ("dvd", "DVD"))),
|
2017-01-23 12:44:57 +00:00
|
|
|
)
|
|
|
|
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
|
|
|
"""
|
2022-02-03 19:24:19 +00:00
|
|
|
self.check_html(
|
|
|
|
self.widget(choices=nested_choices),
|
|
|
|
"nestchoice",
|
|
|
|
("vinyl", "dvd"),
|
|
|
|
html=html,
|
|
|
|
)
|
2017-01-23 12:44:57 +00:00
|
|
|
|
2015-08-31 02:13:42 +00:00
|
|
|
def test_separate_ids(self):
|
|
|
|
"""
|
|
|
|
Each input gets a separate ID.
|
|
|
|
"""
|
2022-02-03 19:24:19 +00:00
|
|
|
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
|
|
|
"""
|
2022-02-03 19:24:19 +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.
|
|
|
|
"""
|
2022-02-03 19:24:19 +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
|
|
|
"""
|
2022-02-03 19:24:19 +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 = [
|
2022-02-03 19:24:19 +00:00
|
|
|
(1, "One"),
|
|
|
|
(1000, "One thousand"),
|
|
|
|
(1000000, "One million"),
|
2017-04-21 16:14:40 +00:00
|
|
|
]
|
|
|
|
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
|
|
|
"""
|
2022-02-03 19:24:19 +00:00
|
|
|
self.check_html(self.widget(choices=choices), "numbers", None, html=html)
|
2017-04-21 16:14:40 +00:00
|
|
|
|
|
|
|
choices = [
|
2022-02-03 19:24:19 +00:00
|
|
|
(datetime.time(0, 0), "midnight"),
|
|
|
|
(datetime.time(12, 0), "noon"),
|
2017-04-21 16:14:40 +00:00
|
|
|
]
|
|
|
|
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
|
|
|
"""
|
2022-02-03 19:24:19 +00:00
|
|
|
self.check_html(self.widget(choices=choices), "times", None, html=html)
|
2017-04-21 16:14:40 +00:00
|
|
|
|
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)
|
2022-02-03 19:24:19 +00:00
|
|
|
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)
|
2022-02-03 19:24:19 +00:00
|
|
|
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.
|
|
|
|
"""
|
2022-02-03 19:24:19 +00:00
|
|
|
|
2016-10-28 17:21:45 +00:00
|
|
|
class TestForm(forms.Form):
|
|
|
|
f = forms.MultipleChoiceField(widget=CheckboxSelectMultiple)
|
|
|
|
|
2022-02-03 19:24:19 +00:00
|
|
|
bound_field = TestForm()["f"]
|
|
|
|
self.assertEqual(bound_field.field.widget.id_for_label("id"), "")
|
|
|
|
self.assertEqual(bound_field.label_tag(), "<label>F:</label>")
|
|
|
|
self.assertEqual(bound_field.legend_tag(), "<legend>F:</legend>")
|