Evgenii Legotckoi
26 марта 2017 г. 11:45

Django - Урок 021. Наследование моделей, абстрактная модель

Содержание

После проведения рефакторинга на сайте, было выделено четыре основных сущности, у которых были выделены общие свойства, а именно:

  • Article - Статьи
  • Comment - Комментарии
  • ForumTopic - Темы форума (они же вопросы)
  • ForumPost - Ответы к темам форума

Конечно, и так было ясно, что эти сущности могут иметь одинаковые поля данных, одинаковые методы и т.д. Но при разработке данного сайта я и сам одновременно изучаю Python и Django. Поэтому проект носит характер хаотичного внесение небольших ToDo с последующим рефакторингом при изучении лучших подходов. Поэтому после изучения возможностей наследования моделей в Django, была выделена одна общая абстрактная модель данных PostBase , которая имеет четыре поля, которые повторяются во всех выше перечиcленных моделях.

Здесь есть один важный момент: Модель, которая объявляется абстрактной, не будет создавать таблицу в базе данных.

Чтобы создать абстрактную модель необходимо установить переменную abstract в значение True для класса Meta.


PostBase

PostBase - это будет базовая абстрактная модель. В этой абстрактной модели данных было выделено четыре поля, которые являются общими для выше перечисленных моделей:

  • author - автор статьи, комментария, темы или ответа;
  • content - содержание;
  • pub_date - дата публикации;
  • moderation - модерация, весь контент может иметь четыре возможных варианта:
  • SPAM - без комментариев;
  • NOT_MODERATED - Не проверенная запись, в данном случае статьи пользователей с данным статусом не будут доступны другим пользователя до модерации;
  • POST_MODERATED - модерация после публикации, статья пользователя будет доступна другим пользователям после публикации, но она ещё не прошла модерацию;
  • MODERATED - запись прошла модерацию.

Объявление данной абстрактной модели будет следующее:

  1. # -*- coding: utf-8 -*-
  2.  
  3. from django.db import models
  4. from django.contrib.auth.models import User
  5. from django.utils.translation import ugettext_lazy as _
  6.  
  7. from ckeditor_uploader.fields import RichTextUploadingField
  8.  
  9. class PostBase(models.Model):
  10. class Meta:
  11. abstract = True # данное поле указывает, что класс абстрактный
  12.   # и что для него не нужно создавать таблицу
  13.  
  14. SPAM = 'S'
  15. NOT_MODERATED = 'N'
  16. POST_MODERATED = 'P'
  17. MODERATED = 'M'
  18. MODERATION_CHOICES = (
  19. (SPAM, 'SPAM'),
  20. (NOT_MODERATED, 'Not Moderated'),
  21. (POST_MODERATED, 'Post Moderated'),
  22. (MODERATED, 'Moderated')
  23. )
  24.  
  25. author = models.ForeignKey(User, verbose_name=_("Автор"))
  26. content = models.TextField(_('Содержание'), blank=True)
  27. pub_date = models.DateTimeField(_('Дата публикации'), blank=True, null=True)
  28. moderation = models.CharField(
  29. _('Модерация'),
  30. max_length=1,
  31. choices=MODERATION_CHOICES,
  32. default=NOT_MODERATED
  33. )

Таким образом имеется возможность сократить программный код проекта и добавить возможность повторного использования повторяющегося кода.

Структура модели, например, для комментариев теперь может выглядеть следующим образом:

  1. class Comment(PostBase):
  2. class Meta:
  3. db_table = "comments"
  4.  
  5. article = models.ForeignKey(Article)

Поля author , pub_date , content и moderation уже не требуется указывать, поскольку они присутствуют в PostBase классе. Главное, не вводите в свою модель поля с такими же именами, как в модели PostBase.

PostBaseAdmin

Также несомненным плюсом является то, что можно точно также сделать одну общую для всех классов настройку админ панели.

Сделаем, например, отображение полей, настройку поиска и фильтрации контента и возможность установки статуса модерации (то есть добавим соответствующие actions ).

  1. class PostBaseAdmin(admin.ModelAdmin):
  2. list_display = ('content', 'author', 'pub_date')
  3. search_fields = ('content', 'author__username')
  4. list_filter = ('moderation',)
  5. actions = ['make_spam', 'make_not_moderated', 'make_post_moderated', 'make_moderated']
  6.  
  7. def moderate(self, request, rows_updated, choice_description):
  8. if rows_updated == 1:
  9. message_bit = "1 запись помечена, как %s" % choice_description
  10. else:
  11. message_bit = "%s записей отмечены, как %s." % (rows_updated, choice_description)
  12. self.message_user(request, "%s" % message_bit)
  13.  
  14. def make_spam(self, request, queryset):
  15. self.moderate(
  16. request=request,
  17. rows_updated=queryset.update(moderation=PostBase.SPAM),
  18. choice_description="SPAM"
  19. )
  20. make_spam.short_description = "Отметить помеченные, как SPAM"
  21.  
  22. def make_not_moderated(self, request, queryset):
  23. self.moderate(
  24. request=request,
  25. rows_updated=queryset.update(moderation=PostBase.NOT_MODERATED),
  26. choice_description="NOT_MODERATED"
  27. )
  28. make_not_moderated.short_description = "Отметить помеченные, как NOT_MODERATED"
  29.  
  30. def make_post_moderated(self, request, queryset):
  31. self.moderate(
  32. request=request,
  33. rows_updated=queryset.update(moderation=PostBase.POST_MODERATED),
  34. choice_description="POST_MODERATED"
  35. )
  36. make_post_moderated.short_description = "Отметить помеченные, как POST_MODERATED"
  37.  
  38. def make_moderated(self, request, queryset):
  39. self.moderate(
  40. request=request,
  41. rows_updated=queryset.update(moderation=PostBase.MODERATED),
  42. choice_description="MODERATED"
  43. )
  44. make_moderated.short_description = "Отметить помеченные, как MODERATED"

Если вы захотите расширить листы фильтрации или добавить actions, для какой-то из моделей, то можно наследоваться от PostBaseAdmin следующим образом:

  1. class ArticleAdmin(PostBaseAdmin):
  2. # Либо полностью переопределить отображаемые поля или поля для поиска
  3. list_display = ('title', 'section', 'author', 'pub_date', 'views', 'moderation')
  4. search_fields = ('title', 'author__username', 'section__title')
  5. # Либо добавить к существующему списку дополнительные
  6. list_filter = PostBaseAdmin.list_filter + ('status', 'section')
  7. # В случае actions подобное объявление просто добавит новые actions к существующим
  8. actions = ['publish', 'unpublish']

Для Django рекомендую VDS-сервера хостера Timeweb .

Вам это нравится? Поделитесь в социальных сетях!

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь