diff --git a/app/core/migrations/0006_delete_hotsearchablepage.py b/app/core/migrations/0006_delete_hotsearchablepage.py new file mode 100644 index 0000000..052077b --- /dev/null +++ b/app/core/migrations/0006_delete_hotsearchablepage.py @@ -0,0 +1,16 @@ +# Generated by Django 4.2.7 on 2024-07-09 22:09 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('core', '0005_remove_hotsearchablepage_intro_delete_testpagepage'), + ] + + operations = [ + migrations.DeleteModel( + name='HOTSearchablePage', + ), + ] diff --git a/app/core/models.py b/app/core/models.py index f5bc52c..cc04e8e 100644 --- a/app/core/models.py +++ b/app/core/models.py @@ -29,7 +29,3 @@ def __str__(self): class Meta: verbose_name_plural = "Partners" - - -class HotSearchablePage(Page): - pass diff --git a/app/members/migrations/0004_membergroupownerpage_membergrouppage.py b/app/members/migrations/0004_membergroupownerpage_membergrouppage.py new file mode 100644 index 0000000..9cb5a03 --- /dev/null +++ b/app/members/migrations/0004_membergroupownerpage_membergrouppage.py @@ -0,0 +1,52 @@ +# Generated by Django 4.2.7 on 2024-07-09 21:59 + +import django.core.validators +from django.db import migrations, models +import django.db.models.deletion +import wagtail.fields + + +class Migration(migrations.Migration): + + dependencies = [ + ('wagtailcore', '0089_log_entry_data_json_null_to_object'), + ('wagtailimages', '0025_alter_image_file_alter_rendition_file'), + ('members', '0003_rename_introduction_individualmemberpage_intro'), + ] + + operations = [ + migrations.CreateModel( + name='MemberGroupOwnerPage', + fields=[ + ('page_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='wagtailcore.page')), + ('search_placeholder', models.CharField(default='Search by name')), + ('filter_by_country', models.CharField(default='Filter by Country')), + ('sort_by_titlea', models.CharField(default='Sort by Title Alphabetical')), + ('sort_by_titlez', models.CharField(default='Sort by Title Reverse Alphabetical')), + ('load_more_text', models.CharField(default='Load more', help_text="This will be a prefix to the title of the page; i.e., if the page title is 'Voting members', and this field is 'Load more', this will end up appearing as 'Load more Voting members'.")), + ('footer_box_title', models.CharField(default='Work for HOT')), + ('footer_box_description', wagtail.fields.RichTextField(blank=True)), + ('footer_box_button_text', models.CharField(default='Check our Job Opportunities')), + ('footer_box_button_link', models.URLField(blank=True)), + ], + options={ + 'abstract': False, + }, + bases=('wagtailcore.page',), + ), + migrations.CreateModel( + name='MemberGroupPage', + fields=[ + ('page_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='wagtailcore.page')), + ('intro', wagtail.fields.RichTextField(blank=True, help_text='Appears in the header.')), + ('body_intro', wagtail.fields.RichTextField(blank=True)), + ('body_description', wagtail.fields.RichTextField(blank=True)), + ('desktop_size_items_per_row', models.SmallIntegerField(default=6, help_text='The number of members shown per row on desktop sizes.', validators=[django.core.validators.MinValueValidator(4), django.core.validators.MaxValueValidator(8)])), + ('header_image', models.ForeignKey(blank=True, help_text='Header image', null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='+', to='wagtailimages.image')), + ], + options={ + 'abstract': False, + }, + bases=('wagtailcore.page',), + ), + ] diff --git a/app/members/migrations/0005_individualmemberpage_member_groups.py b/app/members/migrations/0005_individualmemberpage_member_groups.py new file mode 100644 index 0000000..e151dc3 --- /dev/null +++ b/app/members/migrations/0005_individualmemberpage_member_groups.py @@ -0,0 +1,20 @@ +# Generated by Django 4.2.7 on 2024-07-09 22:14 + +from django.db import migrations +import wagtail.blocks +import wagtail.fields + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0004_membergroupownerpage_membergrouppage'), + ] + + operations = [ + migrations.AddField( + model_name='individualmemberpage', + name='member_groups', + field=wagtail.fields.StreamField([('member_group', wagtail.blocks.PageChooserBlock(page_type=['members.MemberGroupPage']))], blank=True, null=True, use_json_field=True), + ), + ] diff --git a/app/members/migrations/0006_alter_individualmemberpage_member_groups.py b/app/members/migrations/0006_alter_individualmemberpage_member_groups.py new file mode 100644 index 0000000..a981794 --- /dev/null +++ b/app/members/migrations/0006_alter_individualmemberpage_member_groups.py @@ -0,0 +1,20 @@ +# Generated by Django 4.2.7 on 2024-07-09 22:45 + +from django.db import migrations +import wagtail.blocks +import wagtail.fields + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0005_individualmemberpage_member_groups'), + ] + + operations = [ + migrations.AlterField( + model_name='individualmemberpage', + name='member_groups', + field=wagtail.fields.StreamField([('member_group', wagtail.blocks.StructBlock([('group', wagtail.blocks.PageChooserBlock(page_type=['members.MemberGroupPage'])), ('role', wagtail.blocks.CharBlock())]))], blank=True, null=True, use_json_field=True), + ), + ] diff --git a/app/members/migrations/0007_alter_individualmemberpage_member_groups.py b/app/members/migrations/0007_alter_individualmemberpage_member_groups.py new file mode 100644 index 0000000..14c8004 --- /dev/null +++ b/app/members/migrations/0007_alter_individualmemberpage_member_groups.py @@ -0,0 +1,20 @@ +# Generated by Django 4.2.7 on 2024-07-09 22:48 + +from django.db import migrations +import wagtail.blocks +import wagtail.fields + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0006_alter_individualmemberpage_member_groups'), + ] + + operations = [ + migrations.AlterField( + model_name='individualmemberpage', + name='member_groups', + field=wagtail.fields.StreamField([('member_group', wagtail.blocks.StructBlock([('group', wagtail.blocks.PageChooserBlock(page_type=['members.MemberGroupPage'])), ('role', wagtail.blocks.CharBlock(required=False))]))], blank=True, null=True, use_json_field=True), + ), + ] diff --git a/app/members/migrations/0008_membergrouppage_hub_shown_and_more.py b/app/members/migrations/0008_membergrouppage_hub_shown_and_more.py new file mode 100644 index 0000000..b032e78 --- /dev/null +++ b/app/members/migrations/0008_membergrouppage_hub_shown_and_more.py @@ -0,0 +1,23 @@ +# Generated by Django 4.2.7 on 2024-07-09 23:03 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0007_alter_individualmemberpage_member_groups'), + ] + + operations = [ + migrations.AddField( + model_name='membergrouppage', + name='hub_shown', + field=models.BooleanField(blank=True, default=False, null=True), + ), + migrations.AddField( + model_name='membergrouppage', + name='position_shown', + field=models.BooleanField(blank=True, default=False, null=True), + ), + ] diff --git a/app/members/migrations/0009_alter_membergrouppage_hub_shown_and_more.py b/app/members/migrations/0009_alter_membergrouppage_hub_shown_and_more.py new file mode 100644 index 0000000..06f523c --- /dev/null +++ b/app/members/migrations/0009_alter_membergrouppage_hub_shown_and_more.py @@ -0,0 +1,23 @@ +# Generated by Django 4.2.7 on 2024-07-09 23:05 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0008_membergrouppage_hub_shown_and_more'), + ] + + operations = [ + migrations.AlterField( + model_name='membergrouppage', + name='hub_shown', + field=models.BooleanField(default=False), + ), + migrations.AlterField( + model_name='membergrouppage', + name='position_shown', + field=models.BooleanField(default=False), + ), + ] diff --git a/app/members/migrations/0010_membergrouppage_show_search_options.py b/app/members/migrations/0010_membergrouppage_show_search_options.py new file mode 100644 index 0000000..d3c5861 --- /dev/null +++ b/app/members/migrations/0010_membergrouppage_show_search_options.py @@ -0,0 +1,18 @@ +# Generated by Django 4.2.7 on 2024-07-10 17:43 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0009_alter_membergrouppage_hub_shown_and_more'), + ] + + operations = [ + migrations.AddField( + model_name='membergrouppage', + name='show_search_options', + field=models.BooleanField(default=True), + ), + ] diff --git a/app/members/migrations/0011_membergroupownerpage_search_button_text.py b/app/members/migrations/0011_membergroupownerpage_search_button_text.py new file mode 100644 index 0000000..60de3dc --- /dev/null +++ b/app/members/migrations/0011_membergroupownerpage_search_button_text.py @@ -0,0 +1,18 @@ +# Generated by Django 4.2.7 on 2024-07-10 17:53 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0010_membergrouppage_show_search_options'), + ] + + operations = [ + migrations.AddField( + model_name='membergroupownerpage', + name='search_button_text', + field=models.CharField(default='Search'), + ), + ] diff --git a/app/members/migrations/0012_membergroupownerpage_view_all_text.py b/app/members/migrations/0012_membergroupownerpage_view_all_text.py new file mode 100644 index 0000000..f1c7235 --- /dev/null +++ b/app/members/migrations/0012_membergroupownerpage_view_all_text.py @@ -0,0 +1,18 @@ +# Generated by Django 4.2.7 on 2024-07-10 19:21 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('members', '0011_membergroupownerpage_search_button_text'), + ] + + operations = [ + migrations.AddField( + model_name='membergroupownerpage', + name='view_all_text', + field=models.CharField(default='View all'), + ), + ] diff --git a/app/members/models.py b/app/members/models.py index 58047ab..7121960 100644 --- a/app/members/models.py +++ b/app/members/models.py @@ -1,11 +1,14 @@ from django.db import models +from django.core.validators import MinValueValidator, MaxValueValidator from wagtail.admin.panels import FieldPanel, MultiFieldPanel from wagtail.blocks import CharBlock, StreamBlock, StructBlock, URLBlock, PageChooserBlock from wagtail.fields import RichTextField, StreamField from wagtail.models import Page from django.db.models import Q +from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger from app.projects.models import IndividualProjectPage from app.news.models import IndividualNewsPage +from app.mapping_hubs.models import IndividualMappingHubPage from wagtail.search import index class WebLinkStructBlock(StructBlock): @@ -17,6 +20,127 @@ class WebLinkBlock(StreamBlock): blocks = WebLinkStructBlock() +class MemberGroupOwnerPage(Page): + max_count = 1 + + subpage_types = [ + 'members.MemberGroupPage' + ] + + search_placeholder = models.CharField(default="Search by name") + filter_by_country = models.CharField(default="Filter by Country") + sort_by_titlea = models.CharField(default="Sort by Name Alphabetical") + sort_by_titlez = models.CharField(default="Sort by Name Reverse Alphabetical") + search_button_text = models.CharField(default="Search") + + load_more_text = models.CharField(default="Load more", help_text="This will be a prefix to the title of the page; i.e., if the page title is 'Voting members', and this field is 'Load more', this will end up appearing as 'Load more Voting members'.") + + view_all_text = models.CharField(default="View all") + + footer_box_title = models.CharField(default="Work for HOT") + footer_box_description = RichTextField(blank=True) + footer_box_button_text = models.CharField(default="Check our Job Opportunities") + footer_box_button_link = models.URLField(blank=True) + + content_panels = Page.content_panels + [ + MultiFieldPanel([ + FieldPanel('search_placeholder'), + FieldPanel('filter_by_country'), + FieldPanel('sort_by_titlea'), + FieldPanel('sort_by_titlez'), + FieldPanel('search_button_text'), + ], heading="Search options"), + FieldPanel('load_more_text'), + FieldPanel('view_all_text'), + MultiFieldPanel([ + FieldPanel('footer_box_title'), + FieldPanel('footer_box_description'), + FieldPanel('footer_box_button_text'), + FieldPanel('footer_box_button_link'), + ], heading="Footer box"), + ] + + +class MemberGroupPage(Page): + def get_context(self, request, *args, **kwargs): + context = super().get_context(request, *args, **kwargs) + + members = IndividualMemberPage.objects.live().filter( + Q(member_groups__contains=[{'type': 'member_group', 'value': { 'group': context['page'].id }}]) + ).filter(locale=context['page'].locale) + + keyword = request.GET.get('keyword', '') + + if keyword: + members = members.search(keyword).get_queryset() + + hubs = IndividualMappingHubPage.objects.live().filter(locale=context['page'].locale) + query = Q() + for hub in hubs: + if request.GET.get(str(hub), ''): + query = query | Q(location_hub=hub) + members = members.filter(query).distinct() + + match request.GET.get('sort', ''): + case 'sort.titlea': + members = members.order_by('title') + case 'sort.titlez': + members = members.order_by('-title') + case _: + members = members.order_by('title') + + page = request.GET.get('page', 1) + paginator = Paginator(members, 12) # if you want more/less items per page (i.e., per load), change the number here to something else + try: + members = paginator.page(page) + except PageNotAnInteger: + members = paginator.page(1) + except EmptyPage: + members = paginator.page(paginator.num_pages) + + context['members'] = members + context['hubs'] = hubs + context['groups'] = MemberGroupPage.objects.live().filter(locale=context['page'].locale).exclude(id=context['page'].id) + + return context + + parent_page_type = [ + 'members.MemberGroupOwnerPage' + ] + + header_image = models.ForeignKey( + "wagtailimages.Image", + null=True, + blank=True, + on_delete=models.SET_NULL, + related_name="+", + help_text="Header image" + ) + intro = RichTextField(blank=True, help_text="Appears in the header.") + + body_intro = RichTextField(blank=True) + body_description = RichTextField(blank=True) + + show_search_options = models.BooleanField(default=True) + desktop_size_items_per_row = models.SmallIntegerField(default=6, help_text="The number of members shown per row on desktop sizes.", validators=[ + MinValueValidator(4), + MaxValueValidator(8), + ]) + position_shown = models.BooleanField(default=False) + hub_shown = models.BooleanField(default=False) + + content_panels = Page.content_panels + [ + FieldPanel('header_image'), + FieldPanel('intro'), + FieldPanel('body_intro'), + FieldPanel('body_description'), + FieldPanel('show_search_options'), + FieldPanel('desktop_size_items_per_row'), + FieldPanel('position_shown'), + FieldPanel('hub_shown'), + ] + + class MemberOwnerPage(Page): max_count = 1 @@ -30,6 +154,12 @@ class MemberOwnerPage(Page): FieldPanel('project_contribution_title'), ] + +class MemberGroupBlock(StructBlock): + group = PageChooserBlock(page_type="members.MemberGroupPage") + role = CharBlock(required=False) + + """ This page should only be created as a child of a MemberOwnerPage! Its template depends on fields from the MemberOwnerPage in order @@ -63,6 +193,7 @@ def get_context(self, request, *args, **kwargs): related_name="+", help_text="An image of the member", ) + member_groups = StreamField([('member_group', MemberGroupBlock())], use_json_field=True, null=True, blank=True) position = models.CharField() location_hub = models.ForeignKey( 'mapping_hubs.IndividualMappingHubPage', @@ -73,7 +204,6 @@ def get_context(self, request, *args, **kwargs): ) intro = RichTextField() on_the_web_links = StreamField(WebLinkBlock(), blank=True, use_json_field=True) - search_fields = Page.search_fields + [ index.SearchField('title'), @@ -83,6 +213,7 @@ def get_context(self, request, *args, **kwargs): content_panels = Page.content_panels + [ FieldPanel('image'), + FieldPanel('member_groups'), FieldPanel('position'), FieldPanel('location_hub'), FieldPanel('intro'), diff --git a/app/members/templates/members/components/MembersSortOption.html b/app/members/templates/members/components/MembersSortOption.html new file mode 100644 index 0000000..8e3320f --- /dev/null +++ b/app/members/templates/members/components/MembersSortOption.html @@ -0,0 +1,4 @@ +
- Board - / - Voting Member - / - Staff -
+ {% if page.member_groups %} ++ {% for group in page.member_groups %} + {{group.value.role}} + {% if not forloop.last %} / {% endif %} + {% endfor %} +
+ {% endif %}{{page.position}} @@ -34,14 +34,14 @@
+ {% if members.has_next %} + {% comment %} {% endcomment %} + + {% endif %} +
+ + {% comment %} BOTTOM AREA {% endcomment %} ++ {{member.position}} +
+ {% endif %} + + {% if hub_shown and member.location_hub %} + + {% endif %} +