Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How can I list all foreign key related objects in Django admin panel?

I would like to implement a very simple feature with the django admin panel but I couldn't find the right way to achieve this so far:

model.py

class Author(models.Model):
    name = models.CharField()

class Books(models.Model):
    title = models.CharField()
    author = models.ForeignKey(Author)

admin.py

class AuthorAdmin(admin.ModelAdmin):
    pass
admin.site.register(Author, AuthorAdmin)

How can I add a hyperlink to every item (Author) in the authors' list overview that links to a view showing all books of the specific author? For Example:

  • J.K. Rowling (books)
  • J.R.R. Tolkien (books)

where books is a hyperlink to a site showing all books of the author.

like image 482
ashiaka Avatar asked Sep 30 '13 14:09

ashiaka


People also ask

What is ForeignKey relationship in Django?

Introduction to Django Foreign Key. A foreign key is a process through which the fields of one table can be used in another table flexibly. So, two different tables can be easily linked by means of the foreign key. This linking of the two tables can be easily achieved by means of foreign key processes.

Can we customize Django admin panel?

The Django admin is a powerful built-in tool giving you the ability to create, update, and delete objects in your database using a web interface. You can customize the Django admin to do almost anything you want.

How use Django admin panel?

To login to the site, open the /admin URL (e.g. http://127.0.0.1:8000/admin ) and enter your new superuser userid and password credentials (you'll be redirected to the login page, and then back to the /admin URL after you've entered your details).


1 Answers

You are looking for a ModelAdmin.list_filter.

Set list_filter to activate filters in the right sidebar of the change list page of the admin. A listfilter can be a field name, where the specified field should be either a BooleanField, CharField, DateField, DateTimeField, IntegerField, ForeignKey or ManyToManyField, for example:

 # Add a list filter author to BookAdmin.
 # Now you can filter books by author.
 class BookAdmin(ModelAdmin):
    list_filter = ('author', )

Now you can use @Wolph suggestion to add a link in the Author list_display. This link points to the book list filtered by author:

# Add hyperlinks to AuthorAdmin.
# Now you can jump to the book list filtered by autor. 
class AuthorAdmin(admin.ModelAdmin):
    def authors(self):
        return '<a href="/admin/appname/book/?author__id__exact=%d">%s</a>' % (self.author_id, self.author)
    authors.allow_tags = True

ALTERNATIVE. To save a click you can also point to the change view of a book directly:

class Books(models.Model):
    title = models.CharField()
    author = models.ForeignKey(Author)

def get_admin_url(self):
    return "/admin/appname/books/%d/" %self.id


class BookAdmin(admin.ModelAdmin):
    def authors(self):
        html = ""
        for obj in Books.objects.filter(author__id_exact=self.id):
            html += '<p><a href="%s">%s</a></p>' %(obj.get_admin_url(), obj.title)
        return html
    authors.allow_tags = True

    list_display = ['title', authors]

Disclaimer: Not tested, but if you fix the typo's it'll work! :)

Note that these links can also be injected at other points in the admin. When you add it to a widget, you can go from change view to change view.

like image 113
allcaps Avatar answered Sep 29 '22 22:09

allcaps