更改 Django autocomplete_fields 标签

Dor*_*ite 6 python django django-2.1

我正在尝试为autocomplete_fields.

到目前为止,对于下拉列表,人们会使用

...
class CustomDisplay(forms.ModelChoiceField):
    def label_from_instance(self, obj):
        return "Some custom text: {}".format(obj.name)
...
somethings = CustomDisplay(queryset=Something.object.all())
...
Run Code Online (Sandbox Code Playgroud)

但是使用它autocomplete_fields = (somethings,)会导致自动完成取消并向我显示带有自定义文本的下拉列表。

als*_*str 9

该字段显示普通select小部件的原因是,当您定义自定义字段时,您没有将小部件设置为AutocompleteSelect.

ModelAdmin您指定您的类中autocomplete_fields,导入您的CustomDisplayAutocompleteSelect并添加以下方法:

from django.contrib.admin.widgets import AutocompleteSelect


class YourModelAdmin(admin.ModelAdmin):
    autocomplete_fields = ['something']

    ...

    def formfield_for_foreignkey(self, db_field, request, **kwargs):
        db = kwargs.get('using')

        if db_field.name == 'something':
            return CustomDisplay(queryset=Something.object.all(), widget=AutocompleteSelect(db_field.remote_field, self.admin_site, using=db))
        return super().formfield_for_foreignkey(db_field, request, **kwargs)
Run Code Online (Sandbox Code Playgroud)

这只会在您查看现有实例时显示自定义文本。当您查看自动完成下拉菜单并选择一个条目时,标签不是从 生成的label_from_instance(),而是从str()内部的直接调用生成的AutocompleteJsonView

因此,假设您只想更改自动完成小部件中的标签(要全面更改标签,您显然只需更改模型__str()__方法),您还需要创建一个自定义类admin.py来修改 中的get()方法AutocompleteJsonView

from django.contrib.admin.options import AutocompleteJsonView
from django.http import Http404, JsonResponse

class CustomAutocompleteJsonView(AutocompleteJsonView):
    def get(self, request, *args, **kwargs):
        if not self.model_admin.get_search_fields(request):
            raise Http404(
                '%s must have search_fields for the autocomplete_view.' %
                type(self.model_admin).__name__
            )
        if not self.has_perm(request):
            return JsonResponse({'error': '403 Forbidden'}, status=403)

        self.term = request.GET.get('term', '')
        self.paginator_class = self.model_admin.paginator
        self.object_list = self.get_queryset()
        context = self.get_context_data()

        # Replace this with the code below.
        #
        # return JsonResponse({
        #     'results': [
        #         {'id': str(obj.pk), 'text': str(obj)}
        #         for obj in context['object_list']
        #     ],
        #     'pagination': {'more': context['page_obj'].has_next()},
        # })

        return JsonResponse({
            'results': [
                {'id': str(obj.pk), 'text': 'Some custom text: {}'.format(obj.name)}
                for obj in context['object_list']
            ],
            'pagination': {'more': context['page_obj'].has_next()},
    })
Run Code Online (Sandbox Code Playgroud)

现在设置autocomplete_viewModelAdmin是自动完成其中显示了(而不是结果类ModelAdmin,您可以指定autocomplete_fields类):

def autocomplete_view(self, request):
    return CustomAutocompleteJsonView.as_view(model_admin=self)(request)
Run Code Online (Sandbox Code Playgroud)

因此,如果您有一个ModelAdmin名为YourModelAdminwith的类autocomplete_fields = ['something'],您将为您的模型设置autocomplete_view相应的ModelAdminSomething

  • 不幸的是,Django 3.2 将“autocomplete_view”从“ModelAdmin”中移走了,我还没有弄清楚如何适应上面的(看起来很棒!)解决方案。如果我破解了它,我会在这里发布,但想警告其他读者,这个解决方案可能不再按预期工作。 (5认同)
  • @rossettistone 自我最初的回答以来,情况发生了一些变化。我提供了基于 Django 4.2 的更新。 (2认同)