I am trying to build a URL pattern such as /category/article/
but the problem is, one article can be related to many categories, so the URL should be something like /category+category/article/
What I have:
models.py
from django.db import models
from django.core.urlresolvers import reverse
class Category(models.Model):
name = models.CharField(max_length=128, unique=True)
slug = models.SlugField(unique=True)
def __unicode__(self):
return self.name
def get_absolute_url(self):
return reverse('view_category', kwargs={ 'slug': self.slug })
class Meta:
verbose_name_plural = "Categories"
class Work(models.Model):
category = models.ManyToManyField(Category)
title = models.CharField(max_length=128)
slug = models.SlugField(unique=True)
def __unicode__(self):
return self.title
def get_absolute_url(self):
return reverse('view_work', kwargs={'slug': self.slug, 'category': self.category })
views.py
from django.shortcuts import render_to_response, get_object_or_404
from app.models import Work, Category
def index(request):
return render_to_response('app/index.html', {
'works': Work.objects.all()[:5]
})
def all_works(request):
return render_to_response('app/all_works.html', {
'categories': Category.objects.all(),
'works': Work.objects.all()[:5]
})
def view_category(request, slug):
category = get_object_or_404(Category, slug=slug)
return render_to_response('app/categories.html', {
'categories': Category.objects.all(),
'category': category,
'works': Work.objects.filter(category=category)[:5]
})
def view_work(request, slug, category):
return render_to_response('app/work.html', {
'work': get_object_or_404(Work, slug=slug, category=category)
})
urls.py
from django.conf.urls import patterns, url
from app import views
urlpatterns = patterns('',
url(r'^works/$', views.all_works, name='view_all_works'),
url(r'^works/(?P<slug>[-\w]+)/$', views.view_category, name='view_category'),
url(r'^work/(?P<category>[-\w]+)/(?P<slug>[-\w]+)/$', views.view_work, name='view_work'),
url(r'^$', views.index, name='index'),
)
in template:
<a href="{{ work.get_absolute_url|slice:'6:' }}">{{ work.title }}</a>
And currently I get an error:
KeyError at /
u'category'
Request Method: GET
Request URL: http://127.0.0.1:8000/
Django Version: 1.7.2
Exception Type: KeyError
Exception Value:
u'category'
Exception Location: E:\web\env\lib\site-packages\django\core\urlresolvers.py in _reverse_with_prefix, line 448
11 <li>
12 <a href="{{ work.get_absolute_url|slice:'6:' }}" >{{ work.title }}</a>
13 </li>
Could anyone help me with this?
I think this is because Work.category
is a ManyToMany
field. So in your Work.get_absolute_url
function you are trying to get self.category
which is actually returning django.db.models.fields.related.ManyRelatedManager
(which as a result will return a list of Category
's).
You could probably fix this by changing your absolute_url_function to something like:
def get_absolute_url(self):
category = self.category.all()[0]
return return reverse('view_work', kwargs={'slug': self.slug, 'category': category })