Администратор Django: список «многие ко многим» не отображается со сквозным параметром - PullRequest
13 голосов
/ 18 мая 2010

У меня есть следующие модели:

class Message(models.Model):
    date = models.DateTimeField()
    user = models.ForeignKey(User)    
    thread = models.ForeignKey('self', blank=True, null=True)
    ...

class Forum(models.Model):
    name = models.CharField(max_length=24)
    messages = models.ManyToManyField(Message, through="Message_forum", blank=True, null=True)
    ...

class Message_forum(models.Model):
    message = models.ForeignKey(Message)
    forum = models.ForeignKey(Forum)
    status = models.IntegerField()
    position = models.IntegerField(blank=True, null=True)
    tags = models.ManyToManyField(Tag, blank=True, null=True)

На сайте администратора, когда я иду, чтобы добавить / изменить форум, я не вижу список сообщений, как вы ожидаете. Тем не менее, он обнаруживается, если я удаляю параметр through в объявлении ManyToManyField. Что с этим? Я зарегистрировал все три модели (плюс тег) на сайте администратора в admin.py.

ТИА

Ответы [ 3 ]

18 голосов
/ 06 января 2012

Документация гласит:

Когда вы указываете посредническую модель, используя аргумент through для ManyToManyField, администратор не будет отображать виджет по умолчанию.

Но, вероятно, возможно отобразить поля M2M в представлении изменений администратора, даже если определен атрибут through.

class ForumAdminForm(forms.ModelForm):
    mm = forms.ModelMultipleChoiceField(
        queryset=models.Message.objects.all(),
        widget=FilteredSelectMultiple(_('ss'), False, attrs={'rows':'10'}))

    def __init__(self, *args, **kwargs):
        if 'instance' in kwargs:
            initial = kwargs.setdefault('initial', {})
            initial['mm'] = [t.service.pk for t in kwargs['instance'].message_forum_set.all()]

        forms.ModelForm.__init__(self, *args, **kwargs)

    def save(self, commit=True):
        instance = forms.ModelForm.save(self, commit)

        old_save_m2m = self.save_m2m
        def save_m2m():
            old_save_m2m()

            messages = [s for s in self.cleaned_data['ss']]
            for mf in instance.message_forum_set.all():
                if mf.service not in messages:
                    mf.delete()
                else:
                    messages.remove(mf.service)

            for message in messages:
                Message_forum.objects.create(message=message, forum=instance)

        self.save_m2m = save_m2m

        return instance

    class Meta:
        model = models.Forum

class ForumAdmin(admin.ModelAdmin):
    form = ForumAdminForm
10 голосов
/ 22 октября 2010
3 голосов
/ 08 марта 2012

Я многому научился из ответа @ Федора, но некоторые комментарии и исправления могут быть полезны.

class ForumAdminForm(forms.ModelForm):
    messages = forms.ModelMultipleChoiceField(
                   queryset=Message.objects.all(),
                   widget=FilteredSelectMultiple('Message', False))


    # Technically, you don't need to manually set initial here for ForumAdminForm
    # However, you NEED to do the following for MessageAdminForm
    def __init__(self, *args, **kwargs):
        if 'instance' in kwargs:
            # a record is being changed. building initial
            initial = kwargs.setdefault('initial', {})
            initial['messages'] = [t.message.pk for t in kwargs['instance'].message_forum_set.all()]
        super(ForumAdminForm, self).__init__(*args, **kwargs)

    def save(self, commit=True):
        if not self.is_valid():
            raise HttpResponseForbidden
        instance = super(ForumAdminForm, self).save(self, commit)
        def save_m2m_with_through():
            messages = [t for t in self.cleaned_data['messages']
            old_memberships = instance.message_forum_set.all()
            for old in old_memberships:
                if old.message not in messages:
                    # and old membership is cleaned by the user
                    old.delete()
            for message in [x for x in messages not in map(lambda x: x.message, old_memberships)]:                   
                membership = Member_forum(message=messsage, forum=instance) 
                # You may have to initialize status, position and tag for your need
                membership.save()
        if commit:
            save_m2m_with_through()
        else:
            self.save_m2m = save_m2m_with_through
        return instance

    class Meta:
        model = Forum
        fields = {'name', 'messages')

Есть одна оговорка: если у вас есть другое отношение «многие ко многим» в моделях (то есть без сквозного), super(ForumAdminForm, self).save(self, commit) установит self.save_m2m в случае, если commit равно False. Однако вызов этого может вызвать ошибку, потому что эта функция также пытается сохранить многие-ко-многим при помощи сквозного. Возможно, вам придется вручную сохранить все остальные отношения «многие ко многим», или перехватить исключение, или еще.

...