- 1. PostBase
- 2. PostBaseAdmin
После проведения рефакторинга на сайте, было выделено четыре основных сущности, у которых были выделены общие свойства, а именно:
- 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 - запись прошла модерацию.
Объявление данной абстрактной модели будет следующее:
- # -*- coding: utf-8 -*-
- from django.db import models
- from django.contrib.auth.models import User
- from django.utils.translation import ugettext_lazy as _
- from ckeditor_uploader.fields import RichTextUploadingField
- class PostBase(models.Model):
- class Meta:
- abstract = True # данное поле указывает, что класс абстрактный
- # и что для него не нужно создавать таблицу
- SPAM = 'S'
- NOT_MODERATED = 'N'
- POST_MODERATED = 'P'
- MODERATED = 'M'
- MODERATION_CHOICES = (
- (SPAM, 'SPAM'),
- (NOT_MODERATED, 'Not Moderated'),
- (POST_MODERATED, 'Post Moderated'),
- (MODERATED, 'Moderated')
- )
- author = models.ForeignKey(User, verbose_name=_("Автор"))
- content = models.TextField(_('Содержание'), blank=True)
- pub_date = models.DateTimeField(_('Дата публикации'), blank=True, null=True)
- moderation = models.CharField(
- _('Модерация'),
- max_length=1,
- choices=MODERATION_CHOICES,
- default=NOT_MODERATED
- )
Таким образом имеется возможность сократить программный код проекта и добавить возможность повторного использования повторяющегося кода.
Структура модели, например, для комментариев теперь может выглядеть следующим образом:
- class Comment(PostBase):
- class Meta:
- db_table = "comments"
- article = models.ForeignKey(Article)
Поля author , pub_date , content и moderation уже не требуется указывать, поскольку они присутствуют в PostBase классе. Главное, не вводите в свою модель поля с такими же именами, как в модели PostBase.
PostBaseAdmin
Также несомненным плюсом является то, что можно точно также сделать одну общую для всех классов настройку админ панели.
Сделаем, например, отображение полей, настройку поиска и фильтрации контента и возможность установки статуса модерации (то есть добавим соответствующие actions ).
- class PostBaseAdmin(admin.ModelAdmin):
- list_display = ('content', 'author', 'pub_date')
- search_fields = ('content', 'author__username')
- list_filter = ('moderation',)
- actions = ['make_spam', 'make_not_moderated', 'make_post_moderated', 'make_moderated']
- def moderate(self, request, rows_updated, choice_description):
- if rows_updated == 1:
- message_bit = "1 запись помечена, как %s" % choice_description
- else:
- message_bit = "%s записей отмечены, как %s." % (rows_updated, choice_description)
- self.message_user(request, "%s" % message_bit)
- def make_spam(self, request, queryset):
- self.moderate(
- request=request,
- rows_updated=queryset.update(moderation=PostBase.SPAM),
- choice_description="SPAM"
- )
- make_spam.short_description = "Отметить помеченные, как SPAM"
- def make_not_moderated(self, request, queryset):
- self.moderate(
- request=request,
- rows_updated=queryset.update(moderation=PostBase.NOT_MODERATED),
- choice_description="NOT_MODERATED"
- )
- make_not_moderated.short_description = "Отметить помеченные, как NOT_MODERATED"
- def make_post_moderated(self, request, queryset):
- self.moderate(
- request=request,
- rows_updated=queryset.update(moderation=PostBase.POST_MODERATED),
- choice_description="POST_MODERATED"
- )
- make_post_moderated.short_description = "Отметить помеченные, как POST_MODERATED"
- def make_moderated(self, request, queryset):
- self.moderate(
- request=request,
- rows_updated=queryset.update(moderation=PostBase.MODERATED),
- choice_description="MODERATED"
- )
- make_moderated.short_description = "Отметить помеченные, как MODERATED"
Если вы захотите расширить листы фильтрации или добавить actions, для какой-то из моделей, то можно наследоваться от PostBaseAdmin следующим образом:
- class ArticleAdmin(PostBaseAdmin):
- # Либо полностью переопределить отображаемые поля или поля для поиска
- list_display = ('title', 'section', 'author', 'pub_date', 'views', 'moderation')
- search_fields = ('title', 'author__username', 'section__title')
- # Либо добавить к существующему списку дополнительные
- list_filter = PostBaseAdmin.list_filter + ('status', 'section')
- # В случае actions подобное объявление просто добавит новые actions к существующим
- actions = ['publish', 'unpublish']
Для Django рекомендую VDS-сервера хостера Timeweb .