mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Refs #35706 -- Prefixed 'Error:' to titles of admin pages with form errors.
This improves the screen reader experience.
This commit is contained in:
		
				
					committed by
					
						 Sarah Boyce
						Sarah Boyce
					
				
			
			
				
	
			
			
			
						parent
						
							e4a2e22ddb
						
					
				
				
					commit
					387475c5b2
				
			| @@ -2,6 +2,7 @@ | |||||||
| {% load i18n static %} | {% load i18n static %} | ||||||
| {% load admin_urls %} | {% load admin_urls %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if form.errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrastyle %} | {% block extrastyle %} | ||||||
|   {{ block.super }} |   {{ block.super }} | ||||||
|   <link rel="stylesheet" href="{% static "admin/css/forms.css" %}"> |   <link rel="stylesheet" href="{% static "admin/css/forms.css" %}"> | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| {% extends "admin/base_site.html" %} | {% extends "admin/base_site.html" %} | ||||||
| {% load i18n admin_urls static admin_modify %} | {% load i18n admin_urls static admin_modify %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrahead %}{{ block.super }} | {% block extrahead %}{{ block.super }} | ||||||
| <script src="{% url 'admin:jsi18n' %}"></script> | <script src="{% url 'admin:jsi18n' %}"></script> | ||||||
| {{ media }} | {{ media }} | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| {% extends "admin/base_site.html" %} | {% extends "admin/base_site.html" %} | ||||||
| {% load i18n admin_urls static admin_list %} | {% load i18n admin_urls static admin_list %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if cl.formset and cl.formset.errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrastyle %} | {% block extrastyle %} | ||||||
|   {{ block.super }} |   {{ block.super }} | ||||||
|   <link rel="stylesheet" href="{% static "admin/css/changelists.css" %}"> |   <link rel="stylesheet" href="{% static "admin/css/changelists.css" %}"> | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| {% extends "admin/base_site.html" %} | {% extends "admin/base_site.html" %} | ||||||
| {% load i18n static %} | {% load i18n static %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if form.errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/login.css" %}"> | {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/login.css" %}"> | ||||||
| {{ form.media }} | {{ form.media }} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| {% extends "admin/base_site.html" %} | {% extends "admin/base_site.html" %} | ||||||
| {% load i18n static %} | {% load i18n static %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if form.errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/forms.css" %}">{% endblock %} | {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/forms.css" %}">{% endblock %} | ||||||
| {% block userlinks %} | {% block userlinks %} | ||||||
|   {% url 'django-admindocs-docroot' as docsroot %}{% if docsroot %}<a href="{{ docsroot }}">{% translate 'Documentation' %}</a> / {% endif %} {% translate 'Change password' %} / |   {% url 'django-admindocs-docroot' as docsroot %}{% if docsroot %}<a href="{{ docsroot }}">{% translate 'Documentation' %}</a> / {% endif %} {% translate 'Change password' %} / | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| {% extends "admin/base_site.html" %} | {% extends "admin/base_site.html" %} | ||||||
| {% load i18n static %} | {% load i18n static %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if form.new_password1.errors or form.new_password2.errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/forms.css" %}">{% endblock %} | {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/forms.css" %}">{% endblock %} | ||||||
| {% block breadcrumbs %} | {% block breadcrumbs %} | ||||||
| <div class="breadcrumbs"> | <div class="breadcrumbs"> | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| {% extends "admin/base_site.html" %} | {% extends "admin/base_site.html" %} | ||||||
| {% load i18n static %} | {% load i18n static %} | ||||||
|  |  | ||||||
|  | {% block title %}{% if form.email.errors %}{{ _('Error:') }} {% endif %}{{ block.super }}{% endblock %} | ||||||
| {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/forms.css" %}">{% endblock %} | {% block extrastyle %}{{ block.super }}<link rel="stylesheet" href="{% static "admin/css/forms.css" %}">{% endblock %} | ||||||
| {% block breadcrumbs %} | {% block breadcrumbs %} | ||||||
| <div class="breadcrumbs"> | <div class="breadcrumbs"> | ||||||
|   | |||||||
| @@ -1264,6 +1264,24 @@ class ChangeListTests(TestCase): | |||||||
|             # Check only the first few characters since the UUID may have dashes. |             # Check only the first few characters since the UUID may have dashes. | ||||||
|             self.assertIn(str(a.pk)[:8], context.captured_queries[4]["sql"]) |             self.assertIn(str(a.pk)[:8], context.captured_queries[4]["sql"]) | ||||||
|  |  | ||||||
|  |     def test_list_editable_error_title(self): | ||||||
|  |         a = Swallow.objects.create(origin="Swallow A", load=4, speed=1) | ||||||
|  |         Swallow.objects.create(origin="Swallow B", load=2, speed=2) | ||||||
|  |         data = { | ||||||
|  |             "form-TOTAL_FORMS": "2", | ||||||
|  |             "form-INITIAL_FORMS": "2", | ||||||
|  |             "form-MIN_NUM_FORMS": "0", | ||||||
|  |             "form-MAX_NUM_FORMS": "1000", | ||||||
|  |             "form-0-uuid": str(a.pk), | ||||||
|  |             "form-0-load": "invalid", | ||||||
|  |             "_save": "Save", | ||||||
|  |         } | ||||||
|  |         superuser = self._create_superuser("superuser") | ||||||
|  |         self.client.force_login(superuser) | ||||||
|  |         changelist_url = reverse("admin:admin_changelist_swallow_changelist") | ||||||
|  |         response = self.client.post(changelist_url, data=data) | ||||||
|  |         self.assertContains(response, "Error: Select swallow to change") | ||||||
|  |  | ||||||
|     def test_deterministic_order_for_unordered_model(self): |     def test_deterministic_order_for_unordered_model(self): | ||||||
|         """ |         """ | ||||||
|         The primary key is used in the ordering of the changelist's results to |         The primary key is used in the ordering of the changelist's results to | ||||||
|   | |||||||
| @@ -1508,6 +1508,24 @@ class AdminViewBasicTest(AdminViewBasicTestCase): | |||||||
|         self.assertContains(response, "<h1>Change article</h1>") |         self.assertContains(response, "<h1>Change article</h1>") | ||||||
|         self.assertContains(response, "<h2>Article 2</h2>") |         self.assertContains(response, "<h2>Article 2</h2>") | ||||||
|  |  | ||||||
|  |     def test_error_in_titles(self): | ||||||
|  |         for url, subtitle in [ | ||||||
|  |             ( | ||||||
|  |                 reverse("admin:admin_views_article_change", args=(self.a1.pk,)), | ||||||
|  |                 "Article 1 | Change article", | ||||||
|  |             ), | ||||||
|  |             (reverse("admin:admin_views_article_add"), "Add article"), | ||||||
|  |             (reverse("admin:login"), "Log in"), | ||||||
|  |             (reverse("admin:password_change"), "Password change"), | ||||||
|  |             ( | ||||||
|  |                 reverse("admin:auth_user_password_change", args=(self.superuser.id,)), | ||||||
|  |                 "Change password: super", | ||||||
|  |             ), | ||||||
|  |         ]: | ||||||
|  |             with self.subTest(url=url, subtitle=subtitle): | ||||||
|  |                 response = self.client.post(url, {}) | ||||||
|  |                 self.assertContains(response, f"<title>Error: {subtitle}") | ||||||
|  |  | ||||||
|     def test_view_subtitle_per_object(self): |     def test_view_subtitle_per_object(self): | ||||||
|         viewuser = User.objects.create_user( |         viewuser = User.objects.create_user( | ||||||
|             username="viewuser", |             username="viewuser", | ||||||
|   | |||||||
| @@ -37,6 +37,12 @@ class AuthTemplateTests(TestCase): | |||||||
|         ) |         ) | ||||||
|         self.assertContains(response, "<h1>Password reset</h1>") |         self.assertContains(response, "<h1>Password reset</h1>") | ||||||
|  |  | ||||||
|  |     def test_password_reset_view_error_title(self): | ||||||
|  |         response = self.client.post(reverse("password_reset"), {}) | ||||||
|  |         self.assertContains( | ||||||
|  |             response, "<title>Error: Password reset | Django site admin</title>" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def test_password_reset_done_view(self): |     def test_password_reset_done_view(self): | ||||||
|         response = PasswordResetDoneView.as_view()(self.request) |         response = PasswordResetDoneView.as_view()(self.request) | ||||||
|         self.assertContains( |         self.assertContains( | ||||||
| @@ -77,6 +83,19 @@ class AuthTemplateTests(TestCase): | |||||||
|             '<input class="hidden" autocomplete="username" value="jsmith">', |             '<input class="hidden" autocomplete="username" value="jsmith">', | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     def test_password_reset_confirm_view_error_title(self): | ||||||
|  |         client = PasswordResetConfirmClient() | ||||||
|  |         default_token_generator = PasswordResetTokenGenerator() | ||||||
|  |         token = default_token_generator.make_token(self.user) | ||||||
|  |         uidb64 = urlsafe_base64_encode(str(self.user.pk).encode()) | ||||||
|  |         url = reverse( | ||||||
|  |             "password_reset_confirm", kwargs={"uidb64": uidb64, "token": token} | ||||||
|  |         ) | ||||||
|  |         response = client.post(url, {}) | ||||||
|  |         self.assertContains( | ||||||
|  |             response, "<title>Error: Enter new password | Django site admin</title>" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     @override_settings(AUTH_USER_MODEL="auth_tests.CustomUser") |     @override_settings(AUTH_USER_MODEL="auth_tests.CustomUser") | ||||||
|     def test_password_reset_confirm_view_custom_username_hint(self): |     def test_password_reset_confirm_view_custom_username_hint(self): | ||||||
|         custom_user = CustomUser.custom_objects.create_user( |         custom_user = CustomUser.custom_objects.create_user( | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user