{
  "Selected_candidate": {
    "pr_number": 8362,
    "pr_title": "Fixed #28058 -- Restored empty BoundFields evaluating to True.",
    "pr_body": "https://code.djangoproject.com/ticket/28058",
    "issue_id": 28058,
    "issue_title": "Empty Select widget (no choices) evaluates to False",
    "issue_body": "Hey guys,\nI'm not sure I catched a bug, but I discovered a change between\nDjango==1.10.6\nand 1.11.\nWhen you initialize a form with a\nSelect\nwidget, and the select widget has no choices (basically an empty tuple), when evaluating the field with\nbool()\n, it will evaluate to\nFalse\n. This causes the\ndjango_widget_tweaks\nmodule not to render a widget when trying to modify it from within a template, as it uses a bool-like evaluation to check if anything is passed to the set_attr.\ncode to reproduce:\nfrom django import forms\n\nclass MyForm(forms.Form):\n    select = forms.ChoiceField(choices=())\n\nx = MyForm()\n\nfor item in x:\n    print(bool(item))\nIt will print\nTrue\nwith\nDjango==1.10.6\n, and\nFalse\nwith\nDjango==1.11\n.\nThe underlying problem is that\nlen()\nfor a Boundfield with no choices will return 0 in the newer version, whereas it returns 1 in the older version.\nthe template code that fails to render with\nDjango==1.11\n{{ widget|set_attr('style:width: 100%') }}\nLike I said, I'm not sure if this change is intended, nevertheless I reported it. Please let me know if this is intended, so I can let the creator of\ndjango_widget_tweaks\nknow.\nCheers",
    "issue_closed_at": "2017-04-17T07:49:12",
    "base_commit": "e5dce7b0fbd2965e524ce97114f501319ec2bb6f",
    "changes": [
      {
        "file": "django/forms/boundfield.py",
        "type": "function",
        "name": "subwidgets",
        "class_name": "BoundField",
        "code": "def subwidgets(self):\n        \"\"\"\n        Most widgets yield a single subwidget, but others like RadioSelect and\n        CheckboxSelectMultiple produce one subwidget for each choice.\n\n        This property is cached so that only one database query occurs when\n        rendering ModelChoiceFields.\n        \"\"\"\n        id_ = self.field.widget.attrs.get('id') or self.auto_id\n        attrs = {'id': id_} if id_ else {}\n        attrs = self.build_widget_attrs(attrs)\n        return list(\n            BoundWidget(self.field.widget, widget, self.form.renderer)\n            for widget in self.field.widget.subwidgets(self.html_name, self.value(), attrs=attrs)\n        )"
      }
    ]
  },
  "Justification": "Candidate E is the most relevant as it deals with issues in Django forms, specifically involving the Select widget, which is conceptually linked to the autocomplete functionality discussed in the CURRENT bug report. It shares similar file structure and module focus, particularly in the 'django/forms/boundfield.py' file, which indicates it may help understand issues related to field rendering and attribute handling in forms. The changes and patch solutions discussed in Candidate E may provide insights and alternatives to consider for enhancing the autocomplete response handling, especially regarding how data attributes are rendered, which is a similar context to what is needed in the CURRENT bug report.",
  "instance_id": "django__django-14752",
  "repo": "django/django",
  "created_at": "2021-08-07T16:34:32Z",
  "problem_statement": "Refactor AutocompleteJsonView to support extra fields in autocomplete response\nDescription\n\t \n\t\t(last modified by mrts)\n\t \nAdding data attributes to items in ordinary non-autocomplete foreign key fields that use forms.widgets.Select-based widgets is relatively easy. This enables powerful and dynamic admin site customizations where fields from related models are updated immediately when users change the selected item.\nHowever, adding new attributes to autocomplete field results currently requires extending contrib.admin.views.autocomplete.AutocompleteJsonView and fully overriding the AutocompleteJsonView.get() method. Here's an example:\nclass MyModelAdmin(admin.ModelAdmin):\n\tdef get_urls(self):\n\t\treturn [\n\t\t\tpath('autocomplete/', CustomAutocompleteJsonView.as_view(admin_site=self.admin_site))\n\t\t\tif url.pattern.match('autocomplete/')\n\t\t\telse url for url in super().get_urls()\n\t\t]\nclass CustomAutocompleteJsonView(AutocompleteJsonView):\n\tdef get(self, request, *args, **kwargs):\n\t\tself.term, self.model_admin, self.source_field, to_field_name = self.process_request(request)\n\t\tif not self.has_perm(request):\n\t\t\traise PermissionDenied\n\t\tself.object_list = self.get_queryset()\n\t\tcontext = self.get_context_data()\n\t\treturn JsonResponse({\n\t\t\t'results': [\n\t\t\t\t{'id': str(getattr(obj, to_field_name)), 'text': str(obj), 'notes': obj.notes} # <-- customization here\n\t\t\t\tfor obj in context['object_list']\n\t\t\t],\n\t\t\t'pagination': {'more': context['page_obj'].has_next()},\n\t\t})\nThe problem with this is that as AutocompleteJsonView.get() keeps evolving, there's quite a lot of maintenance overhead required to catch up.\nThe solutions is simple, side-effect- and risk-free: adding a result customization extension point to get() by moving the lines that construct the results inside JsonResponse constructor to a separate method. So instead of\n\t\treturn JsonResponse({\n\t\t\t'results': [\n\t\t\t\t{'id': str(getattr(obj, to_field_name)), 'text': str(obj)}\n\t\t\t\tfor obj in context['object_list']\n\t\t\t],\n\t\t\t'pagination': {'more': context['page_obj'].has_next()},\n\t\t})\nthere would be\n\t\treturn JsonResponse({\n\t\t\t'results': [\n\t\t\t\tself.serialize_result(obj, to_field_name) for obj in context['object_list']\n\t\t\t],\n\t\t\t'pagination': {'more': context['page_obj'].has_next()},\n\t\t})\nwhere serialize_result() contains the original object to dictionary conversion code that would be now easy to override:\ndef serialize_result(self, obj, to_field_name):\n\treturn {'id': str(getattr(obj, to_field_name)), 'text': str(obj)}\nThe example CustomAutocompleteJsonView from above would now become succinct and maintainable:\nclass CustomAutocompleteJsonView(AutocompleteJsonView):\n\tdef serialize_result(self, obj, to_field_name):\n\t\treturn super.serialize_result(obj, to_field_name) | {'notes': obj.notes}\nWhat do you think, is this acceptable? I'm more than happy to provide the patch.\n",
  "patch": "diff --git a/django/contrib/admin/views/autocomplete.py b/django/contrib/admin/views/autocomplete.py\n--- a/django/contrib/admin/views/autocomplete.py\n+++ b/django/contrib/admin/views/autocomplete.py\n@@ -11,7 +11,8 @@ class AutocompleteJsonView(BaseListView):\n \n     def get(self, request, *args, **kwargs):\n         \"\"\"\n-        Return a JsonResponse with search results of the form:\n+        Return a JsonResponse with search results as defined in\n+        serialize_result(), by default:\n         {\n             results: [{id: \"123\" text: \"foo\"}],\n             pagination: {more: true}\n@@ -26,12 +27,19 @@ def get(self, request, *args, **kwargs):\n         context = self.get_context_data()\n         return JsonResponse({\n             'results': [\n-                {'id': str(getattr(obj, to_field_name)), 'text': str(obj)}\n+                self.serialize_result(obj, to_field_name)\n                 for obj in context['object_list']\n             ],\n             'pagination': {'more': context['page_obj'].has_next()},\n         })\n \n+    def serialize_result(self, obj, to_field_name):\n+        \"\"\"\n+        Convert the provided model object to a dictionary that is added to the\n+        results list.\n+        \"\"\"\n+        return {'id': str(getattr(obj, to_field_name)), 'text': str(obj)}\n+\n     def get_paginator(self, *args, **kwargs):\n         \"\"\"Use the ModelAdmin's paginator.\"\"\"\n         return self.model_admin.get_paginator(self.request, *args, **kwargs)\n"
}