skillbox/server/books/models/module.py

188 lines
6.2 KiB
Python

from django.db import models
from django.utils import timezone
from wagtail.admin.panels import FieldPanel, InlinePanel, TabbedInterface, ObjectList, MultiFieldPanel, MultipleChooserPanel
from wagtail.fields import RichTextField
from django.utils.translation import gettext_lazy as _
from core.constants import DEFAULT_RICH_TEXT_FEATURES
from core.wagtail_utils import StrictHierarchyPage, get_default_settings
from users.models import SchoolClass
from django.utils.text import slugify
FILTER_ATTRIBUTE_TYPE = (
('all', 'All'),
('exact', 'Exact')
)
class ModuleLevel(models.Model):
name = models.CharField(max_length=255, unique=True)
filter_attribute_type = models.CharField(
max_length=16,
choices=FILTER_ATTRIBUTE_TYPE,
default='exact'
)
def __str__(self):
return self.name
class Meta:
verbose_name_plural = _("module Levels")
verbose_name = _("module level")
def default_category():
return ModuleLevel.objects.first().pk
class ModuleCategory(models.Model):
class Meta:
verbose_name = _("module type")
verbose_name_plural = _("module types")
ordering = ("name",)
name = models.CharField(max_length=255)
filter_attribute_type = models.CharField(
max_length=16,
choices=FILTER_ATTRIBUTE_TYPE,
default='exact'
)
def __str__(self):
return f"{self.name}"
class Module(StrictHierarchyPage):
class Meta:
verbose_name = "Modul"
verbose_name_plural = "Module"
meta_title = models.CharField(max_length=255, help_text="e.g. 'Intro' or 'Modul 1'")
level = models.ForeignKey(ModuleLevel, on_delete=models.SET_NULL, blank=True, null=True)
category = models.ForeignKey(ModuleCategory, on_delete=models.SET_NULL, blank=True, null=True)
hero_image = models.ForeignKey(
"wagtailimages.Image",
null=True,
blank=True,
on_delete=models.SET_NULL,
related_name="+",
)
hero_source = models.CharField(
max_length=255, help_text="e.g. 'Reuters', 'Wikipedia'", blank=True
)
teaser = models.TextField()
intro = RichTextField(features=DEFAULT_RICH_TEXT_FEATURES)
solutions_enabled_for = models.ManyToManyField(SchoolClass)
content_panels = [
FieldPanel("title", classname="full title"),
FieldPanel("meta_title", classname="full title"),
FieldPanel("level"),
FieldPanel("category"),
FieldPanel("hero_image"),
FieldPanel("hero_source"),
FieldPanel("teaser"),
FieldPanel("intro"),
# TODO: Why is this commented out?
# InlinePanel(
# "assignments",
# label=_("Assignment"),
# classname="collapsed",
# heading=_("linked assignments"),
# help_text=_(
# "These %s are automatically linked, they are shown here only to provide an overview. Please don't change anything here."
# )
# % _("assignments"),
# ),
# InlinePanel(
# "surveys",
# heading=_("linked surveys"),
# label=_("Survey"),
# classname="collapsed",
# help_text=_(
# "These %s are automatically linked, they are shown here only to provide an overview. Please don't change anything here."
# )
# % _("surveys"),
# ),
]
edit_handler = TabbedInterface(
[ObjectList(content_panels, heading="Content"), get_default_settings()]
)
template = "generic_page.html"
parent_page_types = ["books.Topic"]
subpage_types = ["books.Chapter"]
# todo: isn't this a duplicate definition?
def get_child_ids(self):
return self.get_children().values_list("id", flat=True)
def sync_from_school_class(self, school_class_template, school_class_to_sync):
# import here so we don't get a circular import error
from books.models import Chapter, ContentBlock
# get chapters of module
chapters = Chapter.get_by_parent(self)
content_block_query = ContentBlock.objects.none()
# get content blocks of chapters
for chapter in chapters:
content_block_query = content_block_query.union(
ContentBlock.get_by_parent(chapter)
)
# clear all `hidden for` and `visible for` for class `school_class_to_sync`
for content_block in school_class_to_sync.hidden_content_blocks.intersection(
content_block_query
):
content_block.hidden_for.remove(school_class_to_sync)
for content_block in school_class_to_sync.visible_content_blocks.intersection(
content_block_query
):
content_block.visible_for.remove(school_class_to_sync)
# get all content blocks with `hidden for` for class `school_class_pattern`
for content_block in school_class_template.hidden_content_blocks.intersection(
content_block_query
):
# add `school_class_to_sync` to these blocks' `hidden for`
content_block.hidden_for.add(school_class_to_sync)
# get all content blocks with `visible for` for class `school_class_pattern`
for content_block in school_class_template.visible_content_blocks.intersection(
content_block_query
):
# add `school_class_to_sync` to these blocks' `visible for`
content_block.visible_for.add(school_class_to_sync)
for chapter in chapters:
chapter.sync_title_visibility(school_class_template, school_class_to_sync)
chapter.sync_description_visibility(
school_class_template, school_class_to_sync
)
objective_groups = self.objective_groups.all()
for objective_group in objective_groups:
objective_group.sync_visibility(school_class_template, school_class_to_sync)
def get_admin_display_title(self):
return f"{self.meta_title} - {self.title}"
class RecentModule(models.Model):
module = models.ForeignKey(
Module, on_delete=models.CASCADE, related_name="recent_modules"
)
user = models.ForeignKey("users.User", on_delete=models.CASCADE)
visited = models.DateTimeField(default=timezone.now)
class Meta:
get_latest_by = "visited"