Django InlineModelAdmin - set inline field from request on save (set user field automatically) (save_formset vs save_model)
Solution 1
I have solved the first half of my question:
def save_formset(self, request, form, formset, change):
if formset.model != InlineModel:
return super(MainModelAdmin, self).save_formset(request, form, formset, change)
instances = formset.save(commit=False)
for instance in instances:
if not instance.pk:
instance.user = request.user
instance.save()
formset.save_m2m()
Now i'm interested in the bonus behavior:
I'm required to select a user when adding a new inline due to validation rules. My best guess is to not include the 'user' field in my InlineModelInline.fields tuple, but then this won't show the author for existing InlineModel instances. (Edit: adding 'user' to readonly_fields works here)
(Edit) How can I make the existing inlines render 'data' as readonly, but still be able to edit it when adding a new inline?
Solution 2
It worked for me. This approach won't allow me to delete Inline items.
def save_formset(self, request, form, formset, change):
for form in formset.forms:
form.instance.user = request.user
formset.save()
EB.
Updated on June 03, 2022Comments
-
EB. about 2 years
I have two models, a MainModel and a related InlineModel that i'd like to show as an inline in the admin. This InlineModel can be used for, say, making notes about the model and should track the logged in admin user making changes. While this seems simple (and indeed, the docs show an example for this when the user field is part of the MainModel), I can't seem to grasp it when the field is on the Inline.
To be specific, my goal is:
- User edits MainModel
- User adds an InlineModel, not filling in the user field
- User presses save
- Code fills in the user field for newly created InlineModel instances
- (Bonus! user field is readonly for existing instances and hidden for new inlines)
And my questions:
- Is this correct? Its too bas save_model isn't called for InlineModelAdmin instances
- Does doing it this way allow me to save without causing an error? (user is required, validation flags it)
- How can I hide the user input field for new inlines, and have it readonly for existing inlines?
Here are my current ideas:
#models.py class MainModel(models.Model): some_info = models.IntegerField() class InlineModel(models.Model): main = models.ForeignKey(MainModel) data = models.CharField(max_length=255) user = models.ForeignKey('auth.User') #admin.py class InlineModelInline(admin.TabularInline): model = InlineModel fields = ('data', 'user') #readonly_fields = ('data', 'user') #Bonus question later class MainModelAdmin(admin.ModelAdmin): list_display = ('id', 'some_info') inlines = [InlineModelInline] #def save_model(self, request, obj, form, change): #http://docs.djangoproject.com/en/dev/ref/contrib/admin/#django.contrib.admin.ModelAdmin.save_model #Only called for MainModel, not for any of the inlines #Otherwise, would be ideal def save_formset(self, request, form, formset, change): #http://docs.djangoproject.com/en/dev/ref/contrib/admin/#django.contrib.admin.ModelAdmin.save_formset #Experimenting showd this is called once per formset (where the formset is a group of inlines) #See code block at http://code.djangoproject.com/browser/django/tags/releases/1.2.1/django/contrib/admin/options.py#L894 if not isinstance(formset.model, InlineModel): return super(MainModelAdmin, self).save_formset(request, form, formset, change) instances = formset.save(commit=False) for instance in instances: if not instance.pk: instance.user = request.user instance.save() formset.save_m2m()
-
user2471801 almost 13 yearsA note to myself, this save_formset is a method of the admin.ModelAdmin, and it will process all child Inlines assigned in the ModelAdmin.
-
sdailey over 10 yearsThanks for your question/answers here. As far as the 'bonus' functionality -- after marking the user foreignKey in the django model with editable=False, the inline formset passed validation, the user field was still set and all was kosher. user = models.ForeignKey(User, editable=False)
-
EB. over 10 yearsHm, totally forgot about editable=False in the model. Nice catch, thanks.
-
Amichai Schreiber over 7 yearsThis breaks deleting inline objects.