我正在尝试发布使用slug作为post url的帖子。我添加slug字段但它不是自动填充。我想让它自动填充。到目前为止,我已经这样做了。我在最后添加了pre_save。
我试图保存来自django admin的帖子,它说的是这样的
这是必填栏。
文章/ models.py
from django.db import models
from django.core.validators import FileExtensionValidator
from django.db.models.signals import pre_save
from django.utils.text import slugify
# Create your models here.
class Category(models.Model):
title = models.CharField(max_length = 120, verbose_name="Title" )
updated_at = models.DateTimeField(auto_now_add=True, verbose_name="Updated at")
created_at = models.DateTimeField(auto_now_add=True, verbose_name="Created at")
class Meta:
verbose_name = "Category"
verbose_name_plural = "Categories"
ordering = ['title']
def __str__(self):
return self.title
class Posts(models.Model):
title = models.CharField(max_length=60)
slug = models.SlugField(unique = True)
file_upload = models.FileField(null= True, blank=True, validators=[FileExtensionValidator(['pdf'])])
content = models.TextField()
category = models.ForeignKey(Category, null= True,verbose_name="Category", on_delete=models.CASCADE)
updated = models.DateTimeField(auto_now=True, auto_now_add=False)
timestamp = models.DateTimeField(auto_now=False, auto_now_add=True)
# class Meta:
# verbose_name = "Post"
# verbose_name_plural = "Posts"
# ordering = ['-created_at']
def __unicode__(self):
return self.title
def __str__(self):
return self.title
def create_slug(instance, new_slug=None):
slug = slugify(instance.title, allow_unicode = True)
if new_slug is not None:
slug = new_slug
qs = Posts.objects.filter(slug=slug).order_by("-id")
exists = qs.exists()
if exists:
new_slug = "%s-%s"%(slug, qs.first().id)
return create_slug(instance, new_slug=new_slug)
return slug
def pre_save_post_receiver( instance, sender,*args, **kwargs):
if not instance.slug:
instance.slug = create_slug(instance)
pre_save.connect(pre_save_post_receiver, sender=Posts)
从您的代码开始,instance.title
始终对生成slug值很重要。然后,一种不触发pre_save信号的可能方法是覆盖模型保存功能并在那里分配您的值。
def create_slug(title, new_slug=None):
slug = slugify(title, allow_unicode = True)
if new_slug is not None:
slug = new_slug
qs = Posts.objects.filter(slug=slug).order_by("-id")
exists = qs.exists()
if exists:
new_slug = "%s-%s"%(slug, qs.first().id)
return create_slug(title, new_slug=new_slug)
return slug
class Posts(models.Model):
title = models.CharField(max_length=60)
slug = models.SlugField(unique = True)
file_upload = models.FileField(null= True, blank=True, validators=[FileExtensionValidator(['pdf'])])
content = models.TextField()
category = models.ForeignKey(Category, null= True,verbose_name="Category", on_delete=models.CASCADE)
updated = models.DateTimeField(auto_now=True, auto_now_add=False)
timestamp = models.DateTimeField(auto_now=False, auto_now_add=True)
# class Meta:
# verbose_name = "Post"
# verbose_name_plural = "Posts"
# ordering = ['-created_at']
def save(self, *args, **kwargs):
if not self.slug:
self.slug = create_slug(self.title)
return super(Posts, self).save(*args, **kwargs) # important to call this
def __unicode__(self):
return self.title
def __str__(self):
return self.title
我真的希望这个解决方案能够奏效。使用字段默认可调用函数link的另一种可能方法,但是从你的代码我可以看到title
也很重要。我不完全确定但是跟随的东西也可以工作。
def create_slug(title, new_slug=None):
slug = slugify(title, allow_unicode = True)
if new_slug is not None:
slug = new_slug
qs = Posts.objects.filter(slug=slug).order_by("-id")
exists = qs.exists()
if exists:
new_slug = "%s-%s"%(slug, qs.first().id)
return create_slug(title, new_slug=new_slug)
return slug
class Posts(models.Model):
title = models.CharField(max_length=60)
slug = models.SlugField(unique = True, default=create_slug(self.title))
file_upload = models.FileField(null= True, blank=True, validators=[FileExtensionValidator(['pdf'])])
content = models.TextField()
category = models.ForeignKey(Category, null= True,verbose_name="Category", on_delete=models.CASCADE)
updated = models.DateTimeField(auto_now=True, auto_now_add=False)
timestamp = models.DateTimeField(auto_now=False, auto_now_add=True)
# class Meta:
# verbose_name = "Post"
# verbose_name_plural = "Posts"
# ordering = ['-created_at']
def __unicode__(self):
return self.title
def __str__(self):
return self.title