[4.0.x] Fixed #33111 -- Fixed passing object to ModelAdmin.get_inlines() when editing in admin change view.
ModelAdmin.get_inlines() should get an unmutated object when creating
formsets during POST request.
Backport of 2f0f30f973
from main
This commit is contained in:
parent
881b6bd135
commit
668b990bf6
|
@ -1585,12 +1585,16 @@ class ModelAdmin(BaseModelAdmin):
|
|||
)
|
||||
if request.method == 'POST':
|
||||
form = ModelForm(request.POST, request.FILES, instance=obj)
|
||||
formsets, inline_instances = self._create_formsets(
|
||||
request,
|
||||
form.instance if add else obj,
|
||||
change=not add,
|
||||
)
|
||||
form_validated = form.is_valid()
|
||||
if form_validated:
|
||||
new_object = self.save_form(request, form, change=not add)
|
||||
else:
|
||||
new_object = form.instance
|
||||
formsets, inline_instances = self._create_formsets(request, new_object, change=not add)
|
||||
if all_valid(formsets) and form_validated:
|
||||
self.save_model(request, new_object, form, not add)
|
||||
self.save_related(request, form, formsets, not add)
|
||||
|
|
|
@ -11,8 +11,9 @@ from .models import (
|
|||
Inner4Tabular, Inner5Stacked, Inner5Tabular, NonAutoPKBook,
|
||||
NonAutoPKBookChild, Novel, NovelReadonlyChapter, OutfitItem,
|
||||
ParentModelWithCustomPk, Person, Poll, Profile, ProfileCollection,
|
||||
Question, ReadOnlyInline, ShoppingWeakness, Sighting, SomeChildModel,
|
||||
SomeParentModel, SottoCapo, Teacher, Title, TitleCollection,
|
||||
Question, ReadOnlyInline, ShoppingWeakness, ShowInlineChild,
|
||||
ShowInlineParent, Sighting, SomeChildModel, SomeParentModel, SottoCapo,
|
||||
Teacher, Title, TitleCollection,
|
||||
)
|
||||
|
||||
site = admin.AdminSite(name="admin")
|
||||
|
@ -371,6 +372,17 @@ class ChildHiddenFieldOnSingleLineStackedInline(admin.StackedInline):
|
|||
fields = ('name', 'position')
|
||||
|
||||
|
||||
class ShowInlineChildInline(admin.StackedInline):
|
||||
model = ShowInlineChild
|
||||
|
||||
|
||||
class ShowInlineParentAdmin(admin.ModelAdmin):
|
||||
def get_inlines(self, request, obj):
|
||||
if obj is not None and obj.show_inlines:
|
||||
return [ShowInlineChildInline]
|
||||
return []
|
||||
|
||||
|
||||
site.register(TitleCollection, inlines=[TitleInline])
|
||||
# Test bug #12561 and #12778
|
||||
# only ModelAdmin media
|
||||
|
@ -402,6 +414,7 @@ site.register(Course, ClassAdminStackedHorizontal)
|
|||
site.register(CourseProxy, ClassAdminStackedVertical)
|
||||
site.register(CourseProxy1, ClassAdminTabularVertical)
|
||||
site.register(CourseProxy2, ClassAdminTabularHorizontal)
|
||||
site.register(ShowInlineParent, ShowInlineParentAdmin)
|
||||
# Used to test hidden fields in tabular and stacked inlines.
|
||||
site2 = admin.AdminSite(name='tabular_inline_hidden_field_admin')
|
||||
site2.register(SomeParentModel, inlines=[ChildHiddenFieldTabularInline])
|
||||
|
|
|
@ -327,6 +327,12 @@ class CourseProxy2(Course):
|
|||
|
||||
|
||||
# Other models
|
||||
class ShowInlineParent(models.Model):
|
||||
show_inlines = models.BooleanField(default=False)
|
||||
|
||||
|
||||
class ShowInlineChild(models.Model):
|
||||
parent = models.ForeignKey(ShowInlineParent, on_delete=models.CASCADE)
|
||||
|
||||
|
||||
class ProfileCollection(models.Model):
|
||||
|
|
|
@ -12,8 +12,8 @@ from .models import (
|
|||
ChildModel1, ChildModel2, Fashionista, FootNote, Holder, Holder2, Holder3,
|
||||
Holder4, Inner, Inner2, Inner3, Inner4Stacked, Inner4Tabular, Novel,
|
||||
OutfitItem, Parent, ParentModelWithCustomPk, Person, Poll, Profile,
|
||||
ProfileCollection, Question, Sighting, SomeChildModel, SomeParentModel,
|
||||
Teacher, VerboseNamePluralProfile, VerboseNameProfile,
|
||||
ProfileCollection, Question, ShowInlineParent, Sighting, SomeChildModel,
|
||||
SomeParentModel, Teacher, VerboseNamePluralProfile, VerboseNameProfile,
|
||||
)
|
||||
|
||||
INLINE_CHANGELINK_HTML = 'class="inlinechangelink">Change</a>'
|
||||
|
@ -618,6 +618,21 @@ class TestInline(TestDataMixin, TestCase):
|
|||
self.assertContains(response, '<h2>Author</h2>', html=True) # Tabular.
|
||||
self.assertContains(response, '<h2>Fashionista</h2>', html=True) # Stacked.
|
||||
|
||||
def test_inlines_based_on_model_state(self):
|
||||
parent = ShowInlineParent.objects.create(show_inlines=False)
|
||||
data = {
|
||||
'show_inlines': 'on',
|
||||
'_save': 'Save',
|
||||
}
|
||||
change_url = reverse(
|
||||
'admin:admin_inlines_showinlineparent_change',
|
||||
args=(parent.id,),
|
||||
)
|
||||
response = self.client.post(change_url, data)
|
||||
self.assertEqual(response.status_code, 302)
|
||||
parent.refresh_from_db()
|
||||
self.assertIs(parent.show_inlines, True)
|
||||
|
||||
|
||||
@override_settings(ROOT_URLCONF='admin_inlines.urls')
|
||||
class TestInlineMedia(TestDataMixin, TestCase):
|
||||
|
|
Loading…
Reference in New Issue