You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

test_widget.py 7.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. import datetime
  2. from django import forms
  3. from django.test import TestCase
  4. from django.urls import reverse_lazy
  5. from oscar.apps.catalogue.models import Product
  6. from oscar.forms import widgets
  7. from oscar.test.factories import create_product
  8. class ImageInputTestCase(TestCase):
  9. def test_unbound_context(self):
  10. i = widgets.ImageInput()
  11. ctx = i.get_context('test_input', None, {'id': 'test-image-id'})
  12. self.assertEqual(ctx['image_id'], 'test-image-id-image')
  13. self.assertEqual(ctx['image_url'], '')
  14. self.assertEqual(
  15. ctx['widget']['attrs'],
  16. {'accept': 'image/*', 'id': 'test-image-id'}
  17. )
  18. def test_bound_context(self):
  19. i = widgets.ImageInput()
  20. ctx = i.get_context('test_input', '/dummy-image-value', {'id': 'test-image-id'})
  21. self.assertEqual(ctx['image_url'], '/dummy-image-value')
  22. class TimePickerInputTestCase(TestCase):
  23. def test_div_attrs_context(self):
  24. i = widgets.TimePickerInput(format='%H:%M')
  25. ctx = i.get_context('test_input', None, {})
  26. self.assertEqual(ctx['div_attrs'], {
  27. 'data-oscarWidget': 'time',
  28. 'data-timeFormat': 'HH:mm',
  29. })
  30. def test_icon_classes_context(self):
  31. i = widgets.TimePickerInput(format='%H:%M')
  32. ctx = i.get_context('test_input', None, {})
  33. self.assertEqual(ctx['icon_classes'], 'far fa-clock')
  34. def test_input_format_unicode(self):
  35. # Check that the widget can handle unicode formats
  36. i = widgets.TimePickerInput(format='τ-%H:%M')
  37. time = datetime.time(10, 47)
  38. html = i.render('time', time)
  39. self.assertIn('value="τ-10:47"', html)
  40. class DatePickerInputTestCase(TestCase):
  41. def test_div_attrs_context(self):
  42. i = widgets.DatePickerInput(format='%d/%m/%Y')
  43. ctx = i.get_context('test_input', None, {})
  44. self.assertEqual(ctx['div_attrs'], {
  45. 'data-oscarWidget': 'date',
  46. 'data-dateFormat': 'DD/MM/YYYY',
  47. })
  48. def test_icon_classes_context(self):
  49. i = widgets.DatePickerInput(format='%H:%M')
  50. ctx = i.get_context('test_input', None, {})
  51. self.assertEqual(ctx['icon_classes'], 'far fa-calendar-alt')
  52. def test_datepickerinput_format_unicode(self):
  53. # Check that the widget can handle unicode formats
  54. i = widgets.DatePickerInput(format='δ-%d/%m/%Y')
  55. date = datetime.date(2017, 5, 1)
  56. html = i.render('date', date)
  57. self.assertIn('value="δ-01/05/2017"', html)
  58. class DateTimePickerInputTestCase(TestCase):
  59. def test_div_attrs_context(self):
  60. i = widgets.DateTimePickerInput(format='%d/%m/%Y %H:%M')
  61. ctx = i.get_context('test_input', None, {})
  62. self.assertEqual(ctx['div_attrs'], {
  63. 'data-oscarWidget': 'datetime',
  64. 'data-datetimeFormat': 'DD/MM/YYYY HH:mm',
  65. })
  66. def test_icon_classes_context(self):
  67. i = widgets.DateTimePickerInput(format='%d/%m/%Y %H:%M')
  68. ctx = i.get_context('test_input', None, {})
  69. self.assertEqual(ctx['icon_classes'], 'far fa-calendar-alt')
  70. def test_datetimepickerinput_format_unicode(self):
  71. # Check that the widget can handle unicode formats
  72. i = widgets.DateTimePickerInput(format='δ-%d/%m/%Y %H:%M')
  73. date = datetime.datetime(2017, 5, 1, 10, 57)
  74. html = i.render('datetime', date)
  75. self.assertIn('value="δ-01/05/2017 10:57"', html)
  76. class TestWidgetsDatetimeFormat(TestCase):
  77. def test_datetime_to_date_format_conversion(self):
  78. format_testcases = (
  79. ('%Y-%m-%d', 'YYYY-MM-DD'),
  80. ('%Y-%m-%d %H:%M', 'YYYY-MM-DD'),
  81. )
  82. for format_, expected in format_testcases:
  83. self.assertEqual(widgets.datetime_format_to_js_date_format(format_), expected)
  84. def test_datetime_to_time_format_conversion(self):
  85. format_testcases = (
  86. ('%Y-%m-%d %H:%M', 'HH:mm'),
  87. ('%H:%M', 'HH:mm'),
  88. )
  89. for format_, expected in format_testcases:
  90. self.assertEqual(widgets.datetime_format_to_js_time_format(format_), expected)
  91. class AdvancedSelectWidgetTestCase(TestCase):
  92. def test_widget_disabled_options(self):
  93. choices = (
  94. ('red', 'Red'),
  95. ('blue', 'Blue'),
  96. ('green', 'Green'),
  97. )
  98. disabled_values = ('red', 'green')
  99. i = widgets.AdvancedSelect(choices=choices, disabled_values=disabled_values)
  100. html = i.render('advselect', [])
  101. self.assertInHTML('<option value="blue">Blue</option>', html, count=1)
  102. self.assertInHTML('<option value="red" disabled>Red</option>', html, count=1)
  103. self.assertInHTML('<option value="green" disabled>Green</option>', html, count=1)
  104. class RemoteSelectTestCase(TestCase):
  105. def setUp(self):
  106. self.url = reverse_lazy('dashboard:catalogue-product-lookup')
  107. def _get_form_field(self, **kwargs):
  108. return forms.ModelChoiceField(
  109. queryset=Product.objects.all(),
  110. widget=widgets.RemoteSelect(lookup_url=self.url),
  111. **kwargs
  112. )
  113. def _get_multiselect_form_field(self, **kwargs):
  114. return forms.ModelChoiceField(
  115. queryset=Product.objects.all(),
  116. widget=widgets.MultipleRemoteSelect(lookup_url=self.url),
  117. **kwargs
  118. )
  119. def test_remote_url_required(self):
  120. with self.assertRaises(ValueError):
  121. widgets.RemoteSelect()
  122. def test_select_widget_renders_only_selected_choices(self):
  123. create_product()
  124. p2 = create_product()
  125. field = self._get_form_field()
  126. form_choices = list(field.widget.options(name='name', value=[p2.pk]))
  127. # We should only have one choice, not two
  128. self.assertEqual(len(form_choices), 1)
  129. self.assertEqual(form_choices[0]['value'], p2.pk)
  130. def test_widget_attrs(self):
  131. field = self._get_form_field()
  132. attrs = field.widget.get_context(name='my_field', value=None, attrs={})['widget']['attrs']
  133. self.assertEqual(attrs['data-multiple'], '')
  134. self.assertEqual(attrs['data-required'], 'required')
  135. self.assertEqual(attrs['data-ajax-url'], self.url)
  136. def test_not_required_widget_attrs(self):
  137. field = self._get_multiselect_form_field(required=False)
  138. attrs = field.widget.get_context(name='my_field', value=None, attrs={})['widget']['attrs']
  139. self.assertEqual(attrs['data-required'], '')
  140. def test_multiselect_widget_renders_only_selected_choices(self):
  141. create_product()
  142. p2 = create_product()
  143. p3 = create_product()
  144. field = self._get_multiselect_form_field()
  145. form_choices = list(field.widget.options(name='name', value=[p2.pk, p3.pk]))
  146. # We should only have two choices, not three
  147. self.assertEqual(len(form_choices), 2)
  148. def test_multiselect_widget_attrs(self):
  149. field = self._get_multiselect_form_field()
  150. attrs = field.widget.get_context(name='my_field', value=None, attrs={})['widget']['attrs']
  151. self.assertEqual(attrs['data-multiple'], 'multiple')
  152. class NullBooleanSelectTestCase(TestCase):
  153. def test_unknown_choice_label(self):
  154. self.assertEqual(dict(widgets.NullBooleanSelect().choices)['unknown'], '---------')