Je suppose que vous avez ici pour ajouter un nouveau ModelMultipleChoiceField
votre PizzaForm
, et manuellement lien de champ de formulaire avec le champ de modèle, que Django n'allons pas le faire automatiquement pour vous.
L'extrait de code suivant peut être utile :
class PizzaForm(forms.ModelForm):
class Meta:
model = Pizza
# Representing the many to many related field in Pizza
toppings = forms.ModelMultipleChoiceField(queryset=Topping.objects.all())
# Overriding __init__ here allows us to provide initial
# data for 'toppings' field
def __init__(self, *args, **kwargs):
# Only in case we build the form from an instance
# (otherwise, 'toppings' list should be empty)
if 'instance' in kwargs:
# We get the 'initial' keyword argument or initialize it
# as a dict if it didn't exist.
initial = kwargs.setdefault('initial', {})
# The widget for a ModelMultipleChoiceField expects
# a list of primary key for the selected data.
initial['toppings'] = [t.pk for t in kwargs['instance'].topping_set.all()]
forms.ModelForm.__init__(self, *args, **kwargs)
# Overriding save allows us to process the value of 'toppings' field
def save(self, commit=True):
# Get the unsave Pizza instance
instance = forms.ModelForm.save(self, False)
# Prepare a 'save_m2m' method for the form,
old_save_m2m = self.save_m2m
def save_m2m():
old_save_m2m()
# This is where we actually link the pizza with toppings
instance.topping_set.clear()
for topping in self.cleaned_data['toppings']:
instance.topping_set.add(topping)
self.save_m2m = save_m2m
# Do we need to save all changes now?
if commit:
instance.save()
self.save_m2m()
return instance
Cette PizzaForm
peut ensuite être utilisé partout, même dans l'admin :
# yourapp/admin.py
from django.contrib.admin import site, ModelAdmin
from yourapp.models import Pizza
from yourapp.forms import PizzaForm
class PizzaAdmin(ModelAdmin):
form = PizzaForm
site.register(Pizza, PizzaAdmin)
Note
L' save()
méthode peut-être un peu trop verbeux, mais vous pouvez simplifier si vous n'avez pas besoin de soutenir l' commit=False
situation, il sera alors comme ça :
def save(self):
instance = forms.ModelForm.save(self)
instance.topping_set.clear()
for topping in self.cleaned_data['toppings']:
instance.topping_set.add(topping)