通过__unicode()__方法的输出命令Django管理员更改列表列

时间:2009-09-10 17:52:38

标签: django-admin sorting

这是我的Django应用程序 models.py 的一部分:

class Person(models.Model):
    birth_year = WideYear(null=True, blank=True)
    birth_year_uncertain = models.BooleanField()
    death_year = WideYear(null=True, blank=True)
    death_year_uncertain = models.BooleanField()
    flourit_year = WideYear(null=True, blank=True)
    flourit_year_uncertain = models.BooleanField()
    FLOURIT_CHOICES = (
        (u'D', u'Birth and death dates'),
        (u'F', u'Flourit date'),
    )
    use_flourit = models.CharField('Date(s) to use', max_length=2, choices=FLOURIT_CHOICES)
    index_entries = models.ManyToManyField(IndexEntry, null=True, blank=True)
    def __unicode__(self):
        if self.personname_set.filter(default_name__exact=True):
            name = z(self.personname_set.filter(default_name__exact=True)[0])
        else:
            name = u'[Unnamed person]'
        if self.use_flourit == u'D':
            dates = '%s - %s' % (z(self.birth_year), z(self.death_year))
        else:
            dates = 'fl. ' + z(self.flourit_year)
        return '%s (%s)' % (name, dates)

class PersonName(models.Model):
    titles = models.CharField(max_length=65535, null=True, blank=True)
    surname = models.CharField(max_length=255, null=True, blank=True)
    first_name = models.CharField(max_length=255, null=True, blank=True)
    middle_names = models.CharField(max_length=255, null=True, blank=True)
    post_nominals = models.CharField(max_length=65535, null=True, blank=True)
    default_name = models.BooleanField()
    person = models.ForeignKey(Person, null=True, blank=True)
    def __unicode__(self):
        return '%s, %s %s' % (self.surname, self.first_name, self.middle_names)
    class Meta:
        unique_together = ("titles", "surname", "first_name", "middle_names", "post_nominals", "person")
        unique_together = ("default_name", "person")

以下是我的应用admin.py的相应部分:

from reversion.admin import VersionAdmin

class PersonNameInline(admin.TabularInline):
    model = PersonName
    extra = 1

class PersonAdmin(VersionAdmin):
    radio_fields = {"use_flourit": admin.HORIZONTAL}
    inlines = [PersonNameInline]

admin.site.register(Person, PersonAdmin)

在管理员中,这会生成一个更改列表,如下所示:

screencap
(来源:sampablokuper.com

正如您所看到的,虽然更改列表使用 Person 类的__unicode()__方法的输出填充 Person列的每一行,但它确实不订单 类的__unicode()__方法中该列的行。

我该如何做到?

非常感谢提前!

3 个答案:

答案 0 :(得分:2)

Django排序在数据库级别完成。除非您将unicode函数的结果存储在数据库中,否则django将无法原生地返回以这种方式排序的结果。

在DB中存储排序值可能是解决此问题的最有效方法。

答案 1 :(得分:1)

刚刚遇到同样的问题,我发现以下链接很有用。它不是按unicode排序,而是尝试按多列排序,这可能有助于解决问题:

http://djangosnippets.org/snippets/2110/

答案 2 :(得分:0)

关注Paul McMillan's suggestion后,我将以下行添加到的类定义中:

ordering_string = models.CharField(max_length=255, null=True, blank=True)

我还把它放在models.py:

中的 PersonName 定义之上
def post_save_person_and_person_name(sender, **kwargs):
    person_name = kwargs['instance']
    person = person_name.person
    if person.ordering_string != unicode(person)[0:254]:
        person.ordering_string = unicode(person)[0:254]
        super(Person, person).save()

并将其放在models.py:

中的 PersonName 定义下方
post_save.connect(post_save_person_and_person_name, sender=PersonName)

到目前为止,非常好。

我想我可以通过用queryset update()替换上面的save()调用来改进它。我欢迎有关这方面的建议!

相关问题