diff --git a/django/contrib/admin/static/admin/css/base.css b/django/contrib/admin/static/admin/css/base.css index 79ea073bda..c672a3051f 100644 --- a/django/contrib/admin/static/admin/css/base.css +++ b/django/contrib/admin/static/admin/css/base.css @@ -1164,7 +1164,19 @@ a.deletelink:focus, a.deletelink:hover { box-sizing: border-box; } +.paginator ul { + margin: 0; + margin-right: 6px; +} + +.paginator ul li { + display: inline-block; + line-height: 22px; + padding: 0; +} + .paginator a:link, .paginator a:visited { + display: inline-block; padding: 2px 6px; background: var(--button-bg); text-decoration: none; @@ -1182,18 +1194,15 @@ a.deletelink:focus, a.deletelink:hover { color: var(--link-hover-color); } -.paginator .end { - margin-right: 6px; -} - -.paginator .this-page { - padding: 2px 6px; +.paginator a[aria-current="page"] { + color: var(--body-quiet-color); + background: transparent; font-weight: bold; - font-size: 0.8125rem; - vertical-align: top; + cursor: default; } -.paginator a:focus, .paginator a:hover { +.paginator a:not([aria-current="page"]):focus, +.paginator a:not([aria-current="page"]):hover { color: white; background: var(--link-hover-color); } diff --git a/django/contrib/admin/static/admin/css/changelists.css b/django/contrib/admin/static/admin/css/changelists.css index 005b7768c8..1edca65698 100644 --- a/django/contrib/admin/static/admin/css/changelists.css +++ b/django/contrib/admin/static/admin/css/changelists.css @@ -43,6 +43,22 @@ border-bottom: 1px solid var(--hairline-color); } +#changelist .changelist-footer { + display: flex; + align-items: center; + padding: 10px; + border-top: 1px solid var(--hairline-color); + border-bottom: 1px solid var(--hairline-color); +} + +#changelist .changelist-footer .paginator { + color: var(--body-quiet-color); + background: var(--body-bg); + border: none; + padding: 0; + overflow: hidden; +} + #changelist .paginator { color: var(--body-quiet-color); border-bottom: 1px solid var(--hairline-color); @@ -50,6 +66,10 @@ overflow: hidden; } +#changelist .paginator ul { + padding: 0; +} + /* CHANGELIST TABLES */ #changelist table thead th { diff --git a/django/contrib/admin/static/admin/css/rtl.css b/django/contrib/admin/static/admin/css/rtl.css index 0d843726bf..b824d7d0db 100644 --- a/django/contrib/admin/static/admin/css/rtl.css +++ b/django/contrib/admin/static/admin/css/rtl.css @@ -107,7 +107,7 @@ thead th.sorted .text { border-left: none; } -.paginator .end { +.paginator ul { margin-left: 6px; margin-right: 0; } diff --git a/django/contrib/admin/templates/admin/change_list.html b/django/contrib/admin/templates/admin/change_list.html index 162560dabc..3b3ea408d3 100644 --- a/django/contrib/admin/templates/admin/change_list.html +++ b/django/contrib/admin/templates/admin/change_list.html @@ -69,7 +69,12 @@ {% result_list cl %} {% if action_form and actions_on_bottom and cl.show_admin_actions %}{% admin_actions %}{% endif %} {% endblock %} - {% block pagination %}{% pagination cl %}{% endblock %} + {% block pagination %} +
{% block filters %} diff --git a/django/contrib/admin/templates/admin/object_history.html b/django/contrib/admin/templates/admin/object_history.html index 2b3b06caa0..40aaecc40a 100644 --- a/django/contrib/admin/templates/admin/object_history.html +++ b/django/contrib/admin/templates/admin/object_history.html @@ -34,20 +34,23 @@ {% endfor %} -+
+ {% else %}{% translate 'This object doesn’t have a change history. It probably wasn’t added via this admin site.' %}
{% endif %} diff --git a/django/contrib/admin/templates/admin/pagination.html b/django/contrib/admin/templates/admin/pagination.html index bc3117bfcb..954e710856 100644 --- a/django/contrib/admin/templates/admin/pagination.html +++ b/django/contrib/admin/templates/admin/pagination.html @@ -1,12 +1,14 @@ {% load admin_list %} {% load i18n %} --{% if pagination_required %} -{% for i in page_range %} - {% paginator_number cl i %} -{% endfor %} -{% endif %} +
+ diff --git a/django/contrib/admin/templatetags/admin_list.py b/django/contrib/admin/templatetags/admin_list.py index e7cb244af1..1e6f8bf298 100644 --- a/django/contrib/admin/templatetags/admin_list.py +++ b/django/contrib/admin/templatetags/admin_list.py @@ -42,12 +42,14 @@ def paginator_number(cl, i): if i == cl.paginator.ELLIPSIS: return format_html("{} ", cl.paginator.ELLIPSIS) elif i == cl.page_num: - return format_html('{} ', i) + return format_html( + '{} ', + i, + ) else: return format_html( - '{} ', + '{} ', cl.get_query_string({PAGE_VAR: i}), - mark_safe(' class="end"' if i == cl.paginator.num_pages else ""), i, ) diff --git a/tests/admin_changelist/tests.py b/tests/admin_changelist/tests.py index 7203daa6b0..cd2fe7c645 100644 --- a/tests/admin_changelist/tests.py +++ b/tests/admin_changelist/tests.py @@ -946,6 +946,41 @@ class ChangeListTests(TestCase): self.assertEqual(cl.paginator.count, 30) self.assertEqual(list(cl.paginator.page_range), [1, 2, 3]) + def test_pagination_render(self): + objs = [Swallow(origin=f"Swallow {i}", load=i, speed=i) for i in range(1, 5)] + Swallow.objects.bulk_create(objs) + + request = self.factory.get("/child/") + request.user = self.superuser + + admin = SwallowAdmin(Swallow, custom_site) + cl = admin.get_changelist_instance(request) + template = Template( + "{% load admin_list %}{% spaceless %}{% pagination cl %}{% endspaceless %}" + ) + context = Context({"cl": cl, "opts": cl.opts}) + pagination_output = template.render(context) + self.assertTrue( + pagination_output.startswith( + '")) + self.assertInHTML( + '