Фильтрация встроенного администратора django — ограничьте список вариантов

#python-3.x #django #django-admin

Вопрос:

Учитывая моего встроенного администратора:

 class TestInlineAdmin(admin.TabularInline):
    model = Test.questions.through
    extra = 0  
 

и затем

 class QuestionAdmin(admin.ModelAdmin):
    inlines = [TestInlineAdmin, ]
 

Тестовая модель имеет поле вопросов, которое представляет собой множество. И обычно я могу редактировать список вопросов непосредственно из тестовой модели.
Но я хочу иметь возможность выбирать тесты из встроенного администратора в QuestionAdmin (так, наоборот).
Это работает. Но мне нужно отфильтровать тестовые объекты в этой строке, чтобы список вариантов показывал мне только Test.objects.filter(применимо=False).

встроенный элемент администратора

Я пытался использовать get_queryset, но это, похоже, не влияет на список выбора, он только фильтрует фактические элементы, на которые есть ссылки, в строке, но список вариантов для нового элемента всегда показывает мне полный нефильтрованный набор запросов для тестовой модели.

Переопределение formfield_for_manytomany не работает во встроенном режиме — оно вообще не выполняется. Возможно ли это каким-то образом с formfield_overrides? Или, как я полагаю, единственным способом было бы настроить встроенную форму?

-Редактировать-

Моя модель, в которой определено множество:

 class Test(models.Model):
    title = models.CharField(max_length=80)
    description = models.TextField(null=True)
    position = models.ForeignKey('repository.Position', on_delete=models.CASCADE, null=True)
    questions = models.ManyToManyField('questions.Question')
    applicable = models.BooleanField(default=False)
 

— Отредактировано —
обратная связь после предложения Виллема

 2021-09-03 07:50:37 | ERROR | /usr/local/lib64/python3.6/site-packages/django/utils/log.py | log_response | - Internal Server Error: /admin/questions/question/45/change/
Traceback (most recent call last):
  File "/usr/local/lib64/python3.6/site-packages/django/core/handlers/exception.py", line 47, in inner
    response = get_response(request)
  File "/usr/local/lib64/python3.6/site-packages/django/core/handlers/base.py", line 181, in _get_response
    response = wrapped_callback(request, *callback_args, **callback_kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 614, in wrapper
    return self.admin_site.admin_view(view)(*args, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/utils/decorators.py", line 130, in _wrapped_view
    response = view_func(request, *args, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/views/decorators/cache.py", line 44, in _wrapped_view_func
    response = view_func(request, *args, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/sites.py", line 233, in inner
    return view(request, *args, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 1656, in change_view
    return self.changeform_view(request, object_id, form_url, extra_context)
  File "/usr/local/lib64/python3.6/site-packages/django/utils/decorators.py", line 43, in _wrapper
    return bound_method(*args, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/utils/decorators.py", line 130, in _wrapped_view
    response = view_func(request, *args, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 1534, in changeform_view
    return self._changeform_view(request, object_id, form_url, extra_context)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 1598, in _changeform_view
    formsets, inline_instances = self._create_formsets(request, obj, change=True)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 1954, in _create_formsets
    for FormSet, inline in self.get_formsets_with_inlines(*get_formsets_args):
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 798, in get_formsets_with_inlines
    yield inline.get_formset(request, obj), inline
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 2053, in get_formset
    fields = flatten_fieldsets(self.get_fieldsets(request, obj))
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 335, in get_fieldsets
    return [(None, {'fields': self.get_fields(request, obj)})]
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 326, in get_fields
    form = self._get_form_for_get_fields(request, obj)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 2138, in _get_form_for_get_fields
    return self.get_formset(request, obj, fields=None).form
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 2135, in get_formset
    return inlineformset_factory(self.parent_model, self.model, **defaults)
  File "/usr/local/lib64/python3.6/site-packages/django/forms/models.py", line 1082, in inlineformset_factory
    FormSet = modelformset_factory(model, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/forms/models.py", line 879, in modelformset_factory
    error_messages=error_messages, field_classes=field_classes)
  File "/usr/local/lib64/python3.6/site-packages/django/forms/models.py", line 555, in modelform_factory
    return type(form)(class_name, (form,), form_class_attrs)
  File "/usr/local/lib64/python3.6/site-packages/django/forms/models.py", line 258, in __new__
    apply_limit_choices_to=False,
  File "/usr/local/lib64/python3.6/site-packages/django/forms/models.py", line 179, in fields_for_model
    formfield = formfield_callback(f, **kwargs)
  File "/usr/local/lib64/python3.6/site-packages/django/contrib/admin/options.py", line 172, in formfield_for_dbfield
    formfield.widget, db_field.remote_field, self.admin_site, **wrapper_kwargs
AttributeError: 'dict' object has no attribute 'widget'
 

Комментарии:

1. Можете ли вы поделиться моделью, в которой вы определили отношение m2m? Вместе с тем, с чем вы пробовали formfield_for_manytomany ?

2. Я обновил исходный пост, касающийся formfield_for_manytomany, — я не сделал ничего особенного, просто добавил его в встроенную модель администратора и попытался отладить ее, чтобы выяснить, то ли это то, что я ищу. Вы имеете в виду, что это должно быть выполнимо с переопределением этого метода?

Ответ №1:

Вы можете ограничить QuerySet количество TestInlineAdmin с:

 class TestInlineAdmin(admin.TabularInline):
    model = Test.questions.through
    extra = 0
    
    def formfield_for_foreignkey(self, db_field, request, **kwargs):
        if db_field.name == 'test':
            kwargs['queryset'] = Test.objects.filter(applicable=False)
        return super().formfield_for_foreignkey(db_field, request, **kwargs) 

Комментарии:

1. Я пытаюсь отладить это в настоящее время, к сожалению, это вызывает исключение: ibb.co/0Jyk8z9

2. @TomaszSzkudlarek: можете ли вы отредактировать вопрос с полной обратной связью в виде кода , а не изображений кода .

3. Конечно, вот ты где

4. @TomaszSzkudlarek: Я нашел ошибку, это return super().formfield_for_foreignkey(db_field, request, **kwargs) не return kwargs так .

5. @TomaszSzkudlarek: Я слышал хорошие вещи о двух порциях Django-3-x от некоторых коллег.