diff --git a/froide_govplan/admin.py b/froide_govplan/admin.py index 58df0c1..adbfcf8 100644 --- a/froide_govplan/admin.py +++ b/froide_govplan/admin.py @@ -3,11 +3,17 @@ from django.contrib import admin, auth from django.urls import reverse_lazy from django.utils.translation import gettext_lazy as _ +from adminsortable2.admin import SortableAdminMixin from tinymce.widgets import TinyMCE from froide.helper.widgets import TagAutocompleteWidget -from .models import Government, GovernmentPlan, GovernmentPlanUpdate +from .models import ( + Government, + GovernmentPlan, + GovernmentPlanSection, + GovernmentPlanUpdate, +) User = auth.get_user_model() @@ -208,9 +214,26 @@ class GovernmentPlanUpdateAdmin(admin.ModelAdmin): return super().has_change_permission(request, obj=obj) +class GovernmentPlanSectionAdmin(SortableAdminMixin, admin.ModelAdmin): + save_on_top = True + prepopulated_fields = {"slug": ("title",)} + search_fields = ("title",) + raw_id_fields = ("categories",) + list_display = ( + "title", + "featured", + ) + list_filter = ( + "featured", + "categories", + "government", + ) + + admin.site.register(Government, GovernmentAdmin) admin.site.register(GovernmentPlan, GovernmentPlanAdmin) admin.site.register(GovernmentPlanUpdate, GovernmentPlanUpdateAdmin) +admin.site.register(GovernmentPlanSection, GovernmentPlanSectionAdmin) govplan_admin_site = GovPlanAdminSite(name="govplan") govplan_admin_site.register(GovernmentPlan, GovernmentPlanAdmin) diff --git a/froide_govplan/migrations/0005_governmentplansection.py b/froide_govplan/migrations/0005_governmentplansection.py new file mode 100644 index 0000000..3cf2947 --- /dev/null +++ b/froide_govplan/migrations/0005_governmentplansection.py @@ -0,0 +1,41 @@ +# Generated by Django 3.2.12 on 2022-03-14 10:14 + +import cms.models.fields +from django.conf import settings +from django.db import migrations, models +import django.db.models.deletion +import filer.fields.image + + +class Migration(migrations.Migration): + + dependencies = [ + ('publicbody', '0039_publicbody_alternative_emails'), + migrations.swappable_dependency(settings.FILER_IMAGE_MODEL), + ('cms', '0022_auto_20180620_1551'), + ('froide_govplan', '0004_auto_20220311_2330'), + ] + + operations = [ + migrations.CreateModel( + name='GovernmentPlanSection', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('title', models.CharField(max_length=255, verbose_name='title')), + ('slug', models.SlugField(max_length=255, unique=True, verbose_name='slug')), + ('description', models.TextField(blank=True, verbose_name='description')), + ('icon', models.CharField(blank=True, help_text='Enter an icon name from the FontAwesome 4 icon set', max_length=50, verbose_name='Icon')), + ('order', models.PositiveIntegerField(default=0)), + ('featured', models.DateTimeField(blank=True, null=True)), + ('categories', models.ManyToManyField(blank=True, to='publicbody.Category')), + ('content_placeholder', cms.models.fields.PlaceholderField(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, slotname='content', to='cms.placeholder')), + ('government', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='froide_govplan.government', verbose_name='government')), + ('image', filer.fields.image.FilerImageField(blank=True, default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.FILER_IMAGE_MODEL, verbose_name='image')), + ], + options={ + 'verbose_name': 'Government plan section', + 'verbose_name_plural': 'Government plan sections', + 'ordering': ('order', 'title'), + }, + ), + ] diff --git a/froide_govplan/models.py b/froide_govplan/models.py index bd100df..67b45da 100644 --- a/froide_govplan/models.py +++ b/froide_govplan/models.py @@ -15,9 +15,11 @@ from froide.organization.models import Organization from froide.publicbody.models import Category, Jurisdiction, PublicBody try: + from cms.models.fields import PlaceholderField from cms.models.pluginmodel import CMSPlugin except ImportError: CMSPlugin = None + PlaceholderField = None class PlanStatus(models.TextChoices): @@ -251,6 +253,60 @@ class GovernmentPlanFollower(Follower): verbose_name_plural = _("Government plan followers") +class GovernmentPlanSection(models.Model): + government = models.ForeignKey( + Government, on_delete=models.CASCADE, verbose_name=_("government") + ) + + title = models.CharField(max_length=255, verbose_name=_("title")) + slug = models.SlugField(max_length=255, unique=True, verbose_name=_("slug")) + + categories = models.ManyToManyField(Category, blank=True) + + description = models.TextField(blank=True, verbose_name=_("description")) + image = FilerImageField( + null=True, + blank=True, + default=None, + verbose_name=_("image"), + on_delete=models.SET_NULL, + ) + + icon = models.CharField( + _("Icon"), + max_length=50, + blank=True, + help_text=_( + """Enter an icon name from the FontAwesome 4 icon set""" + ), + ) + order = models.PositiveIntegerField(default=0) + featured = models.DateTimeField(null=True, blank=True) + + if PlaceholderField: + content_placeholder = PlaceholderField("content") + + class Meta: + verbose_name = _("Government plan section") + verbose_name_plural = _("Government plan sections") + ordering = ( + "order", + "title", + ) + + def __str__(self): + return self.title + + def get_absolute_url(self): + return reverse( + "govplan:section", + kwargs={"gov": self.government.slug, "section": self.slug}, + ) + + def get_absolute_domain_url(self): + return settings.SITE_URL + self.get_absolute_url() + + if CMSPlugin: PLUGIN_TEMPLATES = [ diff --git a/froide_govplan/templates/froide_govplan/section.html b/froide_govplan/templates/froide_govplan/section.html new file mode 100644 index 0000000..f58575c --- /dev/null +++ b/froide_govplan/templates/froide_govplan/section.html @@ -0,0 +1,29 @@ +{% extends CMS_TEMPLATE %} + +{% load i18n %} +{% load markup %} +{% load cms_tags %} +{% load follow_tags %} + +{% block title %}{{ object.title }}{% endblock %} + + +{% block app_body %} +