PageRenderTime 62ms CodeModel.GetById 19ms RepoModel.GetById 1ms app.codeStats 0ms

/tests/regressiontests/forms/tests/widgets.py

https://code.google.com/p/mango-py/
Python | 1158 lines | 1145 code | 5 blank | 8 comment | 1 complexity | 6a791d43e857d509e116bb2f1471044d MD5 | raw file
Possible License(s): BSD-3-Clause
  1. # -*- coding: utf-8 -*-
  2. import datetime
  3. from decimal import Decimal
  4. import re
  5. import time
  6. from django.conf import settings
  7. from django.core.files.uploadedfile import SimpleUploadedFile
  8. from django.forms import *
  9. from django.forms.widgets import RadioFieldRenderer
  10. from django.utils import copycompat as copy
  11. from django.utils import formats
  12. from django.utils.safestring import mark_safe
  13. from django.utils.translation import activate, deactivate
  14. from django.utils.unittest import TestCase
  15. class FormsWidgetTestCase(TestCase):
  16. # Each Widget class corresponds to an HTML form widget. A Widget knows how to
  17. # render itself, given a field name and some data. Widgets don't perform
  18. # validation.
  19. def test_textinput(self):
  20. w = TextInput()
  21. self.assertEqual(w.render('email', ''), u'<input type="text" name="email" />')
  22. self.assertEqual(w.render('email', None), u'<input type="text" name="email" />')
  23. self.assertEqual(w.render('email', 'test@example.com'), u'<input type="text" name="email" value="test@example.com" />')
  24. self.assertEqual(w.render('email', 'some "quoted" & ampersanded value'), u'<input type="text" name="email" value="some &quot;quoted&quot; &amp; ampersanded value" />')
  25. self.assertEqual(w.render('email', 'test@example.com', attrs={'class': 'fun'}), u'<input type="text" name="email" value="test@example.com" class="fun" />')
  26. # Note that doctest in Python 2.4 (and maybe 2.5?) doesn't support non-ascii
  27. # characters in output, so we're displaying the repr() here.
  28. self.assertEqual(w.render('email', 'Š??Ž?žš?', attrs={'class': 'fun'}), u'<input type="text" name="email" value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" class="fun" />')
  29. # You can also pass 'attrs' to the constructor:
  30. w = TextInput(attrs={'class': 'fun'})
  31. self.assertEqual(w.render('email', ''), u'<input type="text" class="fun" name="email" />')
  32. self.assertEqual(w.render('email', 'foo@example.com'), u'<input type="text" class="fun" value="foo@example.com" name="email" />')
  33. # 'attrs' passed to render() get precedence over those passed to the constructor:
  34. w = TextInput(attrs={'class': 'pretty'})
  35. self.assertEqual(w.render('email', '', attrs={'class': 'special'}), u'<input type="text" class="special" name="email" />')
  36. # 'attrs' can be safe-strings if needed)
  37. w = TextInput(attrs={'onBlur': mark_safe("function('foo')")})
  38. self.assertEqual(w.render('email', ''), u'<input onBlur="function(\'foo\')" type="text" name="email" />')
  39. def test_passwordinput(self):
  40. w = PasswordInput()
  41. self.assertEqual(w.render('email', ''), u'<input type="password" name="email" />')
  42. self.assertEqual(w.render('email', None), u'<input type="password" name="email" />')
  43. self.assertEqual(w.render('email', 'secret'), u'<input type="password" name="email" />')
  44. # The render_value argument lets you specify whether the widget should render
  45. # its value. For security reasons, this is off by default.
  46. w = PasswordInput(render_value=True)
  47. self.assertEqual(w.render('email', ''), u'<input type="password" name="email" />')
  48. self.assertEqual(w.render('email', None), u'<input type="password" name="email" />')
  49. self.assertEqual(w.render('email', 'test@example.com'), u'<input type="password" name="email" value="test@example.com" />')
  50. self.assertEqual(w.render('email', 'some "quoted" & ampersanded value'), u'<input type="password" name="email" value="some &quot;quoted&quot; &amp; ampersanded value" />')
  51. self.assertEqual(w.render('email', 'test@example.com', attrs={'class': 'fun'}), u'<input type="password" name="email" value="test@example.com" class="fun" />')
  52. # You can also pass 'attrs' to the constructor:
  53. w = PasswordInput(attrs={'class': 'fun'}, render_value=True)
  54. self.assertEqual(w.render('email', ''), u'<input type="password" class="fun" name="email" />')
  55. self.assertEqual(w.render('email', 'foo@example.com'), u'<input type="password" class="fun" value="foo@example.com" name="email" />')
  56. # 'attrs' passed to render() get precedence over those passed to the constructor:
  57. w = PasswordInput(attrs={'class': 'pretty'}, render_value=True)
  58. self.assertEqual(w.render('email', '', attrs={'class': 'special'}), u'<input type="password" class="special" name="email" />')
  59. self.assertEqual(w.render('email', 'Š??Ž?žš?', attrs={'class': 'fun'}), u'<input type="password" class="fun" value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" name="email" />')
  60. def test_hiddeninput(self):
  61. w = HiddenInput()
  62. self.assertEqual(w.render('email', ''), u'<input type="hidden" name="email" />')
  63. self.assertEqual(w.render('email', None), u'<input type="hidden" name="email" />')
  64. self.assertEqual(w.render('email', 'test@example.com'), u'<input type="hidden" name="email" value="test@example.com" />')
  65. self.assertEqual(w.render('email', 'some "quoted" & ampersanded value'), u'<input type="hidden" name="email" value="some &quot;quoted&quot; &amp; ampersanded value" />')
  66. self.assertEqual(w.render('email', 'test@example.com', attrs={'class': 'fun'}), u'<input type="hidden" name="email" value="test@example.com" class="fun" />')
  67. # You can also pass 'attrs' to the constructor:
  68. w = HiddenInput(attrs={'class': 'fun'})
  69. self.assertEqual(w.render('email', ''), u'<input type="hidden" class="fun" name="email" />')
  70. self.assertEqual(w.render('email', 'foo@example.com'), u'<input type="hidden" class="fun" value="foo@example.com" name="email" />')
  71. # 'attrs' passed to render() get precedence over those passed to the constructor:
  72. w = HiddenInput(attrs={'class': 'pretty'})
  73. self.assertEqual(w.render('email', '', attrs={'class': 'special'}), u'<input type="hidden" class="special" name="email" />')
  74. self.assertEqual(w.render('email', 'Š??Ž?žš?', attrs={'class': 'fun'}), u'<input type="hidden" class="fun" value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" name="email" />')
  75. # 'attrs' passed to render() get precedence over those passed to the constructor:
  76. w = HiddenInput(attrs={'class': 'pretty'})
  77. self.assertEqual(w.render('email', '', attrs={'class': 'special'}), u'<input type="hidden" class="special" name="email" />')
  78. # Boolean values are rendered to their string forms ("True" and "False").
  79. w = HiddenInput()
  80. self.assertEqual(w.render('get_spam', False), u'<input type="hidden" name="get_spam" value="False" />')
  81. self.assertEqual(w.render('get_spam', True), u'<input type="hidden" name="get_spam" value="True" />')
  82. def test_multiplehiddeninput(self):
  83. w = MultipleHiddenInput()
  84. self.assertEqual(w.render('email', []), u'')
  85. self.assertEqual(w.render('email', None), u'')
  86. self.assertEqual(w.render('email', ['test@example.com']), u'<input type="hidden" name="email" value="test@example.com" />')
  87. self.assertEqual(w.render('email', ['some "quoted" & ampersanded value']), u'<input type="hidden" name="email" value="some &quot;quoted&quot; &amp; ampersanded value" />')
  88. self.assertEqual(w.render('email', ['test@example.com', 'foo@example.com']), u'<input type="hidden" name="email" value="test@example.com" />\n<input type="hidden" name="email" value="foo@example.com" />')
  89. self.assertEqual(w.render('email', ['test@example.com'], attrs={'class': 'fun'}), u'<input type="hidden" name="email" value="test@example.com" class="fun" />')
  90. self.assertEqual(w.render('email', ['test@example.com', 'foo@example.com'], attrs={'class': 'fun'}), u'<input type="hidden" name="email" value="test@example.com" class="fun" />\n<input type="hidden" name="email" value="foo@example.com" class="fun" />')
  91. # You can also pass 'attrs' to the constructor:
  92. w = MultipleHiddenInput(attrs={'class': 'fun'})
  93. self.assertEqual(w.render('email', []), u'')
  94. self.assertEqual(w.render('email', ['foo@example.com']), u'<input type="hidden" class="fun" value="foo@example.com" name="email" />')
  95. self.assertEqual(w.render('email', ['foo@example.com', 'test@example.com']), u'<input type="hidden" class="fun" value="foo@example.com" name="email" />\n<input type="hidden" class="fun" value="test@example.com" name="email" />')
  96. # 'attrs' passed to render() get precedence over those passed to the constructor:
  97. w = MultipleHiddenInput(attrs={'class': 'pretty'})
  98. self.assertEqual(w.render('email', ['foo@example.com'], attrs={'class': 'special'}), u'<input type="hidden" class="special" value="foo@example.com" name="email" />')
  99. self.assertEqual(w.render('email', ['Š??Ž?žš?'], attrs={'class': 'fun'}), u'<input type="hidden" class="fun" value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" name="email" />')
  100. # 'attrs' passed to render() get precedence over those passed to the constructor:
  101. w = MultipleHiddenInput(attrs={'class': 'pretty'})
  102. self.assertEqual(w.render('email', ['foo@example.com'], attrs={'class': 'special'}), u'<input type="hidden" class="special" value="foo@example.com" name="email" />')
  103. # Each input gets a separate ID.
  104. w = MultipleHiddenInput()
  105. self.assertEqual(w.render('letters', list('abc'), attrs={'id': 'hideme'}), u'<input type="hidden" name="letters" value="a" id="hideme_0" />\n<input type="hidden" name="letters" value="b" id="hideme_1" />\n<input type="hidden" name="letters" value="c" id="hideme_2" />')
  106. def test_fileinput(self):
  107. # FileInput widgets don't ever show the value, because the old value is of no use
  108. # if you are updating the form or if the provided file generated an error.
  109. w = FileInput()
  110. self.assertEqual(w.render('email', ''), u'<input type="file" name="email" />')
  111. self.assertEqual(w.render('email', None), u'<input type="file" name="email" />')
  112. self.assertEqual(w.render('email', 'test@example.com'), u'<input type="file" name="email" />')
  113. self.assertEqual(w.render('email', 'some "quoted" & ampersanded value'), u'<input type="file" name="email" />')
  114. self.assertEqual(w.render('email', 'test@example.com', attrs={'class': 'fun'}), u'<input type="file" name="email" class="fun" />')
  115. # You can also pass 'attrs' to the constructor:
  116. w = FileInput(attrs={'class': 'fun'})
  117. self.assertEqual(w.render('email', ''), u'<input type="file" class="fun" name="email" />')
  118. self.assertEqual(w.render('email', 'foo@example.com'), u'<input type="file" class="fun" name="email" />')
  119. self.assertEqual(w.render('email', 'Š??Ž?žš?', attrs={'class': 'fun'}), u'<input type="file" class="fun" name="email" />')
  120. # Test for the behavior of _has_changed for FileInput. The value of data will
  121. # more than likely come from request.FILES. The value of initial data will
  122. # likely be a filename stored in the database. Since its value is of no use to
  123. # a FileInput it is ignored.
  124. w = FileInput()
  125. # No file was uploaded and no initial data.
  126. self.assertFalse(w._has_changed(u'', None))
  127. # A file was uploaded and no initial data.
  128. self.assertTrue(w._has_changed(u'', {'filename': 'resume.txt', 'content': 'My resume'}))
  129. # A file was not uploaded, but there is initial data
  130. self.assertFalse(w._has_changed(u'resume.txt', None))
  131. # A file was uploaded and there is initial data (file identity is not dealt
  132. # with here)
  133. self.assertTrue(w._has_changed('resume.txt', {'filename': 'resume.txt', 'content': 'My resume'}))
  134. def test_textarea(self):
  135. w = Textarea()
  136. self.assertEqual(w.render('msg', ''), u'<textarea rows="10" cols="40" name="msg"></textarea>')
  137. self.assertEqual(w.render('msg', None), u'<textarea rows="10" cols="40" name="msg"></textarea>')
  138. self.assertEqual(w.render('msg', 'value'), u'<textarea rows="10" cols="40" name="msg">value</textarea>')
  139. self.assertEqual(w.render('msg', 'some "quoted" & ampersanded value'), u'<textarea rows="10" cols="40" name="msg">some &quot;quoted&quot; &amp; ampersanded value</textarea>')
  140. self.assertEqual(w.render('msg', mark_safe('pre &quot;quoted&quot; value')), u'<textarea rows="10" cols="40" name="msg">pre &quot;quoted&quot; value</textarea>')
  141. self.assertEqual(w.render('msg', 'value', attrs={'class': 'pretty', 'rows': 20}), u'<textarea class="pretty" rows="20" cols="40" name="msg">value</textarea>')
  142. # You can also pass 'attrs' to the constructor:
  143. w = Textarea(attrs={'class': 'pretty'})
  144. self.assertEqual(w.render('msg', ''), u'<textarea rows="10" cols="40" name="msg" class="pretty"></textarea>')
  145. self.assertEqual(w.render('msg', 'example'), u'<textarea rows="10" cols="40" name="msg" class="pretty">example</textarea>')
  146. # 'attrs' passed to render() get precedence over those passed to the constructor:
  147. w = Textarea(attrs={'class': 'pretty'})
  148. self.assertEqual(w.render('msg', '', attrs={'class': 'special'}), u'<textarea rows="10" cols="40" name="msg" class="special"></textarea>')
  149. self.assertEqual(w.render('msg', 'Š??Ž?žš?', attrs={'class': 'fun'}), u'<textarea rows="10" cols="40" name="msg" class="fun">\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111</textarea>')
  150. def test_checkboxinput(self):
  151. w = CheckboxInput()
  152. self.assertEqual(w.render('is_cool', ''), u'<input type="checkbox" name="is_cool" />')
  153. self.assertEqual(w.render('is_cool', None), u'<input type="checkbox" name="is_cool" />')
  154. self.assertEqual(w.render('is_cool', False), u'<input type="checkbox" name="is_cool" />')
  155. self.assertEqual(w.render('is_cool', True), u'<input checked="checked" type="checkbox" name="is_cool" />')
  156. # Using any value that's not in ('', None, False, True) will check the checkbox
  157. # and set the 'value' attribute.
  158. self.assertEqual(w.render('is_cool', 'foo'), u'<input checked="checked" type="checkbox" name="is_cool" value="foo" />')
  159. self.assertEqual(w.render('is_cool', False, attrs={'class': 'pretty'}), u'<input type="checkbox" name="is_cool" class="pretty" />')
  160. # You can also pass 'attrs' to the constructor:
  161. w = CheckboxInput(attrs={'class': 'pretty'})
  162. self.assertEqual(w.render('is_cool', ''), u'<input type="checkbox" class="pretty" name="is_cool" />')
  163. # 'attrs' passed to render() get precedence over those passed to the constructor:
  164. w = CheckboxInput(attrs={'class': 'pretty'})
  165. self.assertEqual(w.render('is_cool', '', attrs={'class': 'special'}), u'<input type="checkbox" class="special" name="is_cool" />')
  166. # You can pass 'check_test' to the constructor. This is a callable that takes the
  167. # value and returns True if the box should be checked.
  168. w = CheckboxInput(check_test=lambda value: value.startswith('hello'))
  169. self.assertEqual(w.render('greeting', ''), u'<input type="checkbox" name="greeting" />')
  170. self.assertEqual(w.render('greeting', 'hello'), u'<input checked="checked" type="checkbox" name="greeting" value="hello" />')
  171. self.assertEqual(w.render('greeting', 'hello there'), u'<input checked="checked" type="checkbox" name="greeting" value="hello there" />')
  172. self.assertEqual(w.render('greeting', 'hello & goodbye'), u'<input checked="checked" type="checkbox" name="greeting" value="hello &amp; goodbye" />')
  173. # A subtlety: If the 'check_test' argument cannot handle a value and raises any
  174. # exception during its __call__, then the exception will be swallowed and the box
  175. # will not be checked. In this example, the 'check_test' assumes the value has a
  176. # startswith() method, which fails for the values True, False and None.
  177. self.assertEqual(w.render('greeting', True), u'<input type="checkbox" name="greeting" />')
  178. self.assertEqual(w.render('greeting', False), u'<input type="checkbox" name="greeting" />')
  179. self.assertEqual(w.render('greeting', None), u'<input type="checkbox" name="greeting" />')
  180. # The CheckboxInput widget will return False if the key is not found in the data
  181. # dictionary (because HTML form submission doesn't send any result for unchecked
  182. # checkboxes).
  183. self.assertFalse(w.value_from_datadict({}, {}, 'testing'))
  184. self.assertFalse(w._has_changed(None, None))
  185. self.assertFalse(w._has_changed(None, u''))
  186. self.assertFalse(w._has_changed(u'', None))
  187. self.assertFalse(w._has_changed(u'', u''))
  188. self.assertTrue(w._has_changed(False, u'on'))
  189. self.assertFalse(w._has_changed(True, u'on'))
  190. self.assertTrue(w._has_changed(True, u''))
  191. def test_select(self):
  192. w = Select()
  193. self.assertEqual(w.render('beatle', 'J', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select name="beatle">
  194. <option value="J" selected="selected">John</option>
  195. <option value="P">Paul</option>
  196. <option value="G">George</option>
  197. <option value="R">Ringo</option>
  198. </select>""")
  199. # If the value is None, none of the options are selected:
  200. self.assertEqual(w.render('beatle', None, choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select name="beatle">
  201. <option value="J">John</option>
  202. <option value="P">Paul</option>
  203. <option value="G">George</option>
  204. <option value="R">Ringo</option>
  205. </select>""")
  206. # If the value corresponds to a label (but not to an option value), none of the options are selected:
  207. self.assertEqual(w.render('beatle', 'John', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select name="beatle">
  208. <option value="J">John</option>
  209. <option value="P">Paul</option>
  210. <option value="G">George</option>
  211. <option value="R">Ringo</option>
  212. </select>""")
  213. # The value is compared to its str():
  214. self.assertEqual(w.render('num', 2, choices=[('1', '1'), ('2', '2'), ('3', '3')]), """<select name="num">
  215. <option value="1">1</option>
  216. <option value="2" selected="selected">2</option>
  217. <option value="3">3</option>
  218. </select>""")
  219. self.assertEqual(w.render('num', '2', choices=[(1, 1), (2, 2), (3, 3)]), """<select name="num">
  220. <option value="1">1</option>
  221. <option value="2" selected="selected">2</option>
  222. <option value="3">3</option>
  223. </select>""")
  224. self.assertEqual(w.render('num', 2, choices=[(1, 1), (2, 2), (3, 3)]), """<select name="num">
  225. <option value="1">1</option>
  226. <option value="2" selected="selected">2</option>
  227. <option value="3">3</option>
  228. </select>""")
  229. # The 'choices' argument can be any iterable:
  230. from itertools import chain
  231. def get_choices():
  232. for i in range(5):
  233. yield (i, i)
  234. self.assertEqual(w.render('num', 2, choices=get_choices()), """<select name="num">
  235. <option value="0">0</option>
  236. <option value="1">1</option>
  237. <option value="2" selected="selected">2</option>
  238. <option value="3">3</option>
  239. <option value="4">4</option>
  240. </select>""")
  241. things = ({'id': 1, 'name': 'And Boom'}, {'id': 2, 'name': 'One More Thing!'})
  242. class SomeForm(Form):
  243. somechoice = ChoiceField(choices=chain((('', '-'*9),), [(thing['id'], thing['name']) for thing in things]))
  244. f = SomeForm()
  245. self.assertEqual(f.as_table(), u'<tr><th><label for="id_somechoice">Somechoice:</label></th><td><select name="somechoice" id="id_somechoice">\n<option value="" selected="selected">---------</option>\n<option value="1">And Boom</option>\n<option value="2">One More Thing!</option>\n</select></td></tr>')
  246. self.assertEqual(f.as_table(), u'<tr><th><label for="id_somechoice">Somechoice:</label></th><td><select name="somechoice" id="id_somechoice">\n<option value="" selected="selected">---------</option>\n<option value="1">And Boom</option>\n<option value="2">One More Thing!</option>\n</select></td></tr>')
  247. f = SomeForm({'somechoice': 2})
  248. self.assertEqual(f.as_table(), u'<tr><th><label for="id_somechoice">Somechoice:</label></th><td><select name="somechoice" id="id_somechoice">\n<option value="">---------</option>\n<option value="1">And Boom</option>\n<option value="2" selected="selected">One More Thing!</option>\n</select></td></tr>')
  249. # You can also pass 'choices' to the constructor:
  250. w = Select(choices=[(1, 1), (2, 2), (3, 3)])
  251. self.assertEqual(w.render('num', 2), """<select name="num">
  252. <option value="1">1</option>
  253. <option value="2" selected="selected">2</option>
  254. <option value="3">3</option>
  255. </select>""")
  256. # If 'choices' is passed to both the constructor and render(), then they'll both be in the output:
  257. self.assertEqual(w.render('num', 2, choices=[(4, 4), (5, 5)]), """<select name="num">
  258. <option value="1">1</option>
  259. <option value="2" selected="selected">2</option>
  260. <option value="3">3</option>
  261. <option value="4">4</option>
  262. <option value="5">5</option>
  263. </select>""")
  264. # Choices are escaped correctly
  265. self.assertEqual(w.render('escape', None, choices=(('bad', 'you & me'), ('good', mark_safe('you &gt; me')))), """<select name="escape">
  266. <option value="1">1</option>
  267. <option value="2">2</option>
  268. <option value="3">3</option>
  269. <option value="bad">you &amp; me</option>
  270. <option value="good">you &gt; me</option>
  271. </select>""")
  272. # Unicode choices are correctly rendered as HTML
  273. self.assertEqual(w.render('email', 'Š??Ž?žš?', choices=[('Š??Ž?žš?', 'Š?abc?Ž?žš?'), ('?žš?', 'abc?žš?')]), u'<select name="email">\n<option value="1">1</option>\n<option value="2">2</option>\n<option value="3">3</option>\n<option value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" selected="selected">\u0160\u0110abc\u0106\u017d\u0107\u017e\u0161\u0111</option>\n<option value="\u0107\u017e\u0161\u0111">abc\u0107\u017e\u0161\u0111</option>\n</select>')
  274. # If choices is passed to the constructor and is a generator, it can be iterated
  275. # over multiple times without getting consumed:
  276. w = Select(choices=get_choices())
  277. self.assertEqual(w.render('num', 2), """<select name="num">
  278. <option value="0">0</option>
  279. <option value="1">1</option>
  280. <option value="2" selected="selected">2</option>
  281. <option value="3">3</option>
  282. <option value="4">4</option>
  283. </select>""")
  284. self.assertEqual(w.render('num', 3), """<select name="num">
  285. <option value="0">0</option>
  286. <option value="1">1</option>
  287. <option value="2">2</option>
  288. <option value="3" selected="selected">3</option>
  289. <option value="4">4</option>
  290. </select>""")
  291. # Choices can be nested one level in order to create HTML optgroups:
  292. w.choices=(('outer1', 'Outer 1'), ('Group "1"', (('inner1', 'Inner 1'), ('inner2', 'Inner 2'))))
  293. self.assertEqual(w.render('nestchoice', None), """<select name="nestchoice">
  294. <option value="outer1">Outer 1</option>
  295. <optgroup label="Group &quot;1&quot;">
  296. <option value="inner1">Inner 1</option>
  297. <option value="inner2">Inner 2</option>
  298. </optgroup>
  299. </select>""")
  300. self.assertEqual(w.render('nestchoice', 'outer1'), """<select name="nestchoice">
  301. <option value="outer1" selected="selected">Outer 1</option>
  302. <optgroup label="Group &quot;1&quot;">
  303. <option value="inner1">Inner 1</option>
  304. <option value="inner2">Inner 2</option>
  305. </optgroup>
  306. </select>""")
  307. self.assertEqual(w.render('nestchoice', 'inner1'), """<select name="nestchoice">
  308. <option value="outer1">Outer 1</option>
  309. <optgroup label="Group &quot;1&quot;">
  310. <option value="inner1" selected="selected">Inner 1</option>
  311. <option value="inner2">Inner 2</option>
  312. </optgroup>
  313. </select>""")
  314. def test_nullbooleanselect(self):
  315. w = NullBooleanSelect()
  316. self.assertTrue(w.render('is_cool', True), """<select name="is_cool">
  317. <option value="1">Unknown</option>
  318. <option value="2" selected="selected">Yes</option>
  319. <option value="3">No</option>
  320. </select>""")
  321. self.assertEqual(w.render('is_cool', False), """<select name="is_cool">
  322. <option value="1">Unknown</option>
  323. <option value="2">Yes</option>
  324. <option value="3" selected="selected">No</option>
  325. </select>""")
  326. self.assertEqual(w.render('is_cool', None), """<select name="is_cool">
  327. <option value="1" selected="selected">Unknown</option>
  328. <option value="2">Yes</option>
  329. <option value="3">No</option>
  330. </select>""")
  331. self.assertEqual(w.render('is_cool', '2'), """<select name="is_cool">
  332. <option value="1">Unknown</option>
  333. <option value="2" selected="selected">Yes</option>
  334. <option value="3">No</option>
  335. </select>""")
  336. self.assertEqual(w.render('is_cool', '3'), """<select name="is_cool">
  337. <option value="1">Unknown</option>
  338. <option value="2">Yes</option>
  339. <option value="3" selected="selected">No</option>
  340. </select>""")
  341. self.assertTrue(w._has_changed(False, None))
  342. self.assertTrue(w._has_changed(None, False))
  343. self.assertFalse(w._has_changed(None, None))
  344. self.assertFalse(w._has_changed(False, False))
  345. self.assertTrue(w._has_changed(True, False))
  346. self.assertTrue(w._has_changed(True, None))
  347. self.assertTrue(w._has_changed(True, False))
  348. def test_selectmultiple(self):
  349. w = SelectMultiple()
  350. self.assertEqual(w.render('beatles', ['J'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select multiple="multiple" name="beatles">
  351. <option value="J" selected="selected">John</option>
  352. <option value="P">Paul</option>
  353. <option value="G">George</option>
  354. <option value="R">Ringo</option>
  355. </select>""")
  356. self.assertEqual(w.render('beatles', ['J', 'P'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select multiple="multiple" name="beatles">
  357. <option value="J" selected="selected">John</option>
  358. <option value="P" selected="selected">Paul</option>
  359. <option value="G">George</option>
  360. <option value="R">Ringo</option>
  361. </select>""")
  362. self.assertEqual(w.render('beatles', ['J', 'P', 'R'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select multiple="multiple" name="beatles">
  363. <option value="J" selected="selected">John</option>
  364. <option value="P" selected="selected">Paul</option>
  365. <option value="G">George</option>
  366. <option value="R" selected="selected">Ringo</option>
  367. </select>""")
  368. # If the value is None, none of the options are selected:
  369. self.assertEqual(w.render('beatles', None, choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select multiple="multiple" name="beatles">
  370. <option value="J">John</option>
  371. <option value="P">Paul</option>
  372. <option value="G">George</option>
  373. <option value="R">Ringo</option>
  374. </select>""")
  375. # If the value corresponds to a label (but not to an option value), none of the options are selected:
  376. self.assertEqual(w.render('beatles', ['John'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select multiple="multiple" name="beatles">
  377. <option value="J">John</option>
  378. <option value="P">Paul</option>
  379. <option value="G">George</option>
  380. <option value="R">Ringo</option>
  381. </select>""")
  382. # If multiple values are given, but some of them are not valid, the valid ones are selected:
  383. self.assertEqual(w.render('beatles', ['J', 'G', 'foo'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<select multiple="multiple" name="beatles">
  384. <option value="J" selected="selected">John</option>
  385. <option value="P">Paul</option>
  386. <option value="G" selected="selected">George</option>
  387. <option value="R">Ringo</option>
  388. </select>""")
  389. # The value is compared to its str():
  390. self.assertEqual(w.render('nums', [2], choices=[('1', '1'), ('2', '2'), ('3', '3')]), """<select multiple="multiple" name="nums">
  391. <option value="1">1</option>
  392. <option value="2" selected="selected">2</option>
  393. <option value="3">3</option>
  394. </select>""")
  395. self.assertEqual(w.render('nums', ['2'], choices=[(1, 1), (2, 2), (3, 3)]), """<select multiple="multiple" name="nums">
  396. <option value="1">1</option>
  397. <option value="2" selected="selected">2</option>
  398. <option value="3">3</option>
  399. </select>""")
  400. self.assertEqual(w.render('nums', [2], choices=[(1, 1), (2, 2), (3, 3)]), """<select multiple="multiple" name="nums">
  401. <option value="1">1</option>
  402. <option value="2" selected="selected">2</option>
  403. <option value="3">3</option>
  404. </select>""")
  405. # The 'choices' argument can be any iterable:
  406. def get_choices():
  407. for i in range(5):
  408. yield (i, i)
  409. self.assertEqual(w.render('nums', [2], choices=get_choices()), """<select multiple="multiple" name="nums">
  410. <option value="0">0</option>
  411. <option value="1">1</option>
  412. <option value="2" selected="selected">2</option>
  413. <option value="3">3</option>
  414. <option value="4">4</option>
  415. </select>""")
  416. # You can also pass 'choices' to the constructor:
  417. w = SelectMultiple(choices=[(1, 1), (2, 2), (3, 3)])
  418. self.assertEqual(w.render('nums', [2]), """<select multiple="multiple" name="nums">
  419. <option value="1">1</option>
  420. <option value="2" selected="selected">2</option>
  421. <option value="3">3</option>
  422. </select>""")
  423. # If 'choices' is passed to both the constructor and render(), then they'll both be in the output:
  424. self.assertEqual(w.render('nums', [2], choices=[(4, 4), (5, 5)]), """<select multiple="multiple" name="nums">
  425. <option value="1">1</option>
  426. <option value="2" selected="selected">2</option>
  427. <option value="3">3</option>
  428. <option value="4">4</option>
  429. <option value="5">5</option>
  430. </select>""")
  431. # Choices are escaped correctly
  432. self.assertEqual(w.render('escape', None, choices=(('bad', 'you & me'), ('good', mark_safe('you &gt; me')))), """<select multiple="multiple" name="escape">
  433. <option value="1">1</option>
  434. <option value="2">2</option>
  435. <option value="3">3</option>
  436. <option value="bad">you &amp; me</option>
  437. <option value="good">you &gt; me</option>
  438. </select>""")
  439. # Unicode choices are correctly rendered as HTML
  440. self.assertEqual(w.render('nums', ['Š??Ž?žš?'], choices=[('Š??Ž?žš?', 'Š?abc?Ž?žš?'), ('?žš?', 'abc?žš?')]), u'<select multiple="multiple" name="nums">\n<option value="1">1</option>\n<option value="2">2</option>\n<option value="3">3</option>\n<option value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" selected="selected">\u0160\u0110abc\u0106\u017d\u0107\u017e\u0161\u0111</option>\n<option value="\u0107\u017e\u0161\u0111">abc\u0107\u017e\u0161\u0111</option>\n</select>')
  441. # Test the usage of _has_changed
  442. self.assertFalse(w._has_changed(None, None))
  443. self.assertFalse(w._has_changed([], None))
  444. self.assertTrue(w._has_changed(None, [u'1']))
  445. self.assertFalse(w._has_changed([1, 2], [u'1', u'2']))
  446. self.assertTrue(w._has_changed([1, 2], [u'1']))
  447. self.assertTrue(w._has_changed([1, 2], [u'1', u'3']))
  448. # Choices can be nested one level in order to create HTML optgroups:
  449. w.choices = (('outer1', 'Outer 1'), ('Group "1"', (('inner1', 'Inner 1'), ('inner2', 'Inner 2'))))
  450. self.assertEqual(w.render('nestchoice', None), """<select multiple="multiple" name="nestchoice">
  451. <option value="outer1">Outer 1</option>
  452. <optgroup label="Group &quot;1&quot;">
  453. <option value="inner1">Inner 1</option>
  454. <option value="inner2">Inner 2</option>
  455. </optgroup>
  456. </select>""")
  457. self.assertEqual(w.render('nestchoice', ['outer1']), """<select multiple="multiple" name="nestchoice">
  458. <option value="outer1" selected="selected">Outer 1</option>
  459. <optgroup label="Group &quot;1&quot;">
  460. <option value="inner1">Inner 1</option>
  461. <option value="inner2">Inner 2</option>
  462. </optgroup>
  463. </select>""")
  464. self.assertEqual(w.render('nestchoice', ['inner1']), """<select multiple="multiple" name="nestchoice">
  465. <option value="outer1">Outer 1</option>
  466. <optgroup label="Group &quot;1&quot;">
  467. <option value="inner1" selected="selected">Inner 1</option>
  468. <option value="inner2">Inner 2</option>
  469. </optgroup>
  470. </select>""")
  471. self.assertEqual(w.render('nestchoice', ['outer1', 'inner2']), """<select multiple="multiple" name="nestchoice">
  472. <option value="outer1" selected="selected">Outer 1</option>
  473. <optgroup label="Group &quot;1&quot;">
  474. <option value="inner1">Inner 1</option>
  475. <option value="inner2" selected="selected">Inner 2</option>
  476. </optgroup>
  477. </select>""")
  478. def test_radioselect(self):
  479. w = RadioSelect()
  480. self.assertEqual(w.render('beatle', 'J', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  481. <li><label><input checked="checked" type="radio" name="beatle" value="J" /> John</label></li>
  482. <li><label><input type="radio" name="beatle" value="P" /> Paul</label></li>
  483. <li><label><input type="radio" name="beatle" value="G" /> George</label></li>
  484. <li><label><input type="radio" name="beatle" value="R" /> Ringo</label></li>
  485. </ul>""")
  486. # If the value is None, none of the options are checked:
  487. self.assertEqual(w.render('beatle', None, choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  488. <li><label><input type="radio" name="beatle" value="J" /> John</label></li>
  489. <li><label><input type="radio" name="beatle" value="P" /> Paul</label></li>
  490. <li><label><input type="radio" name="beatle" value="G" /> George</label></li>
  491. <li><label><input type="radio" name="beatle" value="R" /> Ringo</label></li>
  492. </ul>""")
  493. # If the value corresponds to a label (but not to an option value), none of the options are checked:
  494. self.assertEqual(w.render('beatle', 'John', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  495. <li><label><input type="radio" name="beatle" value="J" /> John</label></li>
  496. <li><label><input type="radio" name="beatle" value="P" /> Paul</label></li>
  497. <li><label><input type="radio" name="beatle" value="G" /> George</label></li>
  498. <li><label><input type="radio" name="beatle" value="R" /> Ringo</label></li>
  499. </ul>""")
  500. # The value is compared to its str():
  501. self.assertEqual(w.render('num', 2, choices=[('1', '1'), ('2', '2'), ('3', '3')]), """<ul>
  502. <li><label><input type="radio" name="num" value="1" /> 1</label></li>
  503. <li><label><input checked="checked" type="radio" name="num" value="2" /> 2</label></li>
  504. <li><label><input type="radio" name="num" value="3" /> 3</label></li>
  505. </ul>""")
  506. self.assertEqual(w.render('num', '2', choices=[(1, 1), (2, 2), (3, 3)]), """<ul>
  507. <li><label><input type="radio" name="num" value="1" /> 1</label></li>
  508. <li><label><input checked="checked" type="radio" name="num" value="2" /> 2</label></li>
  509. <li><label><input type="radio" name="num" value="3" /> 3</label></li>
  510. </ul>""")
  511. self.assertEqual(w.render('num', 2, choices=[(1, 1), (2, 2), (3, 3)]), """<ul>
  512. <li><label><input type="radio" name="num" value="1" /> 1</label></li>
  513. <li><label><input checked="checked" type="radio" name="num" value="2" /> 2</label></li>
  514. <li><label><input type="radio" name="num" value="3" /> 3</label></li>
  515. </ul>""")
  516. # The 'choices' argument can be any iterable:
  517. def get_choices():
  518. for i in range(5):
  519. yield (i, i)
  520. self.assertEqual(w.render('num', 2, choices=get_choices()), """<ul>
  521. <li><label><input type="radio" name="num" value="0" /> 0</label></li>
  522. <li><label><input type="radio" name="num" value="1" /> 1</label></li>
  523. <li><label><input checked="checked" type="radio" name="num" value="2" /> 2</label></li>
  524. <li><label><input type="radio" name="num" value="3" /> 3</label></li>
  525. <li><label><input type="radio" name="num" value="4" /> 4</label></li>
  526. </ul>""")
  527. # You can also pass 'choices' to the constructor:
  528. w = RadioSelect(choices=[(1, 1), (2, 2), (3, 3)])
  529. self.assertEqual(w.render('num', 2), """<ul>
  530. <li><label><input type="radio" name="num" value="1" /> 1</label></li>
  531. <li><label><input checked="checked" type="radio" name="num" value="2" /> 2</label></li>
  532. <li><label><input type="radio" name="num" value="3" /> 3</label></li>
  533. </ul>""")
  534. # If 'choices' is passed to both the constructor and render(), then they'll both be in the output:
  535. self.assertEqual(w.render('num', 2, choices=[(4, 4), (5, 5)]), """<ul>
  536. <li><label><input type="radio" name="num" value="1" /> 1</label></li>
  537. <li><label><input checked="checked" type="radio" name="num" value="2" /> 2</label></li>
  538. <li><label><input type="radio" name="num" value="3" /> 3</label></li>
  539. <li><label><input type="radio" name="num" value="4" /> 4</label></li>
  540. <li><label><input type="radio" name="num" value="5" /> 5</label></li>
  541. </ul>""")
  542. # RadioSelect uses a RadioFieldRenderer to render the individual radio inputs.
  543. # You can manipulate that object directly to customize the way the RadioSelect
  544. # is rendered.
  545. w = RadioSelect()
  546. r = w.get_renderer('beatle', 'J', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo')))
  547. inp_set1 = []
  548. inp_set2 = []
  549. inp_set3 = []
  550. inp_set4 = []
  551. for inp in r:
  552. inp_set1.append(str(inp))
  553. inp_set2.append('%s<br />' % inp)
  554. inp_set3.append('<p>%s %s</p>' % (inp.tag(), inp.choice_label))
  555. inp_set4.append('%s %s %s %s %s' % (inp.name, inp.value, inp.choice_value, inp.choice_label, inp.is_checked()))
  556. self.assertEqual('\n'.join(inp_set1), """<label><input checked="checked" type="radio" name="beatle" value="J" /> John</label>
  557. <label><input type="radio" name="beatle" value="P" /> Paul</label>
  558. <label><input type="radio" name="beatle" value="G" /> George</label>
  559. <label><input type="radio" name="beatle" value="R" /> Ringo</label>""")
  560. self.assertEqual('\n'.join(inp_set2), """<label><input checked="checked" type="radio" name="beatle" value="J" /> John</label><br />
  561. <label><input type="radio" name="beatle" value="P" /> Paul</label><br />
  562. <label><input type="radio" name="beatle" value="G" /> George</label><br />
  563. <label><input type="radio" name="beatle" value="R" /> Ringo</label><br />""")
  564. self.assertEqual('\n'.join(inp_set3), """<p><input checked="checked" type="radio" name="beatle" value="J" /> John</p>
  565. <p><input type="radio" name="beatle" value="P" /> Paul</p>
  566. <p><input type="radio" name="beatle" value="G" /> George</p>
  567. <p><input type="radio" name="beatle" value="R" /> Ringo</p>""")
  568. self.assertEqual('\n'.join(inp_set4), """beatle J J John True
  569. beatle J P Paul False
  570. beatle J G George False
  571. beatle J R Ringo False""")
  572. # You can create your own custom renderers for RadioSelect to use.
  573. class MyRenderer(RadioFieldRenderer):
  574. def render(self):
  575. return u'<br />\n'.join([unicode(choice) for choice in self])
  576. w = RadioSelect(renderer=MyRenderer)
  577. self.assertEqual(w.render('beatle', 'G', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<label><input type="radio" name="beatle" value="J" /> John</label><br />
  578. <label><input type="radio" name="beatle" value="P" /> Paul</label><br />
  579. <label><input checked="checked" type="radio" name="beatle" value="G" /> George</label><br />
  580. <label><input type="radio" name="beatle" value="R" /> Ringo</label>""")
  581. # Or you can use custom RadioSelect fields that use your custom renderer.
  582. class CustomRadioSelect(RadioSelect):
  583. renderer = MyRenderer
  584. w = CustomRadioSelect()
  585. self.assertEqual(w.render('beatle', 'G', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<label><input type="radio" name="beatle" value="J" /> John</label><br />
  586. <label><input type="radio" name="beatle" value="P" /> Paul</label><br />
  587. <label><input checked="checked" type="radio" name="beatle" value="G" /> George</label><br />
  588. <label><input type="radio" name="beatle" value="R" /> Ringo</label>""")
  589. # A RadioFieldRenderer object also allows index access to individual RadioInput
  590. w = RadioSelect()
  591. r = w.get_renderer('beatle', 'J', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo')))
  592. self.assertEqual(str(r[1]), '<label><input type="radio" name="beatle" value="P" /> Paul</label>')
  593. self.assertEqual(str(r[0]), '<label><input checked="checked" type="radio" name="beatle" value="J" /> John</label>')
  594. self.assertTrue(r[0].is_checked())
  595. self.assertFalse(r[1].is_checked())
  596. self.assertEqual((r[1].name, r[1].value, r[1].choice_value, r[1].choice_label), ('beatle', u'J', u'P', u'Paul'))
  597. try:
  598. r[10]
  599. self.fail("This offset should not exist.")
  600. except IndexError:
  601. pass
  602. # Choices are escaped correctly
  603. w = RadioSelect()
  604. self.assertEqual(w.render('escape', None, choices=(('bad', 'you & me'), ('good', mark_safe('you &gt; me')))), """<ul>
  605. <li><label><input type="radio" name="escape" value="bad" /> you &amp; me</label></li>
  606. <li><label><input type="radio" name="escape" value="good" /> you &gt; me</label></li>
  607. </ul>""")
  608. # Unicode choices are correctly rendered as HTML
  609. w = RadioSelect()
  610. self.assertEqual(unicode(w.render('email', 'Š??Ž?žš?', choices=[('Š??Ž?žš?', 'Š?abc?Ž?žš?'), ('?žš?', 'abc?žš?')])), u'<ul>\n<li><label><input checked="checked" type="radio" name="email" value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" /> \u0160\u0110abc\u0106\u017d\u0107\u017e\u0161\u0111</label></li>\n<li><label><input type="radio" name="email" value="\u0107\u017e\u0161\u0111" /> abc\u0107\u017e\u0161\u0111</label></li>\n</ul>')
  611. # Attributes provided at instantiation are passed to the constituent inputs
  612. w = RadioSelect(attrs={'id':'foo'})
  613. self.assertEqual(w.render('beatle', 'J', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  614. <li><label for="foo_0"><input checked="checked" type="radio" id="foo_0" value="J" name="beatle" /> John</label></li>
  615. <li><label for="foo_1"><input type="radio" id="foo_1" value="P" name="beatle" /> Paul</label></li>
  616. <li><label for="foo_2"><input type="radio" id="foo_2" value="G" name="beatle" /> George</label></li>
  617. <li><label for="foo_3"><input type="radio" id="foo_3" value="R" name="beatle" /> Ringo</label></li>
  618. </ul>""")
  619. # Attributes provided at render-time are passed to the constituent inputs
  620. w = RadioSelect()
  621. self.assertEqual(w.render('beatle', 'J', choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo')), attrs={'id':'bar'}), """<ul>
  622. <li><label for="bar_0"><input checked="checked" type="radio" id="bar_0" value="J" name="beatle" /> John</label></li>
  623. <li><label for="bar_1"><input type="radio" id="bar_1" value="P" name="beatle" /> Paul</label></li>
  624. <li><label for="bar_2"><input type="radio" id="bar_2" value="G" name="beatle" /> George</label></li>
  625. <li><label for="bar_3"><input type="radio" id="bar_3" value="R" name="beatle" /> Ringo</label></li>
  626. </ul>""")
  627. def test_checkboxselectmultiple(self):
  628. w = CheckboxSelectMultiple()
  629. self.assertEqual(w.render('beatles', ['J'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  630. <li><label><input checked="checked" type="checkbox" name="beatles" value="J" /> John</label></li>
  631. <li><label><input type="checkbox" name="beatles" value="P" /> Paul</label></li>
  632. <li><label><input type="checkbox" name="beatles" value="G" /> George</label></li>
  633. <li><label><input type="checkbox" name="beatles" value="R" /> Ringo</label></li>
  634. </ul>""")
  635. self.assertEqual(w.render('beatles', ['J', 'P'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  636. <li><label><input checked="checked" type="checkbox" name="beatles" value="J" /> John</label></li>
  637. <li><label><input checked="checked" type="checkbox" name="beatles" value="P" /> Paul</label></li>
  638. <li><label><input type="checkbox" name="beatles" value="G" /> George</label></li>
  639. <li><label><input type="checkbox" name="beatles" value="R" /> Ringo</label></li>
  640. </ul>""")
  641. self.assertEqual(w.render('beatles', ['J', 'P', 'R'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  642. <li><label><input checked="checked" type="checkbox" name="beatles" value="J" /> John</label></li>
  643. <li><label><input checked="checked" type="checkbox" name="beatles" value="P" /> Paul</label></li>
  644. <li><label><input type="checkbox" name="beatles" value="G" /> George</label></li>
  645. <li><label><input checked="checked" type="checkbox" name="beatles" value="R" /> Ringo</label></li>
  646. </ul>""")
  647. # If the value is None, none of the options are selected:
  648. self.assertEqual(w.render('beatles', None, choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  649. <li><label><input type="checkbox" name="beatles" value="J" /> John</label></li>
  650. <li><label><input type="checkbox" name="beatles" value="P" /> Paul</label></li>
  651. <li><label><input type="checkbox" name="beatles" value="G" /> George</label></li>
  652. <li><label><input type="checkbox" name="beatles" value="R" /> Ringo</label></li>
  653. </ul>""")
  654. # If the value corresponds to a label (but not to an option value), none of the options are selected:
  655. self.assertEqual(w.render('beatles', ['John'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  656. <li><label><input type="checkbox" name="beatles" value="J" /> John</label></li>
  657. <li><label><input type="checkbox" name="beatles" value="P" /> Paul</label></li>
  658. <li><label><input type="checkbox" name="beatles" value="G" /> George</label></li>
  659. <li><label><input type="checkbox" name="beatles" value="R" /> Ringo</label></li>
  660. </ul>""")
  661. # If multiple values are given, but some of them are not valid, the valid ones are selected:
  662. self.assertEqual(w.render('beatles', ['J', 'G', 'foo'], choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), """<ul>
  663. <li><label><input checked="checked" type="checkbox" name="beatles" value="J" /> John</label></li>
  664. <li><label><input type="checkbox" name="beatles" value="P" /> Paul</label></li>
  665. <li><label><input checked="checked" type="checkbox" name="beatles" value="G" /> George</label></li>
  666. <li><label><input type="checkbox" name="beatles" value="R" /> Ringo</label></li>
  667. </ul>""")
  668. # The value is compared to its str():
  669. self.assertEqual(w.render('nums', [2], choices=[('1', '1'), ('2', '2'), ('3', '3')]), """<ul>
  670. <li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>
  671. <li><label><input checked="checked" type="checkbox" name="nums" value="2" /> 2</label></li>
  672. <li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>
  673. </ul>""")
  674. self.assertEqual(w.render('nums', ['2'], choices=[(1, 1), (2, 2), (3, 3)]), """<ul>
  675. <li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>
  676. <li><label><input checked="checked" type="checkbox" name="nums" value="2" /> 2</label></li>
  677. <li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>
  678. </ul>""")
  679. self.assertEqual(w.render('nums', [2], choices=[(1, 1), (2, 2), (3, 3)]), """<ul>
  680. <li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>
  681. <li><label><input checked="checked" type="checkbox" name="nums" value="2" /> 2</label></li>
  682. <li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>
  683. </ul>""")
  684. # The 'choices' argument can be any iterable:
  685. def get_choices():
  686. for i in range(5):
  687. yield (i, i)
  688. self.assertEqual(w.render('nums', [2], choices=get_choices()), """<ul>
  689. <li><label><input type="checkbox" name="nums" value="0" /> 0</label></li>
  690. <li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>
  691. <li><label><input checked="checked" type="checkbox" name="nums" value="2" /> 2</label></li>
  692. <li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>
  693. <li><label><input type="checkbox" name="nums" value="4" /> 4</label></li>
  694. </ul>""")
  695. # You can also pass 'choices' to the constructor:
  696. w = CheckboxSelectMultiple(choices=[(1, 1), (2, 2), (3, 3)])
  697. self.assertEqual(w.render('nums', [2]), """<ul>
  698. <li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>
  699. <li><label><input checked="checked" type="checkbox" name="nums" value="2" /> 2</label></li>
  700. <li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>
  701. </ul>""")
  702. # If 'choices' is passed to both the constructor and render(), then they'll both be in the output:
  703. self.assertEqual(w.render('nums', [2], choices=[(4, 4), (5, 5)]), """<ul>
  704. <li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>
  705. <li><label><input checked="checked" type="checkbox" name="nums" value="2" /> 2</label></li>
  706. <li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>
  707. <li><label><input type="checkbox" name="nums" value="4" /> 4</label></li>
  708. <li><label><input type="checkbox" name="nums" value="5" /> 5</label></li>
  709. </ul>""")
  710. # Choices are escaped correctly
  711. self.assertEqual(w.render('escape', None, choices=(('bad', 'you & me'), ('good', mark_safe('you &gt; me')))), """<ul>
  712. <li><label><input type="checkbox" name="escape" value="1" /> 1</label></li>
  713. <li><label><input type="checkbox" name="escape" value="2" /> 2</label></li>
  714. <li><label><input type="checkbox" name="escape" value="3" /> 3</label></li>
  715. <li><label><input type="checkbox" name="escape" value="bad" /> you &amp; me</label></li>
  716. <li><label><input type="checkbox" name="escape" value="good" /> you &gt; me</label></li>
  717. </ul>""")
  718. # Test the usage of _has_changed
  719. self.assertFalse(w._has_changed(None, None))
  720. self.assertFalse(w._has_changed([], None))
  721. self.assertTrue(w._has_changed(None, [u'1']))
  722. self.assertFalse(w._has_changed([1, 2], [u'1', u'2']))
  723. self.assertTrue(w._has_changed([1, 2], [u'1']))
  724. self.assertTrue(w._has_changed([1, 2], [u'1', u'3']))
  725. self.assertFalse(w._has_changed([2, 1], [u'1', u'2']))
  726. # Unicode choices are correctly rendered as HTML
  727. self.assertEqual(w.render('nums', ['Š??Ž?žš?'], choices=[('Š??Ž?žš?', 'Š?abc?Ž?žš?'), ('?žš?', 'abc?žš?')]), u'<ul>\n<li><label><input type="checkbox" name="nums" value="1" /> 1</label></li>\n<li><label><input type="checkbox" name="nums" value="2" /> 2</label></li>\n<li><label><input type="checkbox" name="nums" value="3" /> 3</label></li>\n<li><label><input checked="checked" type="checkbox" name="nums" value="\u0160\u0110\u0106\u017d\u0107\u017e\u0161\u0111" /> \u0160\u0110abc\u0106\u017d\u0107\u017e\u0161\u0111</label></li>\n<li><label><input type="checkbox" name="nums" value="\u0107\u017e\u0161\u0111" /> abc\u0107\u017e\u0161\u0111</label></li>\n</ul>')
  728. # Each input gets a separate ID
  729. self.assertEqual(CheckboxSelectMultiple().render('letters', list('ac'), choices=zip(list('abc'), list('ABC')), attrs={'id': 'abc'}), """<ul>
  730. <li><label for="abc_0"><input checked="checked" type="checkbox" name="letters" value="a" id="abc_0" /> A</label></li>
  731. <li><label for="abc_1"><input type="checkbox" name="letters" value="b" id="abc_1" /> B</label></li>
  732. <li><label for="abc_2"><input checked="checked" type="checkbox" name="letters" value="c" id="abc_2" /> C</label></li>
  733. </ul>""")
  734. def test_multi(self):
  735. class MyMultiWidget(MultiWidget):
  736. def decompress(self, value):
  737. if value:
  738. return value.split('__')
  739. return ['', '']
  740. def format_output(self, rendered_widgets):
  741. return u'<br />'.join(rendered_widgets)
  742. w = MyMultiWidget(widgets=(TextInput(attrs={'class': 'big'}), TextInput(attrs={'class': 'small'})))
  743. self.assertEqual(w.render('name', ['john', 'lennon']), u'<input type="text" class="big" value="john" name="name_0" /><br /><input type="text" class="small" value="lennon" name="name_1" />')
  744. self.assertEqual(w.render('name', 'john__lennon'), u'<input type="text" class="big" value="john" name="name_0" /><br /><input type="text" class="small" value="lennon" name="name_1" />')
  745. self.assertEqual(w.render('name', 'john__lennon', attrs={'id':'foo'}), u'<input id="foo_0" type="text" class="big" value="john" name="name_0" /><br /><input id="foo_1" type="text" class="small" value="lennon" name="name_1" />')
  746. w = MyMultiWidget(widgets=(TextInput(attrs={'class': 'big'}), TextInput(attrs={'class': 'small'})), attrs={'id': 'bar'})
  747. self.assertEqual(w.render('name', ['john', 'lennon']), u'<input id="bar_0" type="text" class="big" value="john" name="name_0" /><br /><input id="bar_1" type="text" class="small" value="lennon" name="name_1" />')
  748. w = MyMultiWidget(widgets=(TextInput(), TextInput()))
  749. # test with no initial data
  750. self.assertTrue(w._has_changed(None, [u'john', u'lennon']))
  751. # test when the data is the same as initial
  752. self.assertFalse(w._has_changed(u'john__lennon', [u'john', u'lennon']))
  753. # test when the first widget's data has changed
  754. self.assertTrue(w._has_changed(u'john__lennon', [u'alfred', u'lennon']))
  755. # test when the last widget's data has changed. this ensures that it is not
  756. # short circuiting while testing the widgets.
  757. self.assertTrue(w._has_changed(u'john__lennon', [u'john', u'denver']))
  758. def test_splitdatetime(self):
  759. w = SplitDateTimeWidget()
  760. self.assertEqual(w.render('date', ''), u'<input type="text" name="date_0" /><input type="text" name="date_1" />')
  761. self.assertEqual(w.render('date', None), u'<input type="text" name="date_0" /><input type="text" name="date_1" />')
  762. self.assertEqual(w.render('date', datetime.datetime(2006, 1, 10, 7, 30)), u'<input type="text" name="date_0" value="2006-01-10" /><input type="text" name="date_1" value="07:30:00" />')
  763. self.assertEqual(w.render('date', [datetime.date(2006, 1, 10), datetime.time(7, 30)]), u'<input type="text" name="date_0" value="2006-01-10" /><input type="text" name="date_1" value="07:30:00" />')
  764. # You can also pass 'attrs' to the constructor. In this case, the attrs will be
  765. w = SplitDateTimeWidget(attrs={'class': 'pretty'})
  766. self.assertEqual(w.render('date', datetime.datetime(2006, 1, 10, 7, 30)), u'<input type="text" class="pretty" value="2006-01-10" name="date_0" /><input type="text" class="pretty" value="07:30:00" name="date_1" />')
  767. # Use 'date_format' and 'time_format' to change the way a value is displayed.
  768. w = SplitDateTimeWidget(date_format='%d/%m/%Y', time_format='%H:%M')
  769. self.assertEqual(w.render('date', datetime.datetime(2006, 1, 10, 7, 30)), u'<input type="text" name="date_0" value="10/01/2006" /><input type="text" name="date_1" value="07:30" />')
  770. self.assertTrue(w._has_changed(datetime.datetime(2008, 5, 6, 12, 40, 00), [u'2008-05-06', u'12:40:00']))
  771. self.assertFalse(w._has_changed(datetime.datetime(2008, 5, 6, 12, 40, 00), [u'06/05/2008', u'12:40']))
  772. self.assertTrue(w._has_changed(datetime.datetime(2008, 5, 6, 12, 40, 00), [u'06/05/2008', u'12:41']))
  773. def test_datetimeinput(self):
  774. w = DateTimeInput()
  775. self.assertEqual(w.render('date', None), u'<input type="text" name="date" />')
  776. d = datetime.datetime(2007, 9, 17, 12, 51, 34, 482548)
  777. self.assertEqual(str(d), '2007-09-17 12:51:34.482548')
  778. # The microseconds are trimmed on display, by default.
  779. self.assertEqual(w.render('date', d), u'<input type="text" name="date" value="2007-09-17 12:51:34" />')
  780. self.assertEqual(w.render('date', datetime.datetime(2007, 9, 17, 12, 51, 34)), u'<input type="text" name="date" value="2007-09-17 12:51:34" />')
  781. self.assertEqual(w.render('date', datetime.datetime(2007, 9, 17, 12, 51)), u'<input type="text" name="date" value="2007-09-17 12:51:00" />')
  782. # Use 'format' to change the way a value is displayed.
  783. w = DateTimeInput(format='%d/%m/%Y %H:%M')
  784. self.assertEqual(w.render('date', d), u'<input type="text" name="date" value="17/09/2007 12:51" />')
  785. self.assertFalse(w._has_changed(d, '17/09/2007 12:51'))
  786. # Make sure a custom format works with _has_changed. The hidden input will use
  787. data = datetime.datetime(2010, 3, 6, 12, 0, 0)
  788. custom_format = '%d.%m.%Y %H:%M'
  789. w = DateTimeInput(format=custom_format)
  790. self.assertFalse(w._has_changed(formats.localize_input(data), data.strftime(custom_format)))
  791. def test_dateinput(self):
  792. w = DateInput()
  793. self.assertEqual(w.render('date', None), u'<input type="text" name="date" />')
  794. d = datetime.date(2007, 9, 17)
  795. self.assertEqual(str(d), '2007-09-17')
  796. self.assertEqual(w.render('date', d), u'<input type="text" name="date" value="2007-09-17" />')
  797. self.assertEqual(w.render('date', datetime.date(2007, 9, 17)), u'<input type="text" name="date" value="2007-09-17" />')
  798. # We should be able to initialize from a unicode value.
  799. self.assertEqual(w.render('date', u'2007-09-17'), u'<input type="text" name="date" value="2007-09-17" />')
  800. # Use 'format' to change the way a value is displayed.
  801. w = DateInput(format='%d/%m/%Y')
  802. self.assertEqual(w.render('date', d), u'<input type="text" name="date" value="17/09/2007" />')
  803. self.assertFalse(w._has_changed(d, '17/09/2007'))
  804. # Make sure a custom format works with _has_changed. The hidden input will use
  805. data = datetime.date(2010, 3, 6)
  806. custom_format = '%d.%m.%Y'
  807. w = DateInput(format=custom_format)
  808. self.assertFalse(w._has_changed(formats.localize_input(data), data.strftime(custom_format)))
  809. def test_timeinput(self):
  810. w = TimeInput()
  811. self.assertEqual(w.render('time', None), u'<input type="text" name="time" />')
  812. t = datetime.time(12, 51, 34, 482548)
  813. self.assertEqual(str(t), '12:51:34.482548')
  814. # The microseconds are trimmed on display, by default.
  815. self.assertEqual(w.render('time', t), u'<input type="text" name="time" value="12:51:34" />')
  816. self.assertEqual(w.render('time', datetime.time(12, 51, 34)), u'<input type="text" name="time" value="12:51:34" />')
  817. self.assertEqual(w.render('time', datetime.time(12, 51)), u'<input type="text" name="time" value="12:51:00" />')
  818. # We should be able to initialize from a unicode value.
  819. self.assertEqual(w.render('time', u'13:12:11'), u'<input type="text" name="time" value="13:12:11" />')
  820. # Use 'format' to change the way a value is displayed.
  821. w = TimeInput(format='%H:%M')
  822. self.assertEqual(w.render('time', t), u'<input type="text" name="time" value="12:51" />')
  823. self.assertFalse(w._has_changed(t, '12:51'))
  824. # Make sure a custom format works with _has_changed. The hidden input will use
  825. data = datetime.time(13, 0)
  826. custom_format = '%I:%M %p'
  827. w = TimeInput(format=custom_format)
  828. self.assertFalse(w._has_changed(formats.localize_input(data), data.strftime(custom_format)))
  829. def test_splithiddendatetime(self):
  830. from django.forms.widgets import SplitHiddenDateTimeWidget
  831. w = SplitHiddenDateTimeWidget()
  832. self.assertEqual(w.render('date', ''), u'<input type="hidden" name="date_0" /><input type="hidden" name="date_1" />')
  833. d = datetime.datetime(2007, 9, 17, 12, 51, 34, 482548)
  834. self.assertEqual(str(d), '2007-09-17 12:51:34.482548')
  835. self.assertEqual(w.render('date', d), u'<input type="hidden" name="date_0" value="2007-09-17" /><input type="hidden" name="date_1" value="12:51:34" />')
  836. self.assertEqual(w.render('date', datetime.datetime(2007, 9, 17, 12, 51, 34)), u'<input type="hidden" name="date_0" value="2007-09-17" /><input type="hidden" name="date_1" value="12:51:34" />')
  837. self.assertEqual(w.render('date', datetime.datetime(2007, 9, 17, 12, 51)), u'<input type="hidden" name="date_0" value="2007-09-17" /><input type="hidden" name="date_1" value="12:51:00" />')
  838. class FormsI18NWidgetsTestCase(TestCase):
  839. def setUp(self):
  840. super(FormsI18NWidgetsTestCase, self).setUp()
  841. self.old_use_l10n = getattr(settings, 'USE_L10N', False)
  842. settings.USE_L10N = True
  843. activate('de-at')
  844. def tearDown(self):
  845. deactivate()
  846. settings.USE_L10N = self.old_use_l10n
  847. super(FormsI18NWidgetsTestCase, self).tearDown()
  848. def test_splitdatetime(self):
  849. w = SplitDateTimeWidget(date_format='%d/%m/%Y', time_format='%H:%M')
  850. self.assertTrue(w._has_changed(datetime.datetime(2008, 5, 6, 12, 40, 00), [u'06.05.2008', u'12:41']))
  851. def test_datetimeinput(self):
  852. w = DateTimeInput()
  853. d = datetime.datetime(2007, 9, 17, 12, 51, 34, 482548)
  854. w.is_localized = True
  855. self.assertEqual(w.render('date', d), u'<input type="text" name="date" value="17.09.2007 12:51:34" />')
  856. def test_dateinput(self):
  857. w = DateInput()
  858. d = datetime.date(2007, 9, 17)
  859. w.is_localized = True
  860. self.assertEqual(w.render('date', d), u'<input type="text" name="date" value="17.09.2007" />')
  861. def test_timeinput(self):
  862. w = TimeInput()
  863. t = datetime.time(12, 51, 34, 482548)
  864. w.is_localized = True
  865. self.assertEqual(w.render('time', t), u'<input type="text" name="time" value="12:51:34" />')
  866. def test_splithiddendatetime(self):
  867. from django.forms.widgets import SplitHiddenDateTimeWidget
  868. w = SplitHiddenDateTimeWidget()
  869. w.is_localized = True
  870. self.assertEqual(w.render('date', datetime.datetime(2007, 9, 17, 12, 51)), u'<input type="hidden" name="date_0" value="17.09.2007" /><input type="hidden" name="date_1" value="12:51:00" />')
  871. class SelectAndTextWidget(MultiWidget):
  872. """
  873. MultiWidget subclass
  874. """
  875. def __init__(self, choices=[]):
  876. widgets = [
  877. RadioSelect(choices=choices),
  878. TextInput
  879. ]
  880. super(SelectAndTextWidget, self).__init__(widgets)
  881. def _set_choices(self, choices):
  882. """
  883. When choices are set for this widget, we want to pass those along to the Select widget
  884. """
  885. self.widgets[0].choices = choices
  886. def _get_choices(self):
  887. """
  888. The choices for this widget are the Select widget's choices
  889. """
  890. return self.widgets[0].choices
  891. choices = property(_get_choices, _set_choices)
  892. class WidgetTests(TestCase):
  893. def test_12048(self):
  894. # See ticket #12048.
  895. w1 = SelectAndTextWidget(choices=[1,2,3])
  896. w2 = copy.deepcopy(w1)
  897. w2.choices = [4,5,6]
  898. # w2 ought to be independent of w1, since MultiWidget ought
  899. # to make a copy of its sub-widgets when it is copied.
  900. self.assertEqual(w1.choices, [1,2,3])
  901. def test_13390(self):
  902. # See ticket #13390
  903. class SplitDateForm(Form):
  904. field = DateTimeField(widget=SplitDateTimeWidget, required=False)
  905. form = SplitDateForm({'field': ''})
  906. self.assertTrue(form.is_valid())
  907. form = SplitDateForm({'field': ['', '']})
  908. self.assertTrue(form.is_valid())
  909. class SplitDateRequiredForm(Form):
  910. field = DateTimeField(widget=SplitDateTimeWidget, required=True)
  911. form = SplitDateRequiredForm({'field': ''})
  912. self.assertFalse(form.is_valid())
  913. form = SplitDateRequiredForm({'field': ['', '']})
  914. self.assertFalse(form.is_valid())
  915. class FakeFieldFile(object):
  916. """
  917. Quacks like a FieldFile (has a .url and unicode representation), but
  918. doesn't require us to care about storages etc.
  919. """
  920. url = 'something'
  921. def __unicode__(self):
  922. return self.url
  923. class ClearableFileInputTests(TestCase):
  924. def test_clear_input_renders(self):
  925. """
  926. A ClearableFileInput with is_required False and rendered with
  927. an initial value that is a file renders a clear checkbox.
  928. """
  929. widget = ClearableFileInput()
  930. widget.is_required = False
  931. self.assertEqual(widget.render('myfile', FakeFieldFile()),
  932. u'Currently: <a href="something">something</a> <input type="checkbox" name="myfile-clear" id="myfile-clear_id" /> <label for="myfile-clear_id">Clear</label><br />Change: <input type="file" name="myfile" />')
  933. def test_html_escaped(self):
  934. """
  935. A ClearableFileInput should escape name, filename and URL when
  936. rendering HTML. Refs #15182.
  937. """
  938. class StrangeFieldFile(object):
  939. url = "something?chapter=1&sect=2&copy=3&lang=en"
  940. def __unicode__(self):
  941. return u'''something<div onclick="alert('oops')">.jpg'''
  942. widget = ClearableFileInput()
  943. field = StrangeFieldFile()
  944. output = widget.render('my<div>file', field)
  945. self.assertFalse(field.url in output)
  946. self.assertTrue(u'href="something?chapter=1&amp;sect=2&amp;copy=3&amp;lang=en"' in output)
  947. self.assertFalse(unicode(field) in output)
  948. self.assertTrue(u'something&lt;div onclick=&quot;alert(&#39;oops&#39;)&quot;&gt;.jpg' in output)
  949. self.assertTrue(u'my&lt;div&gt;file' in output)
  950. self.assertFalse(u'my<div>file' in output)
  951. def test_clear_input_renders_only_if_not_required(self):
  952. """
  953. A ClearableFileInput with is_required=False does not render a clear
  954. checkbox.
  955. """
  956. widget = ClearableFileInput()
  957. widget.is_required = True
  958. self.assertEqual(widget.render('myfile', FakeFieldFile()),
  959. u'Currently: <a href="something">something</a> <br />Change: <input type="file" name="myfile" />')
  960. def test_clear_input_renders_only_if_initial(self):
  961. """
  962. A ClearableFileInput instantiated with no initial value does not render
  963. a clear checkbox.
  964. """
  965. widget = ClearableFileInput()
  966. widget.is_required = False
  967. self.assertEqual(widget.render('myfile', None),
  968. u'<input type="file" name="myfile" />')
  969. def test_clear_input_checked_returns_false(self):
  970. """
  971. ClearableFileInput.value_from_datadict returns False if the clear
  972. checkbox is checked, if not required.
  973. """
  974. widget = ClearableFileInput()
  975. widget.is_required = False
  976. self.assertEqual(widget.value_from_datadict(
  977. data={'myfile-clear': True},
  978. files={},
  979. name='myfile'), False)
  980. def test_clear_input_checked_returns_false_only_if_not_required(self):
  981. """
  982. ClearableFileInput.value_from_datadict never returns False if the field
  983. is required.
  984. """
  985. widget = ClearableFileInput()
  986. widget.is_required = True
  987. f = SimpleUploadedFile('something.txt', 'content')
  988. self.assertEqual(widget.value_from_datadict(
  989. data={'myfile-clear': True},
  990. files={'myfile': f},
  991. name='myfile'), f)