99

I have a venue, this venue has many events happening there. My models look like this:

class Event(models.Model):
    title = models.CharField(max_length=200)
    date_published = models.DateTimeField('published date',default=datetime.now, blank=True)
    date_start = models.DateTimeField('start date')
    date_end = models.DateTimeField('end date')
    def __unicode__(self):
        return self.title
    description = models.TextField()
    price = models.IntegerField(null=True, blank=True)
    venue = models.ForeignKey(Venue)

class Venue(models.Model):
    title = models.CharField(max_length=200)
    date_published = models.DateTimeField('published date',default=datetime.now, blank=True)
    venue_latitude = models.CharField(max_length=200)
    venue_longitude = models.CharField(max_length=200)
    venue_address = models.CharField(max_length=200)
    venue_city = models.CharField(max_length=200)
    venue_state = models.CharField(max_length=200)
    venue_country = models.CharField(max_length=200)
    description = models.TextField()
    def __unicode__(self):
        return u'%s' % (self.title)

I'd like to display all the events that are happening at a certain venue. How can I do that? My current view looks like:

def detail(request, venue_id):
    venue = get_object_or_404(Venue, pk=venue_id)
    return render(request, 'venue-detail.html', {'venue': venue})
FLX
  • 4,446
  • 11
  • 44
  • 60

3 Answers3

157

You can use events = venue.event_set to go the other way.

Note that venue.event_set is a manager object, like Event.objects, so you can call .all, .filter, .exclude and similar on it to get a queryset.

See the Django documentation

Fifi
  • 439
  • 5
  • 16
Ric
  • 8,290
  • 3
  • 16
  • 21
  • Great! THanks for that. However, If I want to iterate over the events (for i in events) I'm getting "'RelatedManager' object is not iterable" – FLX Mar 09 '13 at 04:53
  • How did you change it to QueryObject ? – mp3por Feb 06 '16 at 13:27
  • 7
    @mp3por venue.event_set.all() or venue.event_set.filter(), usually. – aptwebapps Mar 05 '16 at 09:26
  • 5
    If anyone stills sees the error `AttributeError: 'Foo' object has no attribute 'bar_set'`, you might want to check if `related_name` is set on the `ForeignKey`/`OneToOneField`/`ManyToManyField` of the child model (`Bar`) referencing the parent model (`Foo`). – Bartleby May 24 '20 at 12:54
  • Will this lookup be fast ? should he add foregin key to Venue ? will it make it faster ? – Sion C May 06 '21 at 09:10
10

To those who have "'RelatedManager' object is not iterable"

Add all to retrieve the elements from the manager.

{% for area in world_areas.all %}

https://stackoverflow.com/a/16909142/2491526 (cannot add this in comment to the first answer)

Community
  • 1
  • 1
Apex
  • 363
  • 1
  • 4
  • 16
5

Go the other way round. Use Event model.

def detail(request, venue_id):
    venue = Event.objects.filter(venue__id=venue_id)
    return render(request, 'venue-detail.html', {'venue': venue})

PS: I have never used get_object_or_404(). Modify code accordingly.

alecxe
  • 441,113
  • 110
  • 1,021
  • 1,148
rjv
  • 5,499
  • 4
  • 28
  • 48
  • So what can I use in my template to display the venue title and all the events happening at this venue? – FLX Mar 09 '13 at 03:53
  • 1
    in the template use {{ event.venue.title }} or as per your variables {{ venue.venue.title }} :) `{% for i in venue %} {{ i.venue.title }}` – rjv Mar 09 '13 at 03:55
  • I get that, but how do I display all the events happening at this venue? – FLX Mar 09 '13 at 04:16
  • the filter returns an array containing all the events that happen at the venue.loop over the array in the templates using the `for` – rjv Mar 09 '13 at 04:54
  • 2
    `venue = Event.objects.filter(venue__id=venue_id)` I think this expression will return QuerySet of Event object instead of Venue object. – smilingwang Sep 11 '18 at 03:23
  • @smilingwang read the question being asked. – boatcoder Feb 27 '22 at 00:30