From 0fd17d86945596859742e079e1fc8e52aa7ff932 Mon Sep 17 00:00:00 2001 From: George Katsikas <giorgakis.katsikas@gmail.com> Date: Wed, 13 Mar 2024 10:49:38 +0100 Subject: [PATCH] add orphaned subsidyattachment list view related to #218 --- scipost_django/finances/forms.py | 91 ++++++++++++++++++- scipost_django/finances/managers.py | 3 + .../_hx_subsidyattachment_link_form.html | 7 ++ .../_hx_subsidyattachment_list_item.html | 15 +++ .../_hx_subsidyattachment_list_page.html | 44 +++++++++ .../templates/finances/subsidy_list.html | 3 + .../subsidyattachment_orphaned_list.html | 30 ++++++ scipost_django/finances/urls.py | 15 +++ scipost_django/finances/views.py | 56 ++++++++++-- 9 files changed, 255 insertions(+), 9 deletions(-) create mode 100644 scipost_django/finances/templates/finances/_hx_subsidyattachment_link_form.html create mode 100644 scipost_django/finances/templates/finances/_hx_subsidyattachment_list_item.html create mode 100644 scipost_django/finances/templates/finances/_hx_subsidyattachment_list_page.html create mode 100644 scipost_django/finances/templates/finances/subsidyattachment_orphaned_list.html diff --git a/scipost_django/finances/forms.py b/scipost_django/finances/forms.py index 0aee1b6cc..ab9cb64d4 100644 --- a/scipost_django/finances/forms.py +++ b/scipost_django/finances/forms.py @@ -286,6 +286,91 @@ class SubsidyPaymentForm(forms.ModelForm): return instance +class SubsidyAttachmentInlineLinkForm(forms.ModelForm): + class Meta: + model = SubsidyAttachment + fields = [ + "subsidy", + ] + + filename = forms.CharField( + label="Filename", + required=True, + ) + subsidy = forms.ModelChoiceField( + queryset=Subsidy.objects.all(), + widget=autocomplete.ModelSelect2( + url=reverse_lazy("finances:subsidy_autocomplete"), + attrs={ + "data-html": True, + "style": "width: 100%", + }, + ), + help_text=("Start typing, and select from the popup."), + required=False, + ) + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + + self.helper = FormHelper() + self.helper.layout = Layout( + Div( + Div(Field("filename"), css_class="col-6 col"), + Div(Field("subsidy"), css_class="col-6 col"), + css_class="row mb-0", + ) + ) + + self.fields["filename"].initial = self.instance.filename + + def clean(self): + orphaned = self.cleaned_data["subsidy"] is None + filename = self.cleaned_data["filename"] + + # Allow misnamed orphans + if orphaned: + return + + filename_regex = ( + "^SciPost_" + "[0-9]{4,}(-[0-9]{4,})?_[A-Z]{2,}_[\w]+_" + "(Agreement|Invoice|ProofOfPayment|Other)" + "(-[0-9]{2,})?(_[\w]+)?\.(pdf|docx|png)$" + ) + pattern = re.compile(filename_regex) + if not pattern.match(filename): + self.add_error( + "filename", + "The filename does not match the required regex pattern " + f"'{filename_regex}'", + ) + + def save(self, commit=True): + instance: "SubsidyAttachment" = super().save(commit=False) + + filename = self.cleaned_data["filename"] + old_relative_path = instance.attachment.name + new_relative_path = instance.attachment.name.replace( + instance.filename, filename + ) + + try: + instance.attachment.storage.save(new_relative_path, instance.attachment) + instance.attachment.storage.delete(old_relative_path) + instance.attachment.name = new_relative_path + except Exception as e: + self.add_error( + "filename", + f"An error occurred while renaming the file: {e}", + ) + + if commit: + instance.save() + + return instance + + class SubsidyAttachmentForm(forms.ModelForm): class Meta: model = SubsidyAttachment @@ -317,7 +402,7 @@ class SubsidyAttachmentForm(forms.ModelForm): def clean(self): orphaned = self.cleaned_data["subsidy"] is None - attachment = self.cleaned_data["attachment"] + attachment_filename = self.cleaned_data["attachment"].name.split("/")[-1] # Allow misnamed orphans if orphaned: @@ -330,7 +415,9 @@ class SubsidyAttachmentForm(forms.ModelForm): "(-[0-9]{2,})?(_[\w]+)?\.(pdf|docx|png)$" ) pattern = re.compile(filename_regex) - if not pattern.match(attachment.name): + + # + if not pattern.match(attachment_filename): self.add_error( "attachment", "The filename does not match the required regex pattern " diff --git a/scipost_django/finances/managers.py b/scipost_django/finances/managers.py index d0acab170..cf400aa15 100644 --- a/scipost_django/finances/managers.py +++ b/scipost_django/finances/managers.py @@ -29,3 +29,6 @@ class SubsidyAttachmentQuerySet(models.QuerySet): def proofs_of_payment(self): return self.filter(kind=self.model.KIND_PROOF_OF_PAYMENT) + + def orphaned(self): + return self.filter(subsidy__isnull=True) diff --git a/scipost_django/finances/templates/finances/_hx_subsidyattachment_link_form.html b/scipost_django/finances/templates/finances/_hx_subsidyattachment_link_form.html new file mode 100644 index 000000000..1c68cc170 --- /dev/null +++ b/scipost_django/finances/templates/finances/_hx_subsidyattachment_link_form.html @@ -0,0 +1,7 @@ +{% load crispy_forms_tags %} + +<form hx-post="{% url "finances:_hx_subsidyattachment_link_form" attachment_id=attachment.id %}" + hx-trigger="change delay:1000ms" + hx-swap="outerHTML"> + {% crispy form %} +</form> diff --git a/scipost_django/finances/templates/finances/_hx_subsidyattachment_list_item.html b/scipost_django/finances/templates/finances/_hx_subsidyattachment_list_item.html new file mode 100644 index 000000000..73e89a671 --- /dev/null +++ b/scipost_django/finances/templates/finances/_hx_subsidyattachment_list_item.html @@ -0,0 +1,15 @@ +<div class="row border-bottom"> + <div class="col-auto d-flex flex-column justify-content-between"> + <span>{{ attachment.get_kind_display }}</span> + <span>{{ attachment.date|date:"SHORT_DATE_FORMAT" }}</span> + <a href="{% url 'finances:subsidyattachment_update' pk=attachment.id %}"><span class="text-warning">Update</span></a> + <a href="{% url 'finances:subsidy_attachment' attachment_id=attachment.id %}">View</a> + </div> + <div class="col" + hx-get="{% url "finances:_hx_subsidyattachment_link_form" attachment_id=attachment.id %}" + hx-trigger="revealed once"> + <div class="spinner-grow spinner-grow-sm ms-2" + role="status" + aria-hidden="true"></div> + </div> +</div> diff --git a/scipost_django/finances/templates/finances/_hx_subsidyattachment_list_page.html b/scipost_django/finances/templates/finances/_hx_subsidyattachment_list_page.html new file mode 100644 index 000000000..d8ee47ac4 --- /dev/null +++ b/scipost_django/finances/templates/finances/_hx_subsidyattachment_list_page.html @@ -0,0 +1,44 @@ +{% for attachment in page_obj %} + {% include 'finances/_hx_subsidyattachment_list_item.html' %} +{% empty %} + <tr id="orphaned-subsidies-results-load-next" hx-swap-oob="true"> + <td colspan="12" class="text-center p-0"> + <div class="p-2 d-flex justify-content-center"> + <strong>No orphaned SubsidyAttachments could be found</strong> + </div> + </td> + </tr> +{% endfor %} + +{% if page_obj.has_next %} + <tr id="orphaned-subsidies-results-load-next" + class="htmx-indicator" + hx-swap-oob="true" + hx-post="{% url 'finances:_hx_subsidyattachment_list_page' identifier_w_vn_nr=submission.preprint.identifier_w_vn_nr %}?page={{ page_obj.next_page_number }}" + hx-target="#orphaned-subsidies-results" + hx-include="#orphaned-subsidies-form" + hx-trigger="revealed" + hx-swap="beforeend" + hx-indicator="#orphaned-subsidies-results-load-next"> + + <td colspan="12" class="text-center p-0"> + <div class="p-2 bg-primary bg-opacity-25 d-flex justify-content-center"> + <strong>Loading page {{ page_obj.next_page_number }} out of {{ page_obj.paginator.num_pages }}</strong> + <div class="spinner-grow spinner-grow-sm ms-2" + role="status" + aria-hidden="true"></div> + </div> + </td> + </tr> +{% else %} + <tr id="orphaned-subsidies-results-load-next" hx-swap-oob="true"> + + <td colspan="12" class="text-center p-0"> + <div class="p-2 d-flex justify-content-center"> + <strong>All SubsidyAttachments loaded</strong> + </div> + </td> + </tr> +{% endif %} + +{{ form_media }} diff --git a/scipost_django/finances/templates/finances/subsidy_list.html b/scipost_django/finances/templates/finances/subsidy_list.html index 0f844698f..bab324d2d 100644 --- a/scipost_django/finances/templates/finances/subsidy_list.html +++ b/scipost_django/finances/templates/finances/subsidy_list.html @@ -29,6 +29,9 @@ <li> <a href="{% url 'finances:subsidyattachment_create' %}">Add a SubsidyAttachment</a> </li> + <li> + <a href="{% url 'finances:subsidyattachment_orphaned_list' %}">Link orphaned SubsidyAttachments</a> + </li> <li> <a href="{% url 'finances:subsidies_old' %}" target="_blank">Go to the old list page</a> </li> diff --git a/scipost_django/finances/templates/finances/subsidyattachment_orphaned_list.html b/scipost_django/finances/templates/finances/subsidyattachment_orphaned_list.html new file mode 100644 index 000000000..03ac95bbb --- /dev/null +++ b/scipost_django/finances/templates/finances/subsidyattachment_orphaned_list.html @@ -0,0 +1,30 @@ +{% extends 'finances/base.html' %} +{% load user_groups %} +{% load crispy_forms_tags %} + + +{% block breadcrumb_items %} + {{ block.super }} + <span class="breadcrumb-item">Orphaned SubsidyAttachments</span> +{% endblock %} + + +{% block meta_description %} + {{ block.super }} Orphaned SubsidyAttachment List +{% endblock meta_description %} + +{% block pagetitle %} + : Orphaned SubsidyAttachments +{% endblock pagetitle %} + +{% block content %} + {% is_ed_admin request.user as is_ed_admin %} + <h1 class="highlight">Orphaned SubsidyAttachment List</h1> + + <div id="orphaned-subsidies-results" + hx-get="{% url 'finances:_hx_subsidyattachment_list_page' %}?page=1" + hx-trigger="load once"></div> + + + +{% endblock content %} diff --git a/scipost_django/finances/urls.py b/scipost_django/finances/urls.py index 02bfde69f..596925d49 100644 --- a/scipost_django/finances/urls.py +++ b/scipost_django/finances/urls.py @@ -139,6 +139,21 @@ urlpatterns = [ views.subsidy_attachment, name="subsidy_attachment", ), + path( + "subsidies/attachments/orphaned/", + views.subsidyattachment_orphaned_list, + name="subsidyattachment_orphaned_list", + ), + path( + "subsidies/attachments/orphaned/_hx_list_page", + views._hx_subsidyattachment_list_page, + name="_hx_subsidyattachment_list_page", + ), + path( + "subsidies/attachments/_hx_link_form/<int:attachment_id>", + views._hx_subsidyattachment_link_form, + name="_hx_subsidyattachment_link_form", + ), # Timesheets path("timesheets", views.timesheets, name="timesheets"), path("timesheets/detailed", views.timesheets_detailed, name="timesheets_detailed"), diff --git a/scipost_django/finances/views.py b/scipost_django/finances/views.py index de0186c44..0d115cb7a 100644 --- a/scipost_django/finances/views.py +++ b/scipost_django/finances/views.py @@ -8,6 +8,7 @@ import mimetypes from dal import autocomplete from django.db.models import Q +from django.template.response import TemplateResponse from django.utils.html import format_html import matplotlib @@ -16,7 +17,7 @@ import matplotlib.pyplot as plt import io, base64 from django.contrib import messages -from django.contrib.auth.decorators import permission_required +from django.contrib.auth.decorators import login_required, permission_required from django.contrib.auth.mixins import LoginRequiredMixin from django.core.exceptions import PermissionDenied from django.core.paginator import Paginator @@ -29,6 +30,7 @@ from django.views.generic.edit import CreateView, UpdateView, DeleteView from django.views.generic.list import ListView from .forms import ( + SubsidyAttachmentInlineLinkForm, SubsidyForm, SubsidySearchForm, SubsidyPaymentForm, @@ -529,9 +531,10 @@ class SubsidyAttachmentUpdateView(PermissionsMixin, UpdateView): return context def get_success_url(self): - return reverse_lazy( - "finances:subsidy_details", kwargs={"pk": self.object.subsidy.id} - ) + if subsidy := self.object.subsidy: + return reverse_lazy("finances:subsidy_details", kwargs={"pk": subsidy.id}) + + return reverse_lazy("finances:subsidies") class SubsidyAttachmentDeleteView(PermissionsMixin, DeleteView): @@ -543,9 +546,48 @@ class SubsidyAttachmentDeleteView(PermissionsMixin, DeleteView): model = SubsidyAttachment def get_success_url(self): - return reverse_lazy( - "finances:subsidy_details", kwargs={"pk": self.object.subsidy.id} - ) + if subsidy := self.object.subsidy: + return reverse_lazy("finances:subsidy_details", kwargs={"pk": subsidy.id}) + + return reverse_lazy("finances:subsidies") + + +@login_required() +@permission_required("scipost.can_manage_subsidies", raise_exception=True) +def subsidyattachment_orphaned_list(request): + return TemplateResponse( + request, "finances/subsidyattachment_orphaned_list.html", {} + ) + + +def _hx_subsidyattachment_list_page(request): + attachments = SubsidyAttachment.objects.orphaned() + paginator = Paginator(attachments, 16) + page_nr = request.GET.get("page") + page_obj = paginator.get_page(page_nr) + count = paginator.count + start_index = page_obj.start_index + + context = { + "count": count, + "page_obj": page_obj, + "start_index": start_index, + "form_media": SubsidyAttachmentForm().media, + } + return render(request, "finances/_hx_subsidyattachment_list_page.html", context) + + +def _hx_subsidyattachment_link_form(request, attachment_id): + attachment = get_object_or_404(SubsidyAttachment, pk=attachment_id) + form = SubsidyAttachmentInlineLinkForm(request.POST or None, instance=attachment) + if form.is_valid(): + form.save() + + context = { + "attachment": attachment, + "form": form, + } + return render(request, "finances/_hx_subsidyattachment_link_form.html", context) def subsidy_attachment(request, attachment_id): -- GitLab