{
  "id": "django__django-11400",
  "question": "Ordering problem in admin.RelatedFieldListFilter and admin.RelatedOnlyFieldListFilter\nDescription\n\t\nRelatedFieldListFilter doesn't fall back to the ordering defined in Model._meta.ordering. \nOrdering gets set to an empty tuple in ​https://github.com/django/django/blob/2.2.1/django/contrib/admin/filters.py#L196 and unless ordering is defined on the related model's ModelAdmin class it stays an empty tuple. IMHO it should fall back to the ordering defined in the related model's Meta.ordering field.\nRelatedOnlyFieldListFilter doesn't order the related model at all, even if ordering is defined on the related model's ModelAdmin class.\nThat's because the call to field.get_choices ​https://github.com/django/django/blob/2.2.1/django/contrib/admin/filters.py#L422 omits the ordering kwarg entirely.\n",
  "answer": "{\"task_id\": \"0e19b05a-2086-4fe7-9dfa-7534def0d643\", \"workflow_run_id\": \"695831a4-c5d1-432d-bdb8-1fe04bb88074\", \"data\": {\"id\": \"695831a4-c5d1-432d-bdb8-1fe04bb88074\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -193,7 +193,11 @@ class RelatedFieldListFilter(FieldListFilter):\\n         other_model = get_model_from_relation(field)\\n         self.lookup_kwarg = '%s__%s__exact' % (field_path, field.target_field.name)\\n         self.lookup_kwarg_isnull = '%s__isnull' % field_path\\n-        self.lookup_choices = self.field_choices(field, request, model_admin)\\n+        # Get ordering from ModelAdmin or fall back to model's Meta.ordering\\n+        if hasattr(other_model, '_default_manager'):\\n+            ordering = getattr(model_admin, 'ordering', None) or other_model._meta.ordering\\n+        else:\\n+            ordering = getattr(model_admin, 'ordering', ())\\n+        self.lookup_choices = self.field_choices(field, request, model_admin, ordering=ordering)\\n         if hasattr(field, 'verbose_name'):\\n             self.lookup_title = field.verbose_name\\n         else:\\n@@ -201,8 +205,8 @@ class RelatedFieldListFilter(FieldListFilter):\\n         self.title = self.lookup_title\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n-    def field_choices(self, field, request, model_admin):\\n-        ordering = self.field_admin_ordering(field, request, model_admin)\\n+    def field_choices(self, field, request, model_admin, ordering=None):\\n+        ordering = ordering or self.field_admin_ordering(field, request, model_admin)\\n         return field.get_choices(include_blank=False, ordering=ordering)\\n \\n     def field_admin_ordering(self, field, request, model_admin):\\n@@ -419,7 +423,12 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        # Get ordering from ModelAdmin or fall back to model's Meta.ordering\\n+        other_model = get_model_from_relation(field)\\n+        if hasattr(other_model, '_default_manager'):\\n+            ordering = getattr(model_admin, 'ordering', None) or other_model._meta.ordering\\n+        else:\\n+            ordering = getattr(model_admin, 'ordering', ())\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\", \"--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -207,7 +207,12 @@ class RelatedFieldListFilter(FieldListFilter):\\n \\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n         Return the model admin's ordering for the related model, if any.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        ordering = getattr(model_admin, 'ordering', None)\\n+        if ordering:\\n+            return ordering\\n+        # Fall back to the related model's Meta.ordering\\n+        other_model = get_model_from_relation(field)\\n+        return getattr(other_model._meta, 'ordering', ())\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +424,8 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        ordering = self.field_admin_ordering(field, request, model_admin)\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\", \"--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -187,6 +187,16 @@ class RelatedFieldListFilter(FieldListFilter):\\n     def __init__(self, field, request, params, model, model_admin, field_path):\\n         other_model = get_model_from_relation(field)\\n         if hasattr(field, 'remote_field'):\\n+    \\n+    def _get_related_ordering(self, field, model_admin):\\n+        \\\"\\\"\\\"\\n+        Get ordering for the related model, preferring ModelAdmin ordering\\n+        over model Meta.ordering.\\n+        \\\"\\\"\\\"\\n+        admin_ordering = getattr(model_admin, 'ordering', None)\\n+        if admin_ordering:\\n+            return admin_ordering\\n+        other_model = get_model_from_relation(field)\\n+        return getattr(other_model._meta, 'ordering', ())\\n \\n     def __init__(self, field, request, params, model, model_admin, field_path):\\n         other_model = get_model_from_relation(field)\\n@@ -207,7 +217,7 @@ class RelatedFieldListFilter(FieldListFilter):\\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n         Return the model admin's ordering for the related model, if any.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        return self._get_related_ordering(field, model_admin)\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +429,8 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        ordering = self._get_related_ordering(field, model_admin)\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\", \"--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -207,7 +207,10 @@ class RelatedFieldListFilter(FieldListFilter):\\n \\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n-        Return the model admin's ordering for the related model, if any.\\n+        Return the model admin's ordering for the related model, or fall back\\n+        to the related model's Meta.ordering if no admin ordering is defined.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        admin_ordering = getattr(model_admin, 'ordering', None)\\n+        if admin_ordering is not None:\\n+            return admin_ordering\\n+        other_model = get_model_from_relation(field)\\n+        return getattr(other_model._meta, 'ordering', ())\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +422,8 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        ordering = self.field_admin_ordering(field, request, model_admin)\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\", \"--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -207,7 +207,15 @@ class RelatedFieldListFilter(FieldListFilter):\\n \\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n         Return the model admin's ordering for the related model, if any.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        # Check if ModelAdmin has explicit ordering\\n+        if hasattr(model_admin, 'ordering') and model_admin.ordering is not None:\\n+            return model_admin.ordering\\n+        \\n+        # Fall back to the related model's Meta.ordering\\n+        try:\\n+            other_model = get_model_from_relation(field)\\n+            return other_model._meta.ordering or ()\\n+        except (AttributeError, TypeError):\\n+            return ()\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +427,12 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        # Use the same ordering logic as the parent class\\n+        try:\\n+            ordering = self.field_admin_ordering(field, request, model_admin)\\n+        except (AttributeError, TypeError):\\n+            ordering = ()\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\", \"diff\\n--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -207,7 +207,12 @@ class RelatedFieldListFilter(FieldListFilter):\\n \\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n-        Return the model admin's ordering for the related model, if any.\\n+        Return the model admin's ordering for the related model, or fall back\\n+        to the related model's Meta.ordering if no admin ordering is defined.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        admin_ordering = getattr(model_admin, 'ordering', None)\\n+        if admin_ordering is not None:\\n+            return admin_ordering\\n+        other_model = get_model_from_relation(field)\\n+        return getattr(other_model._meta, 'ordering', ())\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +424,8 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        ordering = self.field_admin_ordering(field, request, model_admin)\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\\n\", \"diff\\n--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -187,6 +187,16 @@ class RelatedFieldListFilter(FieldListFilter):\\n     def __init__(self, field, request, params, model, model_admin, field_path):\\n         other_model = get_model_from_relation(field)\\n         if hasattr(field, 'remote_field'):\\n+    \\n+    def _get_related_ordering(self, field, model_admin):\\n+        \\\"\\\"\\\"\\n+        Get ordering for the related model, preferring ModelAdmin ordering\\n+        over model Meta.ordering.\\n+        \\\"\\\"\\\"\\n+        admin_ordering = getattr(model_admin, 'ordering', None)\\n+        if admin_ordering:\\n+            return admin_ordering\\n+        other_model = get_model_from_relation(field)\\n+        return getattr(other_model._meta, 'ordering', ())\\n \\n     def __init__(self, field, request, params, model, model_admin, field_path):\\n         other_model = get_model_from_relation(field)\\n@@ -207,7 +217,7 @@ class RelatedFieldListFilter(FieldListFilter):\\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n         Return the model admin's ordering for the related model, if any.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        return self._get_related_ordering(field, model_admin)\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +429,8 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        ordering = self._get_related_ordering(field, model_admin)\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\\n\", \"diff\\n--- a/django/contrib/admin/filters.py\\n+++ b/django/contrib/admin/filters.py\\n@@ -207,7 +207,12 @@ class RelatedFieldListFilter(FieldListFilter):\\n \\n     def field_admin_ordering(self, field, request, model_admin):\\n         \\\"\\\"\\\"\\n-        Return the model admin's ordering for the related model, if any.\\n+        Return the model admin's ordering for the related model, or fall back\\n+        to the related model's Meta.ordering if no admin ordering is defined.\\n         \\\"\\\"\\\"\\n-        return getattr(model_admin, 'ordering', ())\\n+        admin_ordering = getattr(model_admin, 'ordering', None)\\n+        if admin_ordering is not None:\\n+            return admin_ordering\\n+        other_model = get_model_from_relation(field)\\n+        return getattr(other_model._meta, 'ordering', ())\\n \\n     def choices(self, changelist):\\n         yield {\\n@@ -419,7 +424,8 @@ class RelatedOnlyFieldListFilter(RelatedFieldListFilter):\\n         super().__init__(field, request, params, model, model_admin, field_path)\\n \\n     def field_choices(self, field, request, model_admin):\\n-        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to)\\n+        ordering = self.field_admin_ordering(field, request, model_admin)\\n+        return field.get_choices(include_blank=False, limit_choices_to=self.limit_choices_to, ordering=ordering)\\n\"]}, \"error\": \"\", \"elapsed_time\": 119.206258, \"total_tokens\": 45463, \"total_steps\": 30, \"created_at\": 1754649026, \"finished_at\": 1754649145}}"
}