summaryrefslogtreecommitdiff
path: root/tests/forms_tests/widget_tests/test_radioselect.py
diff options
context:
space:
mode:
authorJakub Bagiński <88288777+Jacob1507@users.noreply.github.com>2023-07-28 14:18:07 +0200
committerGitHub <noreply@github.com>2023-07-28 09:18:07 -0300
commitf9c5958b8fe452983122b6a13c8f806e4e4e1eef (patch)
tree41ada0e83893f52c94f327bfeb7c2870fc4c9018 /tests/forms_tests/widget_tests/test_radioselect.py
parent89c8c2e8311bd31b79c0829055d1e494ab88b801 (diff)
Fixed #34655 -- Increased radioselect's test coverage.
Diffstat (limited to 'tests/forms_tests/widget_tests/test_radioselect.py')
-rw-r--r--tests/forms_tests/widget_tests/test_radioselect.py433
1 files changed, 358 insertions, 75 deletions
diff --git a/tests/forms_tests/widget_tests/test_radioselect.py b/tests/forms_tests/widget_tests/test_radioselect.py
index dc3f3d9bad..a861ccf0f3 100644
--- a/tests/forms_tests/widget_tests/test_radioselect.py
+++ b/tests/forms_tests/widget_tests/test_radioselect.py
@@ -1,32 +1,39 @@
import datetime
-from django.forms import ChoiceField, Form, MultiWidget, RadioSelect
+from django.forms import ChoiceField, Form, MultiWidget, RadioSelect, TextInput
from django.test import override_settings
+from django.utils.safestring import mark_safe
-from .base import WidgetTest
+from .test_choicewidget import ChoiceWidgetTest
+BLANK_CHOICE_DASH = (("", "------"),)
-class RadioSelectTest(WidgetTest):
+
+class RadioSelectTest(ChoiceWidgetTest):
widget = RadioSelect
def test_render(self):
- choices = (("", "------"),) + self.beatles
- self.check_html(
- self.widget(choices=choices),
- "beatle",
- "J",
- html="""
- <div>
- <div><label><input type="radio" name="beatle" value=""> ------</label></div>
- <div><label>
- <input checked type="radio" name="beatle" value="J"> John</label></div>
- <div><label><input type="radio" name="beatle" value="P"> Paul</label></div>
- <div><label>
- <input type="radio" name="beatle" value="G"> George</label></div>
- <div><label><input type="radio" name="beatle" value="R"> Ringo</label></div>
- </div>
- """,
- )
+ choices = BLANK_CHOICE_DASH + self.beatles
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="beatle" value="">------</label>
+ </div>
+ <div>
+ <label><input checked type="radio" name="beatle" value="J">John</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="P">Paul</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="G">George</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="R">Ringo</label>
+ </div>
+ </div>
+ """
+ self.check_html(self.widget(choices=choices), "beatle", "J", html=html)
def test_nested_choices(self):
nested_choices = (
@@ -70,27 +77,286 @@ class RadioSelectTest(WidgetTest):
html=html,
)
+ def test_render_none(self):
+ """
+ If value is None, none of the options are selected.
+ """
+ choices = BLANK_CHOICE_DASH + self.beatles
+ html = """
+ <div>
+ <div>
+ <label><input checked type="radio" name="beatle" value="">------</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="J">John</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="P">Paul</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="G">George</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="R">Ringo</label>
+ </div>
+ </div>
+ """
+ self.check_html(self.widget(choices=choices), "beatle", None, html=html)
+
+ def test_render_label_value(self):
+ """
+ If the value corresponds to a label (but not to an option value), none
+ of the options are selected.
+ """
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="beatle" value="J">John</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="P">Paul</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="G">George</label>
+ </div>
+ <div>
+ <label><input type="radio" name="beatle" value="R">Ringo</label>
+ </div>
+ </div>
+ """
+ self.check_html(self.widget(choices=self.beatles), "beatle", "Ringo", html=html)
+
+ def test_render_selected(self):
+ """
+ Only one option can be selected.
+ """
+ choices = [("0", "0"), ("1", "1"), ("2", "2"), ("3", "3"), ("0", "extra")]
+ html = """
+ <div>
+ <div>
+ <label><input checked type="radio" name="choices" value="0">0</label>
+ </div>
+ <div>
+ <label><input type="radio" name="choices" value="1">1</label>
+ </div>
+ <div>
+ <label><input type="radio" name="choices" value="2">2</label>
+ </div>
+ <div>
+ <label><input type="radio" name="choices" value="3">3</label>
+ </div>
+ <div>
+ <label><input type="radio" name="choices" value="0">extra</label>
+ </div>
+ </div>
+ """
+ self.check_html(self.widget(choices=choices), "choices", "0", html=html)
+
def test_constructor_attrs(self):
"""
Attributes provided at instantiation are passed to the constituent
inputs.
"""
- widget = RadioSelect(attrs={"id": "foo"}, choices=self.beatles)
+ widget = self.widget(attrs={"id": "foo"}, choices=self.beatles)
html = """
<div id="foo">
+ <div>
+ <label for="foo_0">
+ <input checked type="radio" id="foo_0" value="J" name="beatle">John</label>
+ </div>
+ <div><label for="foo_1">
+ <input type="radio" id="foo_1" value="P" name="beatle">Paul</label>
+ </div>
+ <div><label for="foo_2">
+ <input type="radio" id="foo_2" value="G" name="beatle">George</label>
+ </div>
+ <div><label for="foo_3">
+ <input type="radio" id="foo_3" value="R" name="beatle">Ringo</label>
+ </div>
+ </div>
+ """
+ self.check_html(widget, "beatle", "J", html=html)
+
+ def test_compare_to_str(self):
+ """
+ The value is compared to its str()
+ """
+ html = """
<div>
- <label for="foo_0">
- <input checked type="radio" id="foo_0" value="J" name="beatle"> John</label>
+ <div>
+ <label><input type="radio" name="num" value="1">1</label>
+ </div>
+ <div>
+ <label><input type="radio" name="num" value="2">2</label>
+ </div>
+ <div>
+ <label><input checked type="radio" name="num" value="3">3</label>
+ </div>
</div>
- <div><label for="foo_1">
- <input type="radio" id="foo_1" value="P" name="beatle"> Paul</label></div>
- <div><label for="foo_2">
- <input type="radio" id="foo_2" value="G" name="beatle"> George</label></div>
- <div><label for="foo_3">
- <input type="radio" id="foo_3" value="R" name="beatle"> Ringo</label></div>
+ """
+ self.check_html(
+ self.widget(choices=[("1", "1"), ("2", "2"), ("3", "3")]),
+ "num",
+ 3,
+ html=html,
+ )
+ self.check_html(
+ self.widget(choices=[(1, 1), (2, 2), (3, 3)]), "num", "3", html=html
+ )
+ self.check_html(
+ self.widget(choices=[(1, 1), (2, 2), (3, 3)]), "num", 3, html=html
+ )
+
+ def test_choices_constructor(self):
+ widget = self.widget(choices=[(1, 1), (2, 2), (3, 3)])
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="num" value="1">1</label>
+ </div>
+ <div>
+ <label><input type="radio" name="num" value="2">2</label>
+ </div>
+ <div>
+ <label><input checked type="radio" name="num" value="3">3</label>
+ </div>
</div>
"""
- self.check_html(widget, "beatle", "J", html=html)
+ self.check_html(widget, "num", 3, html=html)
+
+ def test_choices_constructor_generator(self):
+ """
+ If choices is passed to the constructor and is a generator, it can be
+ iterated over multiple times without getting consumed.
+ """
+
+ def get_choices():
+ for i in range(4):
+ yield (i, i)
+
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="num" value="0">0</label>
+ </div>
+ <div>
+ <label><input type="radio" name="num" value="1">1</label>
+ </div>
+ <div>
+ <label><input type="radio" name="num" value="2">2</label>
+ </div>
+ <div>
+ <label><input checked type="radio" name="num" value="3">3</label>
+ </div>
+ </div>
+ """
+ widget = self.widget(choices=get_choices())
+ self.check_html(widget, "num", 3, html=html)
+
+ def test_choices_escaping(self):
+ choices = (("bad", "you & me"), ("good", mark_safe("you &gt; me")))
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="escape" value="bad">you & me</label>
+ </div>
+ <div>
+ <label><input type="radio" name="escape" value="good">you &gt; me</label>
+ </div>
+ </div>
+ """
+ self.check_html(self.widget(choices=choices), "escape", None, html=html)
+
+ def test_choices_unicode(self):
+ html = """
+ <div>
+ <div>
+ <label>
+ <input checked type="radio" name="email"
+ value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111">
+ \u0160\u0110abc\u0106\u017d\u0107\u017e\u0161\u0111</label>
+ </div>
+ <div>
+ <label>
+ <input type="radio" name="email" value="\u0107\u017e\u0161\u0111">
+ abc\u0107\u017e\u0161\u0111</label>
+ </div>
+ </div>
+ """
+ self.check_html(
+ self.widget(choices=[("ŠĐĆŽćžšđ", "ŠĐabcĆŽćžšđ"), ("ćžšđ", "abcćžšđ")]),
+ "email",
+ "ŠĐĆŽćžšđ",
+ html=html,
+ )
+
+ def test_choices_optgroup(self):
+ """
+ Choices can be nested one level in order to create HTML optgroups
+ """
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="nestchoice" value="outer1">Outer 1</label>
+ </div>
+ <div>
+ <label>Group &quot;1&quot;</label>
+ <div>
+ <label>
+ <input type="radio" name="nestchoice" value="inner1">Inner 1</label>
+ </div>
+ <div>
+ <label>
+ <input type="radio" name="nestchoice" value="inner2">Inner 2</label>
+ </div>
+ </div>
+ </div>
+ """
+ self.check_html(self.nested_widget, "nestchoice", None, html=html)
+
+ def test_choices_select_outer(self):
+ html = """
+ <div>
+ <div>
+ <label>
+ <input checked type="radio" name="nestchoice" value="outer1">Outer 1</label>
+ </div>
+ <div>
+ <label>Group &quot;1&quot;</label>
+ <div>
+ <label>
+ <input type="radio" name="nestchoice" value="inner1">Inner 1</label>
+ </div>
+ <div>
+ <label>
+ <input type="radio" name="nestchoice" value="inner2">Inner 2</label>
+ </div>
+ </div>
+ </div>
+ """
+ self.check_html(self.nested_widget, "nestchoice", "outer1", html=html)
+
+ def test_choices_select_inner(self):
+ html = """
+ <div>
+ <div>
+ <label><input type="radio" name="nestchoice" value="outer1">Outer 1</label>
+ </div>
+ <div>
+ <label>Group &quot;1&quot;</label>
+ <div>
+ <label>
+ <input type="radio" name="nestchoice" value="inner1">Inner 1</label>
+ </div>
+ <div>
+ <label>
+ <input checked type="radio" name="nestchoice" value="inner2">Inner 2
+ </label>
+ </div>
+ </div>
+ </div>
+ """
+ self.check_html(self.nested_widget, "nestchoice", "inner2", html=html)
def test_render_attrs(self):
"""
@@ -99,16 +365,19 @@ class RadioSelectTest(WidgetTest):
"""
html = """
<div id="bar">
- <div>
- <label for="bar_0">
- <input checked type="radio" id="bar_0" value="J" name="beatle"> John</label>
- </div>
- <div><label for="bar_1">
- <input type="radio" id="bar_1" value="P" name="beatle"> Paul</label></div>
- <div><label for="bar_2">
- <input type="radio" id="bar_2" value="G" name="beatle"> George</label></div>
- <div><label for="bar_3">
- <input type="radio" id="bar_3" value="R" name="beatle"> Ringo</label></div>
+ <div>
+ <label for="bar_0">
+ <input checked type="radio" id="bar_0" value="J" name="beatle">John</label>
+ </div>
+ <div><label for="bar_1">
+ <input type="radio" id="bar_1" value="P" name="beatle">Paul</label>
+ </div>
+ <div><label for="bar_2">
+ <input type="radio" id="bar_2" value="G" name="beatle">George</label>
+ </div>
+ <div><label for="bar_3">
+ <input type="radio" id="bar_3" value="R" name="beatle">Ringo</label>
+ </div>
</div>
"""
self.check_html(
@@ -126,15 +395,18 @@ class RadioSelectTest(WidgetTest):
"""
html = """
<div class="bar">
- <div><label>
- <input checked type="radio" class="bar" value="J" name="beatle"> John</label>
- </div>
- <div><label>
- <input type="radio" class="bar" value="P" name="beatle"> Paul</label></div>
- <div><label>
- <input type="radio" class="bar" value="G" name="beatle"> George</label></div>
- <div><label>
- <input type="radio" class="bar" value="R" name="beatle"> Ringo</label></div>
+ <div><label>
+ <input checked type="radio" class="bar" value="J" name="beatle">John</label>
+ </div>
+ <div><label>
+ <input type="radio" class="bar" value="P" name="beatle">Paul</label>
+ </div>
+ <div><label>
+ <input type="radio" class="bar" value="G" name="beatle">George</label>
+ </div>
+ <div><label>
+ <input type="radio" class="bar" value="R" name="beatle">Ringo</label>
+ </div>
</div>
"""
self.check_html(
@@ -154,11 +426,13 @@ class RadioSelectTest(WidgetTest):
]
html = """
<div>
- <div><label><input type="radio" name="number" value="1"> One</label></div>
- <div><label>
- <input type="radio" name="number" value="1000"> One thousand</label></div>
- <div><label>
- <input type="radio" name="number" value="1000000"> One million</label></div>
+ <div><label><input type="radio" name="number" value="1">One</label></div>
+ <div>
+ <label><input type="radio" name="number" value="1000">One thousand</label>
+ </div>
+ <div>
+ <label><input type="radio" name="number" value="1000000">One million</label>
+ </div>
</div>
"""
self.check_html(self.widget(choices=choices), "number", None, html=html)
@@ -169,35 +443,44 @@ class RadioSelectTest(WidgetTest):
]
html = """
<div>
- <div><label>
- <input type="radio" name="time" value="00:00:00"> midnight</label></div>
- <div><label>
- <input type="radio" name="time" value="12:00:00"> noon</label></div>
+ <div>
+ <label><input type="radio" name="time" value="00:00:00">midnight</label>
+ </div>
+ <div>
+ <label><input type="radio" name="time" value="12:00:00">noon</label>
+ </div>
</div>
"""
self.check_html(self.widget(choices=choices), "time", None, html=html)
def test_render_as_subwidget(self):
"""A RadioSelect as a subwidget of MultiWidget."""
- choices = (("", "------"),) + self.beatles
+ choices = BLANK_CHOICE_DASH + self.beatles
+ html = """
+ <div>
+ <div><label>
+ <input type="radio" name="beatle_0" value="">------</label>
+ </div>
+ <div><label>
+ <input checked type="radio" name="beatle_0" value="J">John</label>
+ </div>
+ <div><label>
+ <input type="radio" name="beatle_0" value="P">Paul</label>
+ </div>
+ <div><label>
+ <input type="radio" name="beatle_0" value="G">George</label>
+ </div>
+ <div><label>
+ <input type="radio" name="beatle_0" value="R">Ringo</label>
+ </div>
+ </div>
+ <input name="beatle_1" type="text" value="Some text">
+ """
self.check_html(
- MultiWidget([self.widget(choices=choices)]),
+ MultiWidget([self.widget(choices=choices), TextInput()]),
"beatle",
- ["J"],
- html="""
- <div>
- <div><label>
- <input type="radio" name="beatle_0" value=""> ------</label></div>
- <div><label>
- <input checked type="radio" name="beatle_0" value="J"> John</label></div>
- <div><label>
- <input type="radio" name="beatle_0" value="P"> Paul</label></div>
- <div><label>
- <input type="radio" name="beatle_0" value="G"> George</label></div>
- <div><label>
- <input type="radio" name="beatle_0" value="R"> Ringo</label></div>
- </div>
- """,
+ ["J", "Some text"],
+ html=html,
)
def test_fieldset(self):