mirror of
https://github.com/django/django.git
synced 2025-01-03 06:55:47 +00:00
Refs #35189 -- Improved admin fieldset's accessibility by setting aria-labelledby.
Before this change, HTML <fieldset> elements in the admin site did not have an associated label to describe them. This commit defines a unique HTML id for the heading labeling a fieldset, and sets its aria-labelledby property to link the heading with the fieldset.
This commit is contained in:
parent
9c5fe93349
commit
01ed59f753
@ -47,7 +47,7 @@
|
|||||||
|
|
||||||
{% block field_sets %}
|
{% block field_sets %}
|
||||||
{% for fieldset in adminform %}
|
{% for fieldset in adminform %}
|
||||||
{% include "admin/includes/fieldset.html" %}
|
{% include "admin/includes/fieldset.html" with heading_level=2 id_suffix=forloop.counter0 %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
@ -3,12 +3,14 @@
|
|||||||
id="{{ inline_admin_formset.formset.prefix }}-group"
|
id="{{ inline_admin_formset.formset.prefix }}-group"
|
||||||
data-inline-type="stacked"
|
data-inline-type="stacked"
|
||||||
data-inline-formset="{{ inline_admin_formset.inline_formset_data }}">
|
data-inline-formset="{{ inline_admin_formset.inline_formset_data }}">
|
||||||
<fieldset class="module {{ inline_admin_formset.classes }}">
|
<fieldset class="module {{ inline_admin_formset.classes }}" aria-labelledby="{{ inline_admin_formset.formset.prefix }}-heading">
|
||||||
|
<h2 id="{{ inline_admin_formset.formset.prefix }}-heading" class="inline-heading">
|
||||||
{% if inline_admin_formset.formset.max_num == 1 %}
|
{% if inline_admin_formset.formset.max_num == 1 %}
|
||||||
<h2>{{ inline_admin_formset.opts.verbose_name|capfirst }}</h2>
|
{{ inline_admin_formset.opts.verbose_name|capfirst }}
|
||||||
{% else %}
|
{% else %}
|
||||||
<h2>{{ inline_admin_formset.opts.verbose_name_plural|capfirst }}</h2>
|
{{ inline_admin_formset.opts.verbose_name_plural|capfirst }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
</h2>
|
||||||
{{ inline_admin_formset.formset.management_form }}
|
{{ inline_admin_formset.formset.management_form }}
|
||||||
{{ inline_admin_formset.formset.non_form_errors }}
|
{{ inline_admin_formset.formset.non_form_errors }}
|
||||||
|
|
||||||
@ -19,9 +21,13 @@
|
|||||||
{% if inline_admin_formset.formset.can_delete and inline_admin_formset.has_delete_permission and inline_admin_form.original %}<span class="delete">{{ inline_admin_form.deletion_field.field }} {{ inline_admin_form.deletion_field.label_tag }}</span>{% endif %}
|
{% if inline_admin_formset.formset.can_delete and inline_admin_formset.has_delete_permission and inline_admin_form.original %}<span class="delete">{{ inline_admin_form.deletion_field.field }} {{ inline_admin_form.deletion_field.label_tag }}</span>{% endif %}
|
||||||
</h3>
|
</h3>
|
||||||
{% if inline_admin_form.form.non_field_errors %}{{ inline_admin_form.form.non_field_errors }}{% endif %}
|
{% if inline_admin_form.form.non_field_errors %}{{ inline_admin_form.form.non_field_errors }}{% endif %}
|
||||||
{% for fieldset in inline_admin_form %}
|
|
||||||
{% include "admin/includes/fieldset.html" %}
|
{% with parent_counter=forloop.counter0 %}
|
||||||
{% endfor %}
|
{% for fieldset in inline_admin_form %}
|
||||||
|
{% include "admin/includes/fieldset.html" with heading_level=4 id_prefix=parent_counter id_suffix=forloop.counter0 %}
|
||||||
|
{% endfor %}
|
||||||
|
{% endwith %}
|
||||||
|
|
||||||
{% if inline_admin_form.needs_explicit_pk_field %}{{ inline_admin_form.pk_field.field }}{% endif %}
|
{% if inline_admin_form.needs_explicit_pk_field %}{{ inline_admin_form.pk_field.field }}{% endif %}
|
||||||
{% if inline_admin_form.fk_field %}{{ inline_admin_form.fk_field.field }}{% endif %}
|
{% if inline_admin_form.fk_field %}{{ inline_admin_form.fk_field.field }}{% endif %}
|
||||||
</div>{% endfor %}
|
</div>{% endfor %}
|
||||||
|
@ -4,12 +4,14 @@
|
|||||||
data-inline-formset="{{ inline_admin_formset.inline_formset_data }}">
|
data-inline-formset="{{ inline_admin_formset.inline_formset_data }}">
|
||||||
<div class="tabular inline-related {% if forloop.last %}last-related{% endif %}">
|
<div class="tabular inline-related {% if forloop.last %}last-related{% endif %}">
|
||||||
{{ inline_admin_formset.formset.management_form }}
|
{{ inline_admin_formset.formset.management_form }}
|
||||||
<fieldset class="module {{ inline_admin_formset.classes }}">
|
<fieldset class="module {{ inline_admin_formset.classes }}" aria-labelledby="{{ inline_admin_formset.formset.prefix }}-heading">
|
||||||
{% if inline_admin_formset.formset.max_num == 1 %}
|
<h2 id="{{ inline_admin_formset.formset.prefix }}-heading" class="inline-heading">
|
||||||
<h2>{{ inline_admin_formset.opts.verbose_name|capfirst }}</h2>
|
{% if inline_admin_formset.formset.max_num == 1 %}
|
||||||
{% else %}
|
{{ inline_admin_formset.opts.verbose_name|capfirst }}
|
||||||
<h2>{{ inline_admin_formset.opts.verbose_name_plural|capfirst }}</h2>
|
{% else %}
|
||||||
{% endif %}
|
{{ inline_admin_formset.opts.verbose_name_plural|capfirst }}
|
||||||
|
{% endif %}
|
||||||
|
</h2>
|
||||||
{{ inline_admin_formset.formset.non_form_errors }}
|
{{ inline_admin_formset.formset.non_form_errors }}
|
||||||
<table>
|
<table>
|
||||||
<thead><tr>
|
<thead><tr>
|
||||||
|
@ -1,5 +1,8 @@
|
|||||||
<fieldset class="module aligned {{ fieldset.classes }}">
|
{% with prefix=fieldset.formset.prefix|default:"fieldset" id_prefix=id_prefix|default:"0" id_suffix=id_suffix|default:"0" name=fieldset.name|default:""|slugify %}
|
||||||
{% if fieldset.name %}<h2>{{ fieldset.name }}</h2>{% endif %}
|
<fieldset class="module aligned {{ fieldset.classes }}"{% if name %} aria-labelledby="{{ prefix }}-{{ id_prefix}}-{{ name }}-{{ id_suffix }}-heading"{% endif %}>
|
||||||
|
{% if name %}
|
||||||
|
<h{{ heading_level|default:2 }} id="{{ prefix }}-{{ id_prefix}}-{{ name }}-{{ id_suffix }}-heading" class="fieldset-heading">{{ fieldset.name }}</h{{ heading_level|default:2 }}>
|
||||||
|
{% endif %}
|
||||||
{% if fieldset.description %}
|
{% if fieldset.description %}
|
||||||
<div class="description">{{ fieldset.description|safe }}</div>
|
<div class="description">{{ fieldset.description|safe }}</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -32,3 +35,4 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
{% endwith %}
|
||||||
|
@ -40,6 +40,8 @@ from .models import (
|
|||||||
OutfitItem,
|
OutfitItem,
|
||||||
ParentModelWithCustomPk,
|
ParentModelWithCustomPk,
|
||||||
Person,
|
Person,
|
||||||
|
Photo,
|
||||||
|
Photographer,
|
||||||
Poll,
|
Poll,
|
||||||
Profile,
|
Profile,
|
||||||
ProfileCollection,
|
ProfileCollection,
|
||||||
@ -98,6 +100,57 @@ class AuthorAdmin(admin.ModelAdmin):
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class PhotoInlineMixin:
|
||||||
|
model = Photo
|
||||||
|
extra = 2
|
||||||
|
fieldsets = [
|
||||||
|
(None, {"fields": ["image", "title"]}),
|
||||||
|
(
|
||||||
|
"Details",
|
||||||
|
{"fields": ["description", "creation_date"], "classes": ["collapse"]},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"Details", # Fieldset name intentionally duplicated
|
||||||
|
{"fields": ["update_date", "updated_by"]},
|
||||||
|
),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class PhotoTabularInline(PhotoInlineMixin, admin.TabularInline):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class PhotoStackedExtra2Inline(PhotoInlineMixin, admin.StackedInline):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class PhotoStackedExtra3Inline(PhotoInlineMixin, admin.StackedInline):
|
||||||
|
extra = 3
|
||||||
|
|
||||||
|
|
||||||
|
class PhotoStackedCollapsibleInline(PhotoInlineMixin, admin.StackedInline):
|
||||||
|
fieldsets = []
|
||||||
|
classes = ["collapse"]
|
||||||
|
|
||||||
|
|
||||||
|
class PhotographerAdmin(admin.ModelAdmin):
|
||||||
|
fieldsets = [
|
||||||
|
(None, {"fields": ["firstname", "fullname"]}),
|
||||||
|
("Advanced options", {"fields": ["nationality", "residency"]}),
|
||||||
|
(
|
||||||
|
"Advanced options", # Fieldset name intentionally duplicated
|
||||||
|
{"fields": ["siblings", "children"], "classes": ["collapse"]},
|
||||||
|
),
|
||||||
|
]
|
||||||
|
inlines = [
|
||||||
|
PhotoTabularInline,
|
||||||
|
PhotoTabularInline,
|
||||||
|
PhotoStackedExtra2Inline,
|
||||||
|
PhotoStackedExtra3Inline,
|
||||||
|
PhotoStackedCollapsibleInline,
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
class InnerInline(admin.StackedInline):
|
class InnerInline(admin.StackedInline):
|
||||||
model = Inner
|
model = Inner
|
||||||
can_delete = False
|
can_delete = False
|
||||||
@ -454,6 +507,7 @@ site.register(Teacher, TeacherAdmin)
|
|||||||
site.register(Chapter, inlines=[FootNoteNonEditableInlineCustomForm])
|
site.register(Chapter, inlines=[FootNoteNonEditableInlineCustomForm])
|
||||||
site.register(OutfitItem, inlines=[WeaknessInlineCustomForm])
|
site.register(OutfitItem, inlines=[WeaknessInlineCustomForm])
|
||||||
site.register(Person, inlines=[AuthorTabularInline, FashonistaStackedInline])
|
site.register(Person, inlines=[AuthorTabularInline, FashonistaStackedInline])
|
||||||
|
site.register(Photographer, PhotographerAdmin)
|
||||||
site.register(Course, ClassAdminStackedHorizontal)
|
site.register(Course, ClassAdminStackedHorizontal)
|
||||||
site.register(CourseProxy, ClassAdminStackedVertical)
|
site.register(CourseProxy, ClassAdminStackedVertical)
|
||||||
site.register(CourseProxy1, ClassAdminTabularVertical)
|
site.register(CourseProxy1, ClassAdminTabularVertical)
|
||||||
|
@ -180,6 +180,27 @@ class ShoppingWeakness(models.Model):
|
|||||||
item = models.ForeignKey(OutfitItem, models.CASCADE)
|
item = models.ForeignKey(OutfitItem, models.CASCADE)
|
||||||
|
|
||||||
|
|
||||||
|
# Models for #35189
|
||||||
|
|
||||||
|
|
||||||
|
class Photographer(Person):
|
||||||
|
fullname = models.CharField(max_length=100)
|
||||||
|
nationality = models.CharField(max_length=100)
|
||||||
|
residency = models.CharField(max_length=100)
|
||||||
|
siblings = models.IntegerField()
|
||||||
|
children = models.IntegerField()
|
||||||
|
|
||||||
|
|
||||||
|
class Photo(models.Model):
|
||||||
|
photographer = models.ForeignKey(Photographer, on_delete=models.CASCADE)
|
||||||
|
image = models.CharField(max_length=100)
|
||||||
|
title = models.CharField(max_length=100)
|
||||||
|
description = models.TextField()
|
||||||
|
creation_date = models.DateField()
|
||||||
|
update_date = models.DateField()
|
||||||
|
updated_by = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
|
||||||
# Models for #13510
|
# Models for #13510
|
||||||
|
|
||||||
|
|
||||||
|
@ -117,7 +117,14 @@ class TestInline(TestDataMixin, TestCase):
|
|||||||
"Autogenerated many-to-many inlines are displayed correctly (#13407)"
|
"Autogenerated many-to-many inlines are displayed correctly (#13407)"
|
||||||
response = self.client.get(reverse("admin:admin_inlines_author_add"))
|
response = self.client.get(reverse("admin:admin_inlines_author_add"))
|
||||||
# The heading for the m2m inline block uses the right text
|
# The heading for the m2m inline block uses the right text
|
||||||
self.assertContains(response, "<h2>Author-book relationships</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# The "add another" label is correct
|
# The "add another" label is correct
|
||||||
self.assertContains(response, "Add another Author-book relationship")
|
self.assertContains(response, "Add another Author-book relationship")
|
||||||
# The '+' is dropped from the autogenerated form prefix (Author_books+)
|
# The '+' is dropped from the autogenerated form prefix (Author_books+)
|
||||||
@ -737,13 +744,35 @@ class TestInline(TestDataMixin, TestCase):
|
|||||||
|
|
||||||
def test_inlines_plural_heading_foreign_key(self):
|
def test_inlines_plural_heading_foreign_key(self):
|
||||||
response = self.client.get(reverse("admin:admin_inlines_holder4_add"))
|
response = self.client.get(reverse("admin:admin_inlines_holder4_add"))
|
||||||
self.assertContains(response, "<h2>Inner4 stackeds</h2>", html=True)
|
self.assertContains(
|
||||||
self.assertContains(response, "<h2>Inner4 tabulars</h2>", html=True)
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="inner4stacked_set-heading" class="inline-heading">'
|
||||||
|
"Inner4 stackeds</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="inner4tabular_set-heading" class="inline-heading">'
|
||||||
|
"Inner4 tabulars</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
|
|
||||||
def test_inlines_singular_heading_one_to_one(self):
|
def test_inlines_singular_heading_one_to_one(self):
|
||||||
response = self.client.get(reverse("admin:admin_inlines_person_add"))
|
response = self.client.get(reverse("admin:admin_inlines_person_add"))
|
||||||
self.assertContains(response, "<h2>Author</h2>", html=True) # Tabular.
|
self.assertContains(
|
||||||
self.assertContains(response, "<h2>Fashionista</h2>", html=True) # Stacked.
|
response,
|
||||||
|
'<h2 id="author-heading" class="inline-heading">Author</h2>',
|
||||||
|
html=True,
|
||||||
|
) # Tabular.
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="fashionista-heading" class="inline-heading">Fashionista</h2>',
|
||||||
|
html=True,
|
||||||
|
) # Stacked.
|
||||||
|
|
||||||
def test_inlines_based_on_model_state(self):
|
def test_inlines_based_on_model_state(self):
|
||||||
parent = ShowInlineParent.objects.create(show_inlines=False)
|
parent = ShowInlineParent.objects.create(show_inlines=False)
|
||||||
@ -914,28 +943,50 @@ class TestInlinePermissions(TestCase):
|
|||||||
def test_inline_add_m2m_noperm(self):
|
def test_inline_add_m2m_noperm(self):
|
||||||
response = self.client.get(reverse("admin:admin_inlines_author_add"))
|
response = self.client.get(reverse("admin:admin_inlines_author_add"))
|
||||||
# No change permission on books, so no inline
|
# No change permission on books, so no inline
|
||||||
self.assertNotContains(response, "<h2>Author-book relationships</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Author-Book Relationship")
|
self.assertNotContains(response, "Add another Author-Book Relationship")
|
||||||
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
||||||
|
|
||||||
def test_inline_add_fk_noperm(self):
|
def test_inline_add_fk_noperm(self):
|
||||||
response = self.client.get(reverse("admin:admin_inlines_holder2_add"))
|
response = self.client.get(reverse("admin:admin_inlines_holder2_add"))
|
||||||
# No permissions on Inner2s, so no inline
|
# No permissions on Inner2s, so no inline
|
||||||
self.assertNotContains(response, "<h2>Inner2s</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Inner2")
|
self.assertNotContains(response, "Add another Inner2")
|
||||||
self.assertNotContains(response, 'id="id_inner2_set-TOTAL_FORMS"')
|
self.assertNotContains(response, 'id="id_inner2_set-TOTAL_FORMS"')
|
||||||
|
|
||||||
def test_inline_change_m2m_noperm(self):
|
def test_inline_change_m2m_noperm(self):
|
||||||
response = self.client.get(self.author_change_url)
|
response = self.client.get(self.author_change_url)
|
||||||
# No change permission on books, so no inline
|
# No change permission on books, so no inline
|
||||||
self.assertNotContains(response, "<h2>Author-book relationships</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Author-Book Relationship")
|
self.assertNotContains(response, "Add another Author-Book Relationship")
|
||||||
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
||||||
|
|
||||||
def test_inline_change_fk_noperm(self):
|
def test_inline_change_fk_noperm(self):
|
||||||
response = self.client.get(self.holder_change_url)
|
response = self.client.get(self.holder_change_url)
|
||||||
# No permissions on Inner2s, so no inline
|
# No permissions on Inner2s, so no inline
|
||||||
self.assertNotContains(response, "<h2>Inner2s</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Inner2")
|
self.assertNotContains(response, "Add another Inner2")
|
||||||
self.assertNotContains(response, 'id="id_inner2_set-TOTAL_FORMS"')
|
self.assertNotContains(response, 'id="id_inner2_set-TOTAL_FORMS"')
|
||||||
|
|
||||||
@ -959,7 +1010,14 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.assertIs(
|
self.assertIs(
|
||||||
response.context["inline_admin_formset"].has_delete_permission, False
|
response.context["inline_admin_formset"].has_delete_permission, False
|
||||||
)
|
)
|
||||||
self.assertContains(response, "<h2>Author-book relationships</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
'<input type="hidden" name="Author_books-TOTAL_FORMS" value="0" '
|
'<input type="hidden" name="Author_books-TOTAL_FORMS" value="0" '
|
||||||
@ -975,7 +1033,14 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(reverse("admin:admin_inlines_author_add"))
|
response = self.client.get(reverse("admin:admin_inlines_author_add"))
|
||||||
# No change permission on Books, so no inline
|
# No change permission on Books, so no inline
|
||||||
self.assertNotContains(response, "<h2>Author-book relationships</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Author-Book Relationship")
|
self.assertNotContains(response, "Add another Author-Book Relationship")
|
||||||
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
||||||
|
|
||||||
@ -986,7 +1051,11 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(reverse("admin:admin_inlines_holder2_add"))
|
response = self.client.get(reverse("admin:admin_inlines_holder2_add"))
|
||||||
# Add permission on inner2s, so we get the inline
|
# Add permission on inner2s, so we get the inline
|
||||||
self.assertContains(response, "<h2>Inner2s</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Inner2")
|
self.assertContains(response, "Add another Inner2")
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
@ -1002,7 +1071,14 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(self.author_change_url)
|
response = self.client.get(self.author_change_url)
|
||||||
# No change permission on books, so no inline
|
# No change permission on books, so no inline
|
||||||
self.assertNotContains(response, "<h2>Author-book relationships</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Author-Book Relationship")
|
self.assertNotContains(response, "Add another Author-Book Relationship")
|
||||||
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
self.assertNotContains(response, 'id="id_Author_books-TOTAL_FORMS"')
|
||||||
self.assertNotContains(response, 'id="id_Author_books-0-DELETE"')
|
self.assertNotContains(response, 'id="id_Author_books-0-DELETE"')
|
||||||
@ -1026,7 +1102,14 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.assertIs(
|
self.assertIs(
|
||||||
response.context["inline_admin_formset"].has_delete_permission, False
|
response.context["inline_admin_formset"].has_delete_permission, False
|
||||||
)
|
)
|
||||||
self.assertContains(response, "<h2>Author-book relationships</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
'<input type="hidden" name="Author_books-TOTAL_FORMS" value="1" '
|
'<input type="hidden" name="Author_books-TOTAL_FORMS" value="1" '
|
||||||
@ -1059,7 +1142,14 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.assertIs(
|
self.assertIs(
|
||||||
response.context["inline_admin_formset"].has_delete_permission, True
|
response.context["inline_admin_formset"].has_delete_permission, True
|
||||||
)
|
)
|
||||||
self.assertContains(response, "<h2>Author-book relationships</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="Author_books-heading" class="inline-heading">'
|
||||||
|
"Author-book relationships</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Author-book relationship")
|
self.assertContains(response, "Add another Author-book relationship")
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
@ -1082,7 +1172,11 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(self.holder_change_url)
|
response = self.client.get(self.holder_change_url)
|
||||||
# Add permission on inner2s, so we can add but not modify existing
|
# Add permission on inner2s, so we can add but not modify existing
|
||||||
self.assertContains(response, "<h2>Inner2s</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Inner2")
|
self.assertContains(response, "Add another Inner2")
|
||||||
# 3 extra forms only, not the existing instance form
|
# 3 extra forms only, not the existing instance form
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
@ -1105,7 +1199,16 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(self.holder_change_url)
|
response = self.client.get(self.holder_change_url)
|
||||||
# Change permission on inner2s, so we can change existing but not add new
|
# Change permission on inner2s, so we can change existing but not add new
|
||||||
self.assertContains(response, "<h2>Inner2s</h2>", count=2)
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Just the one form for existing instances
|
# Just the one form for existing instances
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
@ -1148,7 +1251,11 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(self.holder_change_url)
|
response = self.client.get(self.holder_change_url)
|
||||||
# Add/change perm, so we can add new and change existing
|
# Add/change perm, so we can add new and change existing
|
||||||
self.assertContains(response, "<h2>Inner2s</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# One form for existing instance and three extra for new
|
# One form for existing instance and three extra for new
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
@ -1174,7 +1281,11 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(self.holder_change_url)
|
response = self.client.get(self.holder_change_url)
|
||||||
# Change/delete perm on inner2s, so we can change/delete existing
|
# Change/delete perm on inner2s, so we can change/delete existing
|
||||||
self.assertContains(response, "<h2>Inner2s</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# One form for existing instance only, no new
|
# One form for existing instance only, no new
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
@ -1205,7 +1316,16 @@ class TestInlinePermissions(TestCase):
|
|||||||
self.user.user_permissions.add(permission)
|
self.user.user_permissions.add(permission)
|
||||||
response = self.client.get(self.holder_change_url)
|
response = self.client.get(self.holder_change_url)
|
||||||
# All perms on inner2s, so we can add/change/delete
|
# All perms on inner2s, so we can add/change/delete
|
||||||
self.assertContains(response, "<h2>Inner2s</h2>", count=2)
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="inner2_set-2-heading" class="inline-heading">Inner2s</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# One form for existing instance only, three for new
|
# One form for existing instance only, three for new
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
@ -1367,22 +1487,69 @@ class TestVerboseNameInlineForms(TestDataMixin, TestCase):
|
|||||||
response = modeladmin.changeform_view(request)
|
response = modeladmin.changeform_view(request)
|
||||||
self.assertNotContains(response, "Add another Profile")
|
self.assertNotContains(response, "Add another Profile")
|
||||||
# Non-verbose model.
|
# Non-verbose model.
|
||||||
self.assertContains(response, "<h2>Non-verbose childss</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="profile_set-heading" class="inline-heading">'
|
||||||
|
"Non-verbose childss</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Non-verbose child")
|
self.assertContains(response, "Add another Non-verbose child")
|
||||||
self.assertNotContains(response, "<h2>Profiles</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="profile_set-heading" class="inline-heading">Profiles</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with verbose name.
|
# Model with verbose name.
|
||||||
self.assertContains(response, "<h2>Childs with verbose names</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="verbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with verbose names</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Childs with verbose name")
|
self.assertContains(response, "Add another Childs with verbose name")
|
||||||
self.assertNotContains(response, "<h2>Model with verbose name onlys</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="verbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with verbose name onlys</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Model with verbose name only")
|
self.assertNotContains(response, "Add another Model with verbose name only")
|
||||||
# Model with verbose name plural.
|
# Model with verbose name plural.
|
||||||
self.assertContains(response, "<h2>Childs with verbose name plurals</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="verbosenamepluralprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with verbose name plurals</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Childs with verbose name plural")
|
self.assertContains(response, "Add another Childs with verbose name plural")
|
||||||
self.assertNotContains(response, "<h2>Model with verbose name plural only</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="verbosenamepluralprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with verbose name plural only</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with both verbose names.
|
# Model with both verbose names.
|
||||||
self.assertContains(response, "<h2>Childs with both verbose namess</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="bothverbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with both verbose namess</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Childs with both verbose names")
|
self.assertContains(response, "Add another Childs with both verbose names")
|
||||||
self.assertNotContains(response, "<h2>Model with both - plural name</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="bothverbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with both - plural name</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Model with both - name")
|
self.assertNotContains(response, "Add another Model with both - name")
|
||||||
|
|
||||||
def test_verbose_name_plural_inline(self):
|
def test_verbose_name_plural_inline(self):
|
||||||
@ -1415,21 +1582,68 @@ class TestVerboseNameInlineForms(TestDataMixin, TestCase):
|
|||||||
request.user = self.superuser
|
request.user = self.superuser
|
||||||
response = modeladmin.changeform_view(request)
|
response = modeladmin.changeform_view(request)
|
||||||
# Non-verbose model.
|
# Non-verbose model.
|
||||||
self.assertContains(response, "<h2>Non-verbose childs</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="profile_set-heading" class="inline-heading">'
|
||||||
|
"Non-verbose childs</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Profile")
|
self.assertContains(response, "Add another Profile")
|
||||||
self.assertNotContains(response, "<h2>Profiles</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="profile_set-heading" class="inline-heading">Profiles</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with verbose name.
|
# Model with verbose name.
|
||||||
self.assertContains(response, "<h2>Childs with verbose name</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="verbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with verbose name</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Model with verbose name only")
|
self.assertContains(response, "Add another Model with verbose name only")
|
||||||
self.assertNotContains(response, "<h2>Model with verbose name onlys</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="verbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with verbose name onlys</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with verbose name plural.
|
# Model with verbose name plural.
|
||||||
self.assertContains(response, "<h2>Childs with verbose name plural</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="verbosenamepluralprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with verbose name plural</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Profile")
|
self.assertContains(response, "Add another Profile")
|
||||||
self.assertNotContains(response, "<h2>Model with verbose name plural only</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="verbosenamepluralprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with verbose name plural only</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with both verbose names.
|
# Model with both verbose names.
|
||||||
self.assertContains(response, "<h2>Childs with both verbose names</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="bothverbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with both verbose names</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Model with both - name")
|
self.assertContains(response, "Add another Model with both - name")
|
||||||
self.assertNotContains(response, "<h2>Model with both - plural name</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="bothverbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with both - plural name</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
|
|
||||||
def test_both_verbose_names_inline(self):
|
def test_both_verbose_names_inline(self):
|
||||||
class NonVerboseProfileInline(TabularInline):
|
class NonVerboseProfileInline(TabularInline):
|
||||||
@ -1466,30 +1680,148 @@ class TestVerboseNameInlineForms(TestDataMixin, TestCase):
|
|||||||
response = modeladmin.changeform_view(request)
|
response = modeladmin.changeform_view(request)
|
||||||
self.assertNotContains(response, "Add another Profile")
|
self.assertNotContains(response, "Add another Profile")
|
||||||
# Non-verbose model.
|
# Non-verbose model.
|
||||||
self.assertContains(response, "<h2>Non-verbose childs - plural name</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="profile_set-heading" class="inline-heading">'
|
||||||
|
"Non-verbose childs - plural name</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Non-verbose childs - name")
|
self.assertContains(response, "Add another Non-verbose childs - name")
|
||||||
self.assertNotContains(response, "<h2>Profiles</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="profile_set-heading" class="inline-heading">Profiles</h2>',
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with verbose name.
|
# Model with verbose name.
|
||||||
self.assertContains(response, "<h2>Childs with verbose name - plural name</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
(
|
||||||
|
'<h2 id="verbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with verbose name - plural name</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Childs with verbose name - name")
|
self.assertContains(response, "Add another Childs with verbose name - name")
|
||||||
self.assertNotContains(response, "<h2>Model with verbose name onlys</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="verbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with verbose name onlys</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with verbose name plural.
|
# Model with verbose name plural.
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
"<h2>Childs with verbose name plural - plural name</h2>",
|
(
|
||||||
|
'<h2 id="verbosenamepluralprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with verbose name plural - plural name</h2>"
|
||||||
|
),
|
||||||
|
html=True,
|
||||||
)
|
)
|
||||||
self.assertContains(
|
self.assertContains(
|
||||||
response,
|
response,
|
||||||
"Add another Childs with verbose name plural - name",
|
"Add another Childs with verbose name plural - name",
|
||||||
)
|
)
|
||||||
self.assertNotContains(response, "<h2>Model with verbose name plural only</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="verbosenamepluralprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with verbose name plural only</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
# Model with both verbose names.
|
# Model with both verbose names.
|
||||||
self.assertContains(response, "<h2>Childs with both - plural name</h2>")
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="bothverbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Childs with both - plural name</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertContains(response, "Add another Childs with both - name")
|
self.assertContains(response, "Add another Childs with both - name")
|
||||||
self.assertNotContains(response, "<h2>Model with both - plural name</h2>")
|
self.assertNotContains(
|
||||||
|
response,
|
||||||
|
'<h2 id="bothverbosenameprofile_set-heading" class="inline-heading">'
|
||||||
|
"Model with both - plural name</h2>",
|
||||||
|
html=True,
|
||||||
|
)
|
||||||
self.assertNotContains(response, "Add another Model with both - name")
|
self.assertNotContains(response, "Add another Model with both - name")
|
||||||
|
|
||||||
|
|
||||||
|
@override_settings(ROOT_URLCONF="admin_inlines.urls")
|
||||||
|
class TestInlineWithFieldsets(TestDataMixin, TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
self.client.force_login(self.superuser)
|
||||||
|
|
||||||
|
def test_inline_headings(self):
|
||||||
|
response = self.client.get(reverse("admin:admin_inlines_photographer_add"))
|
||||||
|
# Page main title.
|
||||||
|
self.assertContains(response, "<h1>Add photographer</h1>", html=True)
|
||||||
|
|
||||||
|
# Headings for the toplevel fieldsets. The first one has no name.
|
||||||
|
self.assertContains(response, '<fieldset class="module aligned ">')
|
||||||
|
# The second and third have the same "Advanced options" name, but the
|
||||||
|
# second one has the "collapse" class.
|
||||||
|
for x, classes in ((1, ""), (2, "collapse")):
|
||||||
|
heading_id = f"fieldset-0-advanced-options-{x}-heading"
|
||||||
|
with self.subTest(heading_id=heading_id):
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
f'<fieldset class="module aligned {classes}" '
|
||||||
|
f'aria-labelledby="{heading_id}">',
|
||||||
|
)
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
f'<h2 id="{heading_id}" class="fieldset-heading">'
|
||||||
|
"Advanced options</h2>",
|
||||||
|
)
|
||||||
|
self.assertContains(response, f'id="{heading_id}"', count=1)
|
||||||
|
|
||||||
|
# Headings and subheadings for all the inlines.
|
||||||
|
for inline_admin_formset in response.context["inline_admin_formsets"]:
|
||||||
|
prefix = inline_admin_formset.formset.prefix
|
||||||
|
heading_id = f"{prefix}-heading"
|
||||||
|
formset_heading = (
|
||||||
|
f'<h2 id="{heading_id}" class="inline-heading">Photos</h2>'
|
||||||
|
)
|
||||||
|
self.assertContains(response, formset_heading, html=True)
|
||||||
|
self.assertContains(response, f'id="{heading_id}"', count=1)
|
||||||
|
|
||||||
|
# If this is a TabularInline, do not make further asserts since
|
||||||
|
# fieldsets are not shown as such in this table layout.
|
||||||
|
if "tabular" in inline_admin_formset.opts.template:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Headings for every formset (the amount depends on `extra`).
|
||||||
|
for y, inline_admin_form in enumerate(inline_admin_formset):
|
||||||
|
y_plus_one = y + 1
|
||||||
|
form_heading = (
|
||||||
|
f'<h3><b>Photo:</b> <span class="inline_label">#{y_plus_one}</span>'
|
||||||
|
"</h3>"
|
||||||
|
)
|
||||||
|
self.assertContains(response, form_heading, html=True)
|
||||||
|
|
||||||
|
# Every fieldset defined for an inline's form.
|
||||||
|
for z, fieldset in enumerate(inline_admin_form):
|
||||||
|
if fieldset.name:
|
||||||
|
heading_id = f"{prefix}-{y}-details-{z}-heading"
|
||||||
|
self.assertContains(
|
||||||
|
response,
|
||||||
|
f'<fieldset class="module aligned {fieldset.classes}" '
|
||||||
|
f'aria-labelledby="{heading_id}">',
|
||||||
|
)
|
||||||
|
fieldset_heading = (
|
||||||
|
f'<h4 id="{heading_id}" class="fieldset-heading">'
|
||||||
|
f"Details</h4>"
|
||||||
|
)
|
||||||
|
self.assertContains(response, fieldset_heading)
|
||||||
|
self.assertContains(response, f'id="{heading_id}"', count=1)
|
||||||
|
|
||||||
|
else:
|
||||||
|
fieldset_html = (
|
||||||
|
f'<fieldset class="module aligned {fieldset.classes}">'
|
||||||
|
)
|
||||||
|
self.assertContains(response, fieldset_html)
|
||||||
|
|
||||||
|
|
||||||
@override_settings(ROOT_URLCONF="admin_inlines.urls")
|
@override_settings(ROOT_URLCONF="admin_inlines.urls")
|
||||||
class SeleniumTests(AdminSeleniumTestCase):
|
class SeleniumTests(AdminSeleniumTestCase):
|
||||||
available_apps = ["admin_inlines"] + AdminSeleniumTestCase.available_apps
|
available_apps = ["admin_inlines"] + AdminSeleniumTestCase.available_apps
|
||||||
|
Loading…
Reference in New Issue
Block a user