Fixed #11058 - list_display_links doesn't allow callables not defined in the model
Thanks to dvine for the report and julien for the patch. git-svn-id: http://code.djangoproject.com/svn/django/trunk@15619 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
1b062f6613
commit
fdf9602961
|
@ -46,9 +46,8 @@ def validate(cls, model):
|
||||||
if hasattr(cls, 'list_display_links'):
|
if hasattr(cls, 'list_display_links'):
|
||||||
check_isseq(cls, 'list_display_links', cls.list_display_links)
|
check_isseq(cls, 'list_display_links', cls.list_display_links)
|
||||||
for idx, field in enumerate(cls.list_display_links):
|
for idx, field in enumerate(cls.list_display_links):
|
||||||
fetch_attr(cls, model, opts, 'list_display_links[%d]' % idx, field)
|
|
||||||
if field not in cls.list_display:
|
if field not in cls.list_display:
|
||||||
raise ImproperlyConfigured("'%s.list_display_links[%d]'"
|
raise ImproperlyConfigured("'%s.list_display_links[%d]' "
|
||||||
"refers to '%s' which is not defined in 'list_display'."
|
"refers to '%s' which is not defined in 'list_display'."
|
||||||
% (cls.__name__, idx, field))
|
% (cls.__name__, idx, field))
|
||||||
|
|
||||||
|
|
|
@ -479,11 +479,11 @@ subclass::
|
||||||
By default, the change list page will link the first column -- the first
|
By default, the change list page will link the first column -- the first
|
||||||
field specified in ``list_display`` -- to the change page for each item.
|
field specified in ``list_display`` -- to the change page for each item.
|
||||||
But ``list_display_links`` lets you change which columns are linked. Set
|
But ``list_display_links`` lets you change which columns are linked. Set
|
||||||
``list_display_links`` to a list or tuple of field names (in the same
|
``list_display_links`` to a list or tuple of fields (in the same
|
||||||
format as ``list_display``) to link.
|
format as ``list_display``) to link.
|
||||||
|
|
||||||
``list_display_links`` can specify one or many field names. As long as the
|
``list_display_links`` can specify one or many fields. As long as the
|
||||||
field names appear in ``list_display``, Django doesn't care how many (or
|
fields appear in ``list_display``, Django doesn't care how many (or
|
||||||
how few) fields are linked. The only requirement is: If you want to use
|
how few) fields are linked. The only requirement is: If you want to use
|
||||||
``list_display_links``, you must define ``list_display``.
|
``list_display_links``, you must define ``list_display``.
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@ class Band(models.Model):
|
||||||
name = models.CharField(max_length=100)
|
name = models.CharField(max_length=100)
|
||||||
bio = models.TextField()
|
bio = models.TextField()
|
||||||
sign_date = models.DateField()
|
sign_date = models.DateField()
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
|
@ -32,5 +32,8 @@ class ValidationTestModel(models.Model):
|
||||||
pub_date = models.DateTimeField()
|
pub_date = models.DateTimeField()
|
||||||
band = models.ForeignKey(Band)
|
band = models.ForeignKey(Band)
|
||||||
|
|
||||||
|
def decade_published_in(self):
|
||||||
|
return self.pub_date.strftime('%Y')[:3] + "0's"
|
||||||
|
|
||||||
class ValidationTestInlineModel(models.Model):
|
class ValidationTestInlineModel(models.Model):
|
||||||
parent = models.ForeignKey(ValidationTestModel)
|
parent = models.ForeignKey(ValidationTestModel)
|
||||||
|
|
|
@ -771,8 +771,13 @@ class ValidationTests(unittest.TestCase):
|
||||||
ValidationTestModel,
|
ValidationTestModel,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def a_callable(obj):
|
||||||
|
pass
|
||||||
|
|
||||||
class ValidationTestModelAdmin(ModelAdmin):
|
class ValidationTestModelAdmin(ModelAdmin):
|
||||||
list_display = ('name',)
|
def a_method(self, obj):
|
||||||
|
pass
|
||||||
|
list_display = ('name', 'decade_published_in', 'a_method', a_callable)
|
||||||
|
|
||||||
validate(ValidationTestModelAdmin, ValidationTestModel)
|
validate(ValidationTestModelAdmin, ValidationTestModel)
|
||||||
|
|
||||||
|
@ -794,7 +799,7 @@ class ValidationTests(unittest.TestCase):
|
||||||
|
|
||||||
self.assertRaisesRegexp(
|
self.assertRaisesRegexp(
|
||||||
ImproperlyConfigured,
|
ImproperlyConfigured,
|
||||||
"'ValidationTestModelAdmin.list_display_links\[0\]' refers to 'non_existent_field' that is neither a field, method or property of model 'ValidationTestModel'.",
|
"'ValidationTestModelAdmin.list_display_links\[0\]' refers to 'non_existent_field' which is not defined in 'list_display'.",
|
||||||
validate,
|
validate,
|
||||||
ValidationTestModelAdmin,
|
ValidationTestModelAdmin,
|
||||||
ValidationTestModel,
|
ValidationTestModel,
|
||||||
|
@ -805,15 +810,20 @@ class ValidationTests(unittest.TestCase):
|
||||||
|
|
||||||
self.assertRaisesRegexp(
|
self.assertRaisesRegexp(
|
||||||
ImproperlyConfigured,
|
ImproperlyConfigured,
|
||||||
"'ValidationTestModelAdmin.list_display_links\[0\]'refers to 'name' which is not defined in 'list_display'.",
|
"'ValidationTestModelAdmin.list_display_links\[0\]' refers to 'name' which is not defined in 'list_display'.",
|
||||||
validate,
|
validate,
|
||||||
ValidationTestModelAdmin,
|
ValidationTestModelAdmin,
|
||||||
ValidationTestModel,
|
ValidationTestModel,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def a_callable(obj):
|
||||||
|
pass
|
||||||
|
|
||||||
class ValidationTestModelAdmin(ModelAdmin):
|
class ValidationTestModelAdmin(ModelAdmin):
|
||||||
list_display = ('name',)
|
def a_method(self, obj):
|
||||||
list_display_links = ('name',)
|
pass
|
||||||
|
list_display = ('name', 'decade_published_in', 'a_method', a_callable)
|
||||||
|
list_display_links = ('name', 'decade_published_in', 'a_method', a_callable)
|
||||||
|
|
||||||
validate(ValidationTestModelAdmin, ValidationTestModel)
|
validate(ValidationTestModelAdmin, ValidationTestModel)
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue