widgets.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. """
  2. Form Widget classes specific to the Django admin site.
  3. """
  4. import copy
  5. import json
  6. from django import forms
  7. from django.conf import settings
  8. from django.core.exceptions import ValidationError
  9. from django.core.validators import URLValidator
  10. from django.db.models import CASCADE
  11. from django.urls import reverse
  12. from django.urls.exceptions import NoReverseMatch
  13. from django.utils.html import smart_urlquote
  14. from django.utils.http import urlencode
  15. from django.utils.text import Truncator
  16. from django.utils.translation import get_language, gettext as _
  17. class FilteredSelectMultiple(forms.SelectMultiple):
  18. """
  19. A SelectMultiple with a JavaScript filter interface.
  20. Note that the resulting JavaScript assumes that the jsi18n
  21. catalog has been loaded in the page
  22. """
  23. class Media:
  24. js = [
  25. 'admin/js/core.js',
  26. 'admin/js/SelectBox.js',
  27. 'admin/js/SelectFilter2.js',
  28. ]
  29. def __init__(self, verbose_name, is_stacked, attrs=None, choices=()):
  30. self.verbose_name = verbose_name
  31. self.is_stacked = is_stacked
  32. super().__init__(attrs, choices)
  33. def get_context(self, name, value, attrs):
  34. context = super().get_context(name, value, attrs)
  35. context['widget']['attrs']['class'] = 'selectfilter'
  36. if self.is_stacked:
  37. context['widget']['attrs']['class'] += 'stacked'
  38. context['widget']['attrs']['data-field-name'] = self.verbose_name
  39. context['widget']['attrs']['data-is-stacked'] = int(self.is_stacked)
  40. return context
  41. class AdminDateWidget(forms.DateInput):
  42. class Media:
  43. js = [
  44. 'admin/js/calendar.js',
  45. 'admin/js/admin/DateTimeShortcuts.js',
  46. ]
  47. def __init__(self, attrs=None, format=None):
  48. attrs = {'class': 'vDateField', 'size': '10', **(attrs or {})}
  49. super().__init__(attrs=attrs, format=format)
  50. class AdminTimeWidget(forms.TimeInput):
  51. class Media:
  52. js = [
  53. 'admin/js/calendar.js',
  54. 'admin/js/admin/DateTimeShortcuts.js',
  55. ]
  56. def __init__(self, attrs=None, format=None):
  57. attrs = {'class': 'vTimeField', 'size': '8', **(attrs or {})}
  58. super().__init__(attrs=attrs, format=format)
  59. class AdminSplitDateTime(forms.SplitDateTimeWidget):
  60. """
  61. A SplitDateTime Widget that has some admin-specific styling.
  62. """
  63. template_name = 'admin/widgets/split_datetime.html'
  64. def __init__(self, attrs=None):
  65. widgets = [AdminDateWidget, AdminTimeWidget]
  66. # Note that we're calling MultiWidget, not SplitDateTimeWidget, because
  67. # we want to define widgets.
  68. forms.MultiWidget.__init__(self, widgets, attrs)
  69. def get_context(self, name, value, attrs):
  70. context = super().get_context(name, value, attrs)
  71. context['date_label'] = _('Date:')
  72. context['time_label'] = _('Time:')
  73. return context
  74. class AdminRadioSelect(forms.RadioSelect):
  75. template_name = 'admin/widgets/radio.html'
  76. class AdminFileWidget(forms.ClearableFileInput):
  77. template_name = 'admin/widgets/clearable_file_input.html'
  78. def url_params_from_lookup_dict(lookups):
  79. """
  80. Convert the type of lookups specified in a ForeignKey limit_choices_to
  81. attribute to a dictionary of query parameters
  82. """
  83. params = {}
  84. if lookups and hasattr(lookups, 'items'):
  85. for k, v in lookups.items():
  86. if callable(v):
  87. v = v()
  88. if isinstance(v, (tuple, list)):
  89. v = ','.join(str(x) for x in v)
  90. elif isinstance(v, bool):
  91. v = ('0', '1')[v]
  92. else:
  93. v = str(v)
  94. params[k] = v
  95. return params
  96. class ForeignKeyRawIdWidget(forms.TextInput):
  97. """
  98. A Widget for displaying ForeignKeys in the "raw_id" interface rather than
  99. in a <select> box.
  100. """
  101. template_name = 'admin/widgets/foreign_key_raw_id.html'
  102. def __init__(self, rel, admin_site, attrs=None, using=None):
  103. self.rel = rel
  104. self.admin_site = admin_site
  105. self.db = using
  106. super().__init__(attrs)
  107. def get_context(self, name, value, attrs):
  108. context = super().get_context(name, value, attrs)
  109. rel_to = self.rel.model
  110. if rel_to in self.admin_site._registry:
  111. # The related object is registered with the same AdminSite
  112. related_url = reverse(
  113. 'admin:%s_%s_changelist' % (
  114. rel_to._meta.app_label,
  115. rel_to._meta.model_name,
  116. ),
  117. current_app=self.admin_site.name,
  118. )
  119. params = self.url_parameters()
  120. if params:
  121. related_url += '?' + urlencode(params)
  122. context['related_url'] = related_url
  123. context['link_title'] = _('Lookup')
  124. # The JavaScript code looks for this class.
  125. context['widget']['attrs'].setdefault('class', 'vForeignKeyRawIdAdminField')
  126. else:
  127. context['related_url'] = None
  128. if context['widget']['value']:
  129. context['link_label'], context['link_url'] = self.label_and_url_for_value(value)
  130. else:
  131. context['link_label'] = None
  132. return context
  133. def base_url_parameters(self):
  134. limit_choices_to = self.rel.limit_choices_to
  135. if callable(limit_choices_to):
  136. limit_choices_to = limit_choices_to()
  137. return url_params_from_lookup_dict(limit_choices_to)
  138. def url_parameters(self):
  139. from django.contrib.admin.views.main import TO_FIELD_VAR
  140. params = self.base_url_parameters()
  141. params.update({TO_FIELD_VAR: self.rel.get_related_field().name})
  142. return params
  143. def label_and_url_for_value(self, value):
  144. key = self.rel.get_related_field().name
  145. try:
  146. obj = self.rel.model._default_manager.using(self.db).get(**{key: value})
  147. except (ValueError, self.rel.model.DoesNotExist, ValidationError):
  148. return '', ''
  149. try:
  150. url = reverse(
  151. '%s:%s_%s_change' % (
  152. self.admin_site.name,
  153. obj._meta.app_label,
  154. obj._meta.object_name.lower(),
  155. ),
  156. args=(obj.pk,)
  157. )
  158. except NoReverseMatch:
  159. url = '' # Admin not registered for target model.
  160. return Truncator(obj).words(14), url
  161. class ManyToManyRawIdWidget(ForeignKeyRawIdWidget):
  162. """
  163. A Widget for displaying ManyToMany ids in the "raw_id" interface rather than
  164. in a <select multiple> box.
  165. """
  166. template_name = 'admin/widgets/many_to_many_raw_id.html'
  167. def get_context(self, name, value, attrs):
  168. context = super().get_context(name, value, attrs)
  169. if self.rel.model in self.admin_site._registry:
  170. # The related object is registered with the same AdminSite
  171. context['widget']['attrs']['class'] = 'vManyToManyRawIdAdminField'
  172. return context
  173. def url_parameters(self):
  174. return self.base_url_parameters()
  175. def label_and_url_for_value(self, value):
  176. return '', ''
  177. def value_from_datadict(self, data, files, name):
  178. value = data.get(name)
  179. if value:
  180. return value.split(',')
  181. def format_value(self, value):
  182. return ','.join(str(v) for v in value) if value else ''
  183. class RelatedFieldWidgetWrapper(forms.Widget):
  184. """
  185. This class is a wrapper to a given widget to add the add icon for the
  186. admin interface.
  187. """
  188. template_name = 'admin/widgets/related_widget_wrapper.html'
  189. def __init__(self, widget, rel, admin_site, can_add_related=None,
  190. can_change_related=False, can_delete_related=False,
  191. can_view_related=False):
  192. self.needs_multipart_form = widget.needs_multipart_form
  193. self.attrs = widget.attrs
  194. self.choices = widget.choices
  195. self.widget = widget
  196. self.rel = rel
  197. # Backwards compatible check for whether a user can add related
  198. # objects.
  199. if can_add_related is None:
  200. can_add_related = rel.model in admin_site._registry
  201. self.can_add_related = can_add_related
  202. # XXX: The UX does not support multiple selected values.
  203. multiple = getattr(widget, 'allow_multiple_selected', False)
  204. self.can_change_related = not multiple and can_change_related
  205. # XXX: The deletion UX can be confusing when dealing with cascading deletion.
  206. cascade = getattr(rel, 'on_delete', None) is CASCADE
  207. self.can_delete_related = not multiple and not cascade and can_delete_related
  208. self.can_view_related = not multiple and can_view_related
  209. # so we can check if the related object is registered with this AdminSite
  210. self.admin_site = admin_site
  211. def __deepcopy__(self, memo):
  212. obj = copy.copy(self)
  213. obj.widget = copy.deepcopy(self.widget, memo)
  214. obj.attrs = self.widget.attrs
  215. memo[id(self)] = obj
  216. return obj
  217. @property
  218. def is_hidden(self):
  219. return self.widget.is_hidden
  220. @property
  221. def media(self):
  222. return self.widget.media
  223. def get_related_url(self, info, action, *args):
  224. return reverse("admin:%s_%s_%s" % (info + (action,)),
  225. current_app=self.admin_site.name, args=args)
  226. def get_context(self, name, value, attrs):
  227. from django.contrib.admin.views.main import IS_POPUP_VAR, TO_FIELD_VAR
  228. rel_opts = self.rel.model._meta
  229. info = (rel_opts.app_label, rel_opts.model_name)
  230. self.widget.choices = self.choices
  231. url_params = '&'.join("%s=%s" % param for param in [
  232. (TO_FIELD_VAR, self.rel.get_related_field().name),
  233. (IS_POPUP_VAR, 1),
  234. ])
  235. context = {
  236. 'rendered_widget': self.widget.render(name, value, attrs),
  237. 'is_hidden': self.is_hidden,
  238. 'name': name,
  239. 'url_params': url_params,
  240. 'model': rel_opts.verbose_name,
  241. 'can_add_related': self.can_add_related,
  242. 'can_change_related': self.can_change_related,
  243. 'can_delete_related': self.can_delete_related,
  244. 'can_view_related': self.can_view_related,
  245. }
  246. if self.can_add_related:
  247. context['add_related_url'] = self.get_related_url(info, 'add')
  248. if self.can_delete_related:
  249. context['delete_related_template_url'] = self.get_related_url(info, 'delete', '__fk__')
  250. if self.can_view_related or self.can_change_related:
  251. context['change_related_template_url'] = self.get_related_url(info, 'change', '__fk__')
  252. return context
  253. def value_from_datadict(self, data, files, name):
  254. return self.widget.value_from_datadict(data, files, name)
  255. def value_omitted_from_data(self, data, files, name):
  256. return self.widget.value_omitted_from_data(data, files, name)
  257. def id_for_label(self, id_):
  258. return self.widget.id_for_label(id_)
  259. class AdminTextareaWidget(forms.Textarea):
  260. def __init__(self, attrs=None):
  261. super().__init__(attrs={'class': 'vLargeTextField', **(attrs or {})})
  262. class AdminTextInputWidget(forms.TextInput):
  263. def __init__(self, attrs=None):
  264. super().__init__(attrs={'class': 'vTextField', **(attrs or {})})
  265. class AdminEmailInputWidget(forms.EmailInput):
  266. def __init__(self, attrs=None):
  267. super().__init__(attrs={'class': 'vTextField', **(attrs or {})})
  268. class AdminURLFieldWidget(forms.URLInput):
  269. template_name = 'admin/widgets/url.html'
  270. def __init__(self, attrs=None, validator_class=URLValidator):
  271. super().__init__(attrs={'class': 'vURLField', **(attrs or {})})
  272. self.validator = validator_class()
  273. def get_context(self, name, value, attrs):
  274. try:
  275. self.validator(value if value else '')
  276. url_valid = True
  277. except ValidationError:
  278. url_valid = False
  279. context = super().get_context(name, value, attrs)
  280. context['current_label'] = _('Currently:')
  281. context['change_label'] = _('Change:')
  282. context['widget']['href'] = smart_urlquote(context['widget']['value']) if value else ''
  283. context['url_valid'] = url_valid
  284. return context
  285. class AdminIntegerFieldWidget(forms.NumberInput):
  286. class_name = 'vIntegerField'
  287. def __init__(self, attrs=None):
  288. super().__init__(attrs={'class': self.class_name, **(attrs or {})})
  289. class AdminBigIntegerFieldWidget(AdminIntegerFieldWidget):
  290. class_name = 'vBigIntegerField'
  291. class AdminUUIDInputWidget(forms.TextInput):
  292. def __init__(self, attrs=None):
  293. super().__init__(attrs={'class': 'vUUIDField', **(attrs or {})})
  294. # Mapping of lowercase language codes [returned by Django's get_language()] to
  295. # language codes supported by select2.
  296. # See django/contrib/admin/static/admin/js/vendor/select2/i18n/*
  297. SELECT2_TRANSLATIONS = {x.lower(): x for x in [
  298. 'ar', 'az', 'bg', 'ca', 'cs', 'da', 'de', 'el', 'en', 'es', 'et',
  299. 'eu', 'fa', 'fi', 'fr', 'gl', 'he', 'hi', 'hr', 'hu', 'id', 'is',
  300. 'it', 'ja', 'km', 'ko', 'lt', 'lv', 'mk', 'ms', 'nb', 'nl', 'pl',
  301. 'pt-BR', 'pt', 'ro', 'ru', 'sk', 'sr-Cyrl', 'sr', 'sv', 'th',
  302. 'tr', 'uk', 'vi',
  303. ]}
  304. SELECT2_TRANSLATIONS.update({'zh-hans': 'zh-CN', 'zh-hant': 'zh-TW'})
  305. class AutocompleteMixin:
  306. """
  307. Select widget mixin that loads options from AutocompleteJsonView via AJAX.
  308. Renders the necessary data attributes for select2 and adds the static form
  309. media.
  310. """
  311. url_name = '%s:autocomplete'
  312. def __init__(self, field, admin_site, attrs=None, choices=(), using=None):
  313. self.field = field
  314. self.admin_site = admin_site
  315. self.db = using
  316. self.choices = choices
  317. self.attrs = {} if attrs is None else attrs.copy()
  318. def get_url(self):
  319. return reverse(self.url_name % self.admin_site.name)
  320. def build_attrs(self, base_attrs, extra_attrs=None):
  321. """
  322. Set select2's AJAX attributes.
  323. Attributes can be set using the html5 data attribute.
  324. Nested attributes require a double dash as per
  325. https://select2.org/configuration/data-attributes#nested-subkey-options
  326. """
  327. attrs = super().build_attrs(base_attrs, extra_attrs=extra_attrs)
  328. attrs.setdefault('class', '')
  329. attrs.update({
  330. 'data-ajax--cache': 'true',
  331. 'data-ajax--delay': 250,
  332. 'data-ajax--type': 'GET',
  333. 'data-ajax--url': self.get_url(),
  334. 'data-app-label': self.field.model._meta.app_label,
  335. 'data-model-name': self.field.model._meta.model_name,
  336. 'data-field-name': self.field.name,
  337. 'data-theme': 'admin-autocomplete',
  338. 'data-allow-clear': json.dumps(not self.is_required),
  339. 'data-placeholder': '', # Allows clearing of the input.
  340. 'class': attrs['class'] + (' ' if attrs['class'] else '') + 'admin-autocomplete',
  341. })
  342. return attrs
  343. def optgroups(self, name, value, attr=None):
  344. """Return selected options based on the ModelChoiceIterator."""
  345. default = (None, [], 0)
  346. groups = [default]
  347. has_selected = False
  348. selected_choices = {
  349. str(v) for v in value
  350. if str(v) not in self.choices.field.empty_values
  351. }
  352. if not self.is_required and not self.allow_multiple_selected:
  353. default[1].append(self.create_option(name, '', '', False, 0))
  354. remote_model_opts = self.field.remote_field.model._meta
  355. to_field_name = getattr(self.field.remote_field, 'field_name', remote_model_opts.pk.attname)
  356. to_field_name = remote_model_opts.get_field(to_field_name).attname
  357. choices = (
  358. (getattr(obj, to_field_name), self.choices.field.label_from_instance(obj))
  359. for obj in self.choices.queryset.using(self.db).filter(**{'%s__in' % to_field_name: selected_choices})
  360. )
  361. for option_value, option_label in choices:
  362. selected = (
  363. str(option_value) in value and
  364. (has_selected is False or self.allow_multiple_selected)
  365. )
  366. has_selected |= selected
  367. index = len(default[1])
  368. subgroup = default[1]
  369. subgroup.append(self.create_option(name, option_value, option_label, selected_choices, index))
  370. return groups
  371. @property
  372. def media(self):
  373. extra = '' if settings.DEBUG else '.min'
  374. i18n_name = SELECT2_TRANSLATIONS.get(get_language())
  375. i18n_file = ('admin/js/vendor/select2/i18n/%s.js' % i18n_name,) if i18n_name else ()
  376. return forms.Media(
  377. js=(
  378. 'admin/js/vendor/jquery/jquery%s.js' % extra,
  379. 'admin/js/vendor/select2/select2.full%s.js' % extra,
  380. ) + i18n_file + (
  381. 'admin/js/jquery.init.js',
  382. 'admin/js/autocomplete.js',
  383. ),
  384. css={
  385. 'screen': (
  386. 'admin/css/vendor/select2/select2%s.css' % extra,
  387. 'admin/css/autocomplete.css',
  388. ),
  389. },
  390. )
  391. class AutocompleteSelect(AutocompleteMixin, forms.Select):
  392. pass
  393. class AutocompleteSelectMultiple(AutocompleteMixin, forms.SelectMultiple):
  394. pass