Проблема взаимоотношений Django M2M с администратором при использовании промежуточной таблицы с множественным выбором

Я работаю с отношениями m2m, используя, чтобы установить промежуточную таблицу. Проблема в том, что мне нужно показывать множественный выбор вместо обычного спада, но когда я выбираю несколько элементов и сохраняю их, я получаю сообщение об ошибке.

ValueError: Cannot assign "<QuerySet [<Facility: facility1>, <Facility: facility2>]>": "Property.facility" must be a "Facility" instance.

Также я показываю эту модель в admin.TabularInline, которая позволяет мне выбирать только один элемент в строке, так как табличный inline дает возможность вставлять несколько форм.

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

models.py


class Facility(models.Model):
    name = models.CharField(max_length=200)

class Property(models.Model):
    name = models.CharField(max_length=200)
    area = models.CharField(max_length=200)
    facility = models.ManyToManyField(Facility, through="PropertyFacility")

class PropertyFacility(models.Model):
    prop = models.ForeignKey(
        Property, related_name="facilities", on_delete=models.CASCADE
    )
    facility = models.ForeignKey(
        Facility, related_name="properties", on_delete=models.CASCADE
    )

admin.py


from django.contrib.admin.widgets import FilteredSelectMultiple
from django.utils.translation import ugettext_lazy as _

class PropertyFacilityForm(forms.ModelForm):
    facility = forms.ModelMultipleChoiceField(Facility.objects.all(), required=True, widget=FilteredSelectMultiple(_('ss'), False, attrs={'rows':'10'})

class PropertyFacilityInline(admin.TabularInline):
    model = Property.facility.through
    form = PropertyFacilityForm

class PropertyAdmin(TabbedModelAdmin):
    model = Property
    tab_facilities = (PropertyFacilityInline,)
    tab_property = (
        (
            "Property Details",
            {
                "fields": (
                    "name",
                    "area",
                )
            },
        ),
    )

    tabs = [
        ("Property", tab_property),
        ("Facilities", tab_facilities),
    ]

Это позволяет мне показывать в админке вот так. Здесь вы можете видеть, что я могу выбрать несколько вариантов выбора, но здесь должна быть одна форма, а не несколько.

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

1 ответ

Решение

Я исправил проблему, добавив некоторую пользовательскую логику для сохранения поля m2m.

admin.py

class PropertyForm(forms.ModelForm):
    facility = forms.ModelMultipleChoiceField(Facility.objects.all(), required=False, widget=forms.CheckboxSelectMultiple)
    class Meta:
        model = Property
        fields = ["id", "name", "area", "city"]

    def save(self, commit=True):
        prop_facilities = self.cleaned_data.pop('facility')
        instance = forms.ModelForm.save(self, commit=False)
        instance.facility.clear()
        for facility in prop_facilities:
            PropertyFacility.objects.create(facility=facility, prop=instance)
        return instance


class PropertyAdmin(TabbedModelAdmin):
    model = Property
    tab_facilities = (("Facilities", {"fields": ("facility", )},),)
    tab_property = (
        (
            "Property Details",
            {
                "fields": (
                    "name",
                    "area",
                )
            },
        ),
    )

    tabs = [
        ("Property", tab_property),
        ("Facilities", tab_facilities),
    ]
Другие вопросы по тегам