mirror of
https://github.com/django/django.git
synced 2025-10-24 14:16:09 +00:00
Fixed #2445 -- Allowed limit_choices_to attribute to be a callable.
ForeignKey or ManyToManyField attribute ``limit_choices_to`` can now be a callable that returns either a ``Q`` object or a dict. Thanks michael at actrix.gen.nz for the original suggestion.
This commit is contained in:
committed by
Tim Graham
parent
a718fcf201
commit
eefc88feef
@@ -240,7 +240,7 @@ class BaseModelAdmin(six.with_metaclass(RenameBaseModelAdminMethods)):
|
||||
if related_admin is not None:
|
||||
ordering = related_admin.get_ordering(request)
|
||||
if ordering is not None and ordering != ():
|
||||
return db_field.rel.to._default_manager.using(db).order_by(*ordering).complex_filter(db_field.rel.limit_choices_to)
|
||||
return db_field.rel.to._default_manager.using(db).order_by(*ordering)
|
||||
return None
|
||||
|
||||
def formfield_for_foreignkey(self, db_field, request=None, **kwargs):
|
||||
@@ -383,6 +383,9 @@ class BaseModelAdmin(six.with_metaclass(RenameBaseModelAdminMethods)):
|
||||
# ForeignKeyRawIdWidget, on the basis of ForeignKey.limit_choices_to,
|
||||
# are allowed to work.
|
||||
for l in model._meta.related_fkey_lookups:
|
||||
# As ``limit_choices_to`` can be a callable, invoke it here.
|
||||
if callable(l):
|
||||
l = l()
|
||||
for k, v in widgets.url_params_from_lookup_dict(l).items():
|
||||
if k == lookup and v == value:
|
||||
return True
|
||||
|
||||
@@ -459,17 +459,17 @@ def get_limit_choices_to_from_path(model, path):
|
||||
""" Return Q object for limiting choices if applicable.
|
||||
|
||||
If final model in path is linked via a ForeignKey or ManyToManyField which
|
||||
has a `limit_choices_to` attribute, return it as a Q object.
|
||||
has a ``limit_choices_to`` attribute, return it as a Q object.
|
||||
"""
|
||||
|
||||
fields = get_fields_from_path(model, path)
|
||||
fields = remove_trailing_data_field(fields)
|
||||
limit_choices_to = (
|
||||
get_limit_choices_to = (
|
||||
fields and hasattr(fields[-1], 'rel') and
|
||||
getattr(fields[-1].rel, 'limit_choices_to', None))
|
||||
if not limit_choices_to:
|
||||
getattr(fields[-1].rel, 'get_limit_choices_to', None))
|
||||
if not get_limit_choices_to:
|
||||
return models.Q() # empty Q
|
||||
elif isinstance(limit_choices_to, models.Q):
|
||||
limit_choices_to = get_limit_choices_to()
|
||||
if isinstance(limit_choices_to, models.Q):
|
||||
return limit_choices_to # already a Q
|
||||
else:
|
||||
return models.Q(**limit_choices_to) # convert dict to Q
|
||||
|
||||
@@ -180,7 +180,10 @@ class ForeignKeyRawIdWidget(forms.TextInput):
|
||||
return mark_safe(''.join(output))
|
||||
|
||||
def base_url_parameters(self):
|
||||
return url_params_from_lookup_dict(self.rel.limit_choices_to)
|
||||
limit_choices_to = self.rel.limit_choices_to
|
||||
if callable(limit_choices_to):
|
||||
limit_choices_to = limit_choices_to()
|
||||
return url_params_from_lookup_dict(limit_choices_to)
|
||||
|
||||
def url_parameters(self):
|
||||
from django.contrib.admin.views.main import TO_FIELD_VAR
|
||||
|
||||
@@ -742,11 +742,11 @@ class Field(RegisterLookupMixin):
|
||||
lst = [(getattr(x, self.rel.get_related_field().attname),
|
||||
smart_text(x))
|
||||
for x in rel_model._default_manager.complex_filter(
|
||||
self.rel.limit_choices_to)]
|
||||
self.get_limit_choices_to())]
|
||||
else:
|
||||
lst = [(x._get_pk_val(), smart_text(x))
|
||||
for x in rel_model._default_manager.complex_filter(
|
||||
self.rel.limit_choices_to)]
|
||||
self.get_limit_choices_to())]
|
||||
return first_choice + lst
|
||||
|
||||
def get_choices_default(self):
|
||||
|
||||
@@ -309,6 +309,35 @@ class RelatedField(Field):
|
||||
if not cls._meta.abstract:
|
||||
self.contribute_to_related_class(other, self.related)
|
||||
|
||||
def get_limit_choices_to(self):
|
||||
"""Returns 'limit_choices_to' for this model field.
|
||||
|
||||
If it is a callable, it will be invoked and the result will be
|
||||
returned.
|
||||
"""
|
||||
if callable(self.rel.limit_choices_to):
|
||||
return self.rel.limit_choices_to()
|
||||
return self.rel.limit_choices_to
|
||||
|
||||
def formfield(self, **kwargs):
|
||||
"""Passes ``limit_choices_to`` to field being constructed.
|
||||
|
||||
Only passes it if there is a type that supports related fields.
|
||||
This is a similar strategy used to pass the ``queryset`` to the field
|
||||
being constructed.
|
||||
"""
|
||||
defaults = {}
|
||||
if hasattr(self.rel, 'get_related_field'):
|
||||
# If this is a callable, do not invoke it here. Just pass
|
||||
# it in the defaults for when the form class will later be
|
||||
# instantiated.
|
||||
limit_choices_to = self.rel.limit_choices_to
|
||||
defaults.update({
|
||||
'limit_choices_to': limit_choices_to,
|
||||
})
|
||||
defaults.update(kwargs)
|
||||
return super(RelatedField, self).formfield(**defaults)
|
||||
|
||||
def related_query_name(self):
|
||||
# This method defines the name that can be used to identify this
|
||||
# related object in a table-spanning query. It uses the lower-cased
|
||||
@@ -1525,6 +1554,9 @@ class ForeignObject(RelatedField):
|
||||
# and swapped models don't get a related descriptor.
|
||||
if not self.rel.is_hidden() and not related.model._meta.swapped:
|
||||
setattr(cls, related.get_accessor_name(), self.related_accessor_class(related))
|
||||
# While 'limit_choices_to' might be a callable, simply pass
|
||||
# it along for later - this is too early because it's still
|
||||
# model load time.
|
||||
if self.rel.limit_choices_to:
|
||||
cls._meta.related_fkey_lookups.append(self.rel.limit_choices_to)
|
||||
|
||||
@@ -1633,7 +1665,7 @@ class ForeignKey(ForeignObject):
|
||||
qs = self.rel.to._default_manager.using(using).filter(
|
||||
**{self.rel.field_name: value}
|
||||
)
|
||||
qs = qs.complex_filter(self.rel.limit_choices_to)
|
||||
qs = qs.complex_filter(self.get_limit_choices_to())
|
||||
if not qs.exists():
|
||||
raise exceptions.ValidationError(
|
||||
self.error_messages['invalid'],
|
||||
@@ -1691,7 +1723,7 @@ class ForeignKey(ForeignObject):
|
||||
(self.name, self.rel.to))
|
||||
defaults = {
|
||||
'form_class': forms.ModelChoiceField,
|
||||
'queryset': self.rel.to._default_manager.using(db).complex_filter(self.rel.limit_choices_to),
|
||||
'queryset': self.rel.to._default_manager.using(db),
|
||||
'to_field_name': self.rel.field_name,
|
||||
}
|
||||
defaults.update(kwargs)
|
||||
@@ -2127,7 +2159,7 @@ class ManyToManyField(RelatedField):
|
||||
db = kwargs.pop('using', None)
|
||||
defaults = {
|
||||
'form_class': forms.ModelMultipleChoiceField,
|
||||
'queryset': self.rel.to._default_manager.using(db).complex_filter(self.rel.limit_choices_to)
|
||||
'queryset': self.rel.to._default_manager.using(db),
|
||||
}
|
||||
defaults.update(kwargs)
|
||||
# If initial is passed in, it's a list of related objects, but the
|
||||
|
||||
@@ -170,6 +170,17 @@ class Field(object):
|
||||
"""
|
||||
return {}
|
||||
|
||||
def get_limit_choices_to(self):
|
||||
"""
|
||||
Returns ``limit_choices_to`` for this form field.
|
||||
|
||||
If it is a callable, it will be invoked and the result will be
|
||||
returned.
|
||||
"""
|
||||
if callable(self.limit_choices_to):
|
||||
return self.limit_choices_to()
|
||||
return self.limit_choices_to
|
||||
|
||||
def _has_changed(self, initial, data):
|
||||
"""
|
||||
Return True if data differs from initial.
|
||||
|
||||
@@ -324,6 +324,15 @@ class BaseModelForm(BaseForm):
|
||||
self._validate_unique = False
|
||||
super(BaseModelForm, self).__init__(data, files, auto_id, prefix, object_data,
|
||||
error_class, label_suffix, empty_permitted)
|
||||
# Apply ``limit_choices_to`` to each field.
|
||||
for field_name in self.fields:
|
||||
formfield = self.fields[field_name]
|
||||
if hasattr(formfield, 'queryset'):
|
||||
limit_choices_to = formfield.limit_choices_to
|
||||
if limit_choices_to is not None:
|
||||
if callable(limit_choices_to):
|
||||
limit_choices_to = limit_choices_to()
|
||||
formfield.queryset = formfield.queryset.complex_filter(limit_choices_to)
|
||||
|
||||
def _get_validation_exclusions(self):
|
||||
"""
|
||||
@@ -1082,7 +1091,8 @@ class ModelChoiceField(ChoiceField):
|
||||
|
||||
def __init__(self, queryset, empty_label="---------", cache_choices=False,
|
||||
required=True, widget=None, label=None, initial=None,
|
||||
help_text='', to_field_name=None, *args, **kwargs):
|
||||
help_text='', to_field_name=None, limit_choices_to=None,
|
||||
*args, **kwargs):
|
||||
if required and (initial is not None):
|
||||
self.empty_label = None
|
||||
else:
|
||||
@@ -1094,6 +1104,7 @@ class ModelChoiceField(ChoiceField):
|
||||
Field.__init__(self, required, widget, label, initial, help_text,
|
||||
*args, **kwargs)
|
||||
self.queryset = queryset
|
||||
self.limit_choices_to = limit_choices_to # limit the queryset later.
|
||||
self.choice_cache = None
|
||||
self.to_field_name = to_field_name
|
||||
|
||||
|
||||
Reference in New Issue
Block a user