228

I am trying to create a SlugField in Django.

I created this simple model:

from django.db import models

class Test(models.Model):
    q = models.CharField(max_length=30)
    s = models.SlugField()

I then do this:

>>> from mysite.books.models import Test
>>> t=Test(q="aa a a a", s="b b b b")
>>> t.s
'b b b b'
>>> t.save()
>>> t.s
'b b b b'

I was expecting b-b-b-b.

daaawx
  • 2,268
  • 2
  • 12
  • 13
Johnd
  • 5,433
  • 9
  • 26
  • 22

9 Answers9

433

You will need to use the slugify function.

>>> from django.template.defaultfilters import slugify
>>> slugify("b b b b")
u'b-b-b-b'
>>>

You can call slugify automatically by overriding the save method:

class Test(models.Model):
    q = models.CharField(max_length=30)
    s = models.SlugField()
    
    def save(self, *args, **kwargs):
        self.s = slugify(self.q)
        super(Test, self).save(*args, **kwargs)

Be aware that the above will cause your URL to change when the q field is edited, which can cause broken links. It may be preferable to generate the slug only once when you create a new object:

class Test(models.Model):
    q = models.CharField(max_length=30)
    s = models.SlugField()
    
    def save(self, *args, **kwargs):
        if not self.id:
            # Newly created object, so set slug
            self.s = slugify(self.q)

        super(Test, self).save(*args, **kwargs)
NaturalBornCamper
  • 3,264
  • 5
  • 30
  • 49
Buddy
  • 6,345
  • 1
  • 19
  • 16
  • 4
    shy have a special model type? why not just slugify CharFields? – Johnd May 08 '09 at 03:31
  • 23
    SlugFields set db_index=True by default, and also use a form field by default that has a validation regex to require valid slugs (if represented in a ModelForm or in the admin). You can do those things manually with a CharField if you prefer, it just makes the intention of your code less clear. Also, don't forget the prepopulate_fields ModelAdmin setting, if you want JS-based auto-prepopulate in the admin. – Carl Meyer May 08 '09 at 14:22
  • 4
    As Dingle said below in his answer, you'll need to replace `def save(self):` with `def save(self, *args, **kwargs):` in order to avoid errors from being thrown when writing something like `test.objects.create(q="blah blah blah")`. – Liam Mar 24 '10 at 17:11
  • 6
    Beware that this code will update the slug each saves. your url will change, and "Cool URIs don't change" http://www.w3.org/Provider/Style/URI.html – dzen Jun 03 '11 at 09:37
  • Why not make the slug optional and override `save()` to create slug automatically from title if current slug is empty? `if self.s == '': self.s = slugify(self.q)` – arifwn Dec 10 '11 at 14:32
  • 19
    `slugify()` can also be found in `django.utils.text.slugify`, not clear when this was added. – mrmagooey Feb 10 '13 at 23:54
  • 2
    @mrmagooey it's new in 1.5 ...in 1.4.5 the slugify method is still with the template filters – Anentropic Feb 27 '13 at 12:57
111

There is corner case with some utf-8 characters

Example:

>>> from django.template.defaultfilters import slugify
>>> slugify(u"test ąęśćółń")
u'test-aescon' # there is no "l"

This can be solved with Unidecode

>>> from unidecode import unidecode
>>> from django.template.defaultfilters import slugify
>>> slugify(unidecode(u"test ąęśćółń"))
u'test-aescoln'
DooBLER
  • 1,160
  • 1
  • 7
  • 8
  • 7
    utf-8 is now handled correctly by slugify (in django 1.8.5) – Rick Westera Nov 20 '15 at 23:53
  • As @RickWestera said this is now handled by slugify, although if for some reason you don't want to use slugify, check iri_to_uri from django.utils.encoding: https://docs.djangoproject.com/en/2.0/ref/unicode/#taking-care-in-get-absolute-url – Erwol Mar 24 '18 at 05:53
64

A small correction to Thepeer's answer: To override save() function in model classes, better add arguments to it:

from django.utils.text import slugify

def save(self, *args, **kwargs):
    if not self.id:
        self.s = slugify(self.q)

    super(test, self).save(*args, **kwargs)

Otherwise, test.objects.create(q="blah blah blah") will result in a force_insert error (unexpected argument).

Jonas Gröger
  • 1,347
  • 2
  • 19
  • 32
Dingle
  • 2,192
  • 17
  • 12
  • 2
    One further very minor thing to add to thepeer's answer: I would make that last line `return super(test, self).save(*args, **kwargs)`. I think this method returns `None`, and I don't know of any plans to change that, but it does no harm to return what the superclass's method does in case it changes sometime in the future. – Duncan Parkes Aug 23 '11 at 21:16
  • Please add that *from django.utils.text import slugify* is required for this solution. – Routhinator May 07 '16 at 22:47
  • 1
    @Routhinator did it – Jonas Gröger Aug 19 '16 at 17:33
  • Putting out some feelers to ask if this is still a preferred method for doing this. – sytech Jan 17 '18 at 15:17
31

If you're using the admin interface to add new items of your model, you can set up a ModelAdmin in your admin.py and utilize prepopulated_fields to automate entering of a slug:

class ClientAdmin(admin.ModelAdmin):
    prepopulated_fields = {'slug': ('name',)}

admin.site.register(Client, ClientAdmin)

Here, when the user enters a value in the admin form for the name field, the slug will be automatically populated with the correct slugified name.

Yaroslav Admin
  • 11,842
  • 6
  • 54
  • 69
henrym
  • 2,545
  • 1
  • 16
  • 13
  • My `slug` and `name` fields have translations. How can I do that with translations? Because I've tried to add `'slug_en':('name_en',)` and got the error that attribute doesn't exist in my model. – patricia Jul 21 '16 at 12:17
22

In most cases the slug should not change, so you really only want to calculate it on first save:

class Test(models.Model):
    q = models.CharField(max_length=30)
    s = models.SlugField(editable=False) # hide from admin

    def save(self):
        if not self.id:
            self.s = slugify(self.q)

        super(Test, self).save()
spiffytech
  • 5,185
  • 5
  • 36
  • 51
thepeer
  • 7,981
  • 2
  • 16
  • 13
6

Use prepopulated_fields in your admin class:

class ArticleAdmin(admin.ModelAdmin):
    prepopulated_fields = {"slug": ("title",)}

admin.site.register(Article, ArticleAdmin)
daaawx
  • 2,268
  • 2
  • 12
  • 13
sergey
  • 61
  • 1
  • 1
5

If you don't want to set the slugfield to Not be editable, then I believe you'll want to set the Null and Blank properties to False. Otherwise you'll get an error when trying to save in Admin.

So a modification to the above example would be::

class test(models.Model):
    q = models.CharField(max_length=30)
    s = models.SlugField(null=True, blank=True) # Allow blank submission in admin.

    def save(self):
        if not self.id:
            self.s = slugify(self.q)

        super(test, self).save()
Streamweaver
  • 337
  • 2
  • 8
4

I'm using Django 1.7

Create a SlugField in your model like this:

slug = models.SlugField()

Then in admin.py define prepopulated_fields;

class ArticleAdmin(admin.ModelAdmin):
    prepopulated_fields = {"slug": ("title",)}
markwalker_
  • 9,401
  • 7
  • 51
  • 86
min2bro
  • 3,776
  • 3
  • 20
  • 48
0

You can look at the docs for the SlugField to get to know more about it in more descriptive way.

Ralf
  • 13,322
  • 4
  • 31
  • 55
Sonia Rani
  • 436
  • 5
  • 3