Каковы шаги, чтобы заставить ModelForm работать с отношением ManyToMany с промежуточной моделью в Django?

  • У меня есть Клиент и Groupe Модель.
  • Клиент может быть частью нескольких групп .
  • Клиенты , которые являются частью группы, могут использовать бесплатную арендную ставку своей группы в любое время, но только один раз. Вот где посредническая модель ( ClientGroupe ) входит с этими дополнительными данными.

На данный момент, когда я пытаюсь сохранить данные m2m, он просто умирает и говорит, что должен использовать ClientGroupe Manager ... так чего не хватает?

Вот мои модели:

class Groupe(models.Model):
    nom = models.CharField(max_length=1500, blank=True)

class Client(models.Model):
    nom = models.CharField(max_length=450, blank=True)
    prenom = models.CharField(max_length=450, blank=True)
    groupes = models.ManyToManyField(Groupe, null = True, blank = True, through='ClientGroupe')

class ClientGroupe(models.Model):
    client = models.ForeignKey(Client)
    groupe = models.ForeignKey(Groupe)
    dt = models.DateField(null=True, blank=True) # the date the client is using its group's free rental rate    

    class Meta:
        db_table = u'clients_groupes'

и вот мое мнение:

def modifier(request, id):
    client = Client.objects.get(id=id)    
    form = ClientForm(instance = client)

    dict = {
        "form": form
        , "instance" : client
    }

    if request.method == "POST":
        form = ClientForm(request.POST, instance = client)

        if form.is_valid():
            client_mod = form.save()

            id = client_mod.id
            return HttpResponseRedirect(
                "/client/%(id)s/?err=success" % {"id" : id}
            )
        else:
            return HttpResponseRedirect(
                "/client/%(id)s/?err=warning" % {"id" : id}
            )

    return render_to_response(
        "client/modifier.html"
        , dict
        , context_instance=RequestContext(request)
    )

РЕДАКТИРОВАТЬ :

и вот код ClientForm:

class ClientForm(ModelForm):
    class Meta:
        model = Client

РЕДАКТИРОВАТЬ # 2 : вот сообщение об ошибке:

AttributeError at /client/445/

Cannot set values on a ManyToManyField which specifies an intermediary model. Use ClientGroupe's Manager instead.

Request Method:     POST
Request URL:    http://localhost/client/445/
Exception Type:     AttributeError
Exception Value:    Cannot set values on a ManyToManyField which specifies an intermediary model.  Use ClientGroupe's Manager instead.

Exception Location:     C:\Python25\lib\site-packages\django\db\models\fields\related.py  in __set__, line 574
Python Executable:  C:\xampp\apache\bin\apache.exe
Python Version:     2.5.2
30
задан Fabian Steeg 25 January 2009 в 22:28
поделиться

3 ответа

Вероятно, необходимо удалить поле ManyToMany из модели Client или иначе тщательно исключить его из формы. К сожалению, виджет по умолчанию для поля ManyToMany не может заполнить Модель ClientGroupe правильно (даже если недостающее поле, dt, было установлено на autonow=True). Это - что-то, что необходимо будет или вспыхнуть в другую форму или обработать в представлении.

4
ответ дан SingleNegationElimination 25 January 2009 в 22:28
поделиться
  • 1
    @kanaka я имею в виду Вас won' t быть в состоянии соединиться с произвольными клиентами и портами по UDP – robertc 27 October 2012 в 03:49

При сохранении формы Вы сохраняете Объект клиента. Теперь, если Вы хотите присвоить клиент группе, необходимо сделать это:

clientgroupe = ClientGroupe.objects.create(client=client_instance, groupe=groupe_instance, dt=datetime.datetime.now())

, где client_instance и groupe_instance Ваш клиент и groupe objets.

0
ответ дан Vladimir Prudnikov 26 January 2009 в 08:28
поделиться
  • 1
    +1 для самого читаемого решения (читает как: " удалите весь nulls") – Pablo Fernandez 20 July 2010 в 08:40

If you use the save method right now, Django will try to save using the manager (which Django doesn't allow). Unfortunately, the behavior you want is a little bit trickier than what ModelForm does by default. What you need to do is create a formset.

First of all, you will need to change the options of your ClientForm so that it doesn't display the groupes attribute.

class ClientForm(ModelForm):
    class Meta:
        model = Client
        exclude = ('groupes',)

Next, you must change the view to display the formset:

from django.forms.models import inlineformset_factory

def modifier(request, id):
    client = Client.objects.get(id=id)    
    form = ClientForm(instance = client)
    # Create the formset class
    GroupeFormset = inlineformset_factory(Client, Groupe)
    # Create the formset
    formset = GroupeFormset(instance = client)

    dict = {
        "form": form
        , "formset" : formset
        , "instance" : client
    }

    if request.method == "POST":
        form = ClientForm(request.POST, instance = client)
        formset = GroupeFormset(request.POST, instance = client)

        if form.is_valid() and formset.is_valid():
            client_mod = form.save()
            formset.save()

            id = client_mod.id
            return HttpResponseRedirect(
                "/client/%(id)s/?err=success" % {"id" : id}
            )
        else:
            return HttpResponseRedirect(
                "/client/%(id)s/?err=warning" % {"id" : id}
            )

    return render_to_response(
        "client/modifier.html"
        , dict
        , context_instance=RequestContext(request)
    )

And obviously, you must also tweak your template to render the formset.

If you need any other advice on formsets, see these articles:

Model formsets
Formsets

18
ответ дан 28 November 2019 в 00:20
поделиться
Другие вопросы по тегам:

Похожие вопросы: