SciPost Code Repository

Skip to content
Snippets Groups Projects
Commit 7c19288b authored by George Katsikas's avatar George Katsikas :goat:
Browse files

add bulk assignment of officer and supervisors

parent 2d110ad3
No related branches found
No related tags found
1 merge request!43Polish up new production page
......@@ -12,6 +12,7 @@ from django.db.models.functions import Greatest
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Div, Field, Submit
from crispy_bootstrap5.bootstrap5 import FloatingField
from django.urls import reverse
from journals.models import Journal
from markup.widgets import TextareaWithPreview
......@@ -399,3 +400,41 @@ class ProductionStreamSearchForm(forms.Form):
)
return streams
class BulkAssignOfficersForm(forms.Form):
officer = forms.ModelChoiceField(
queryset=ProductionUser.objects.active().filter(
user__groups__name="Production Officers"
),
required=False,
empty_label="Unchanged",
)
supervisor = forms.ModelChoiceField(
queryset=ProductionUser.objects.active().filter(
user__groups__name="Production Supervisor"
),
required=False,
empty_label="Unchanged",
)
def __init__(self, *args, **kwargs):
self.productionstreams = kwargs.pop("productionstreams", None)
super().__init__(*args, **kwargs)
self.helper = FormHelper()
self.helper.form_id = "productionstreams-bulk-action-form"
self.helper.attrs = {
"hx-post": reverse(
"production:_hx_productionstream_actions_bulk_assign_officers"
),
"hx-target": "#productionstream-bulk-assign-officers-container",
"hx-swap": "outerHTML",
"hx-confirm": "Are you sure you want to assign the selected production streams to the selected officers?",
}
self.helper.layout = Layout(
Div(
Div(Field("supervisor"), css_class="col-6 col-md-4 col-lg-3"),
Div(Field("officer"), css_class="col-6 col-md-4 col-lg-3"),
css_class="row mb-0",
),
)
{% load crispy_forms_tags %}
<div id="productionstream-bulk-assign-officers-container">
<h3>Bulk Assign Officers</h3>
<div>{% crispy form %}</div>
<div class="row mb-0">
<div class="col-12 col-sm-auto col-md-12 col-lg-auto h-100 d-none-empty">
<div class="row m-0 d-none-empty">
<button id="productionstreams-bulk-action-form-button"
class="btn btn-primary"
form="productionstreams-bulk-action-form">Assign</button>
</div>
</div>
</div>
</div>
......@@ -4,7 +4,7 @@
<div class="col-auto">
<input type="checkbox"
class="form-check-input checkbox-lg"
name="productionstream-bulk-action"
name="productionstream-bulk-action-selected"
value="{{ productionstream.id }}"
id="productionstream-{{ productionstream.id }}-checkbox"
form="productionstreams-bulk-action-form">
......@@ -60,7 +60,7 @@
<div id="productionstream-{{ productionstream.id }}-summary-assignees"
class="col-md-6"
hx-get="{% url 'production:render_stream_assignees_status' productionstream.id %}"
hx-trigger="intersect once, submit from:#productionstream-{{ productionstream.id }}-details target:form delay:500, click from:#productionstream-{{ productionstream.id }}-details target:.proof-action-button delay:500">
hx-trigger="intersect once, submit from:#productionstream-{{ productionstream.id }}-details target:form delay:500, click from:#productionstream-{{ productionstream.id }}-details target:.proof-action-button delay:500, submit from:#productionstreams-filter-details target:#productionstreams-bulk-action-form delay:500">
{% comment %} Placeholder while HTMX is loading {% endcomment %}
<div class="row mb-0 placeholder-glow">
......
......@@ -13,7 +13,7 @@
<h1>Production Streams</h1>
<details class="card my-4">
<details id="productionstreams-filter-details" class="card my-4">
<summary class="card-header fs-6 d-inline-flex align-items-center">
Search / Filter / Bulk Actions
<div class="d-none d-sm-inline-flex ms-auto align-items-center">
......@@ -45,6 +45,13 @@
<div id="search-productionstreams-form">{% crispy search_productionstreams_form %}</div>
</form>
{% comment %} Bulk Action buttons {% endcomment %}
<hr>
<div hx-get="{% url 'production:_hx_productionstream_actions_bulk_assign_officers' %}"
hx-trigger="load"></div>
</div>
</details>
......
......@@ -20,6 +20,11 @@ urlpatterns = [
production_views._hx_productionstream_list,
name="_hx_productionstream_list",
),
path(
"_hx_productionstream_actions_bulk_assign_officers",
production_views._hx_productionstream_actions_bulk_assign_officers,
name="_hx_productionstream_actions_bulk_assign_officers",
),
path(
"productionstreams/<int:productionstream_id>/",
include(
......
......@@ -31,6 +31,7 @@ from .models import (
ProductionEventAttachment,
)
from .forms import (
BulkAssignOfficersForm,
ProductionStreamSearchForm,
ProductionEventForm,
ProductionEventForm_deprec,
......@@ -54,9 +55,11 @@ from .utils import proofs_slug_to_id, ProductionUtils
@is_production_user()
@permission_required("scipost.can_view_production", raise_exception=True)
def production_new(request):
form = ProductionStreamSearchForm(user=request.user)
search_productionstreams_form = ProductionStreamSearchForm(user=request.user)
bulk_assign_officer_form = BulkAssignOfficersForm()
context = {
"search_productionstreams_form": form,
"search_productionstreams_form": search_productionstreams_form,
"bulk_assign_officer_form": bulk_assign_officer_form,
}
return render(request, "production/production_new.html", context)
......@@ -1327,3 +1330,104 @@ def render_stream_events(request, stream_id):
"production/_productionstream_events.html",
context,
)
def _hx_productionstream_actions_bulk_assign_officers(request):
if request.POST:
productionstream_ids = (
request.POST.getlist("productionstream-bulk-action-selected") or []
)
productionstreams = ProductionStream.objects.filter(pk__in=productionstream_ids)
form = BulkAssignOfficersForm(
request.POST,
productionstreams=productionstreams,
auto_id="productionstreams-bulk-action-form-%s",
)
else:
form = BulkAssignOfficersForm()
if form.is_valid():
# Create events, update permissions, send emails for each stream
if officer := form.cleaned_data["officer"]:
for productionstream in form.productionstreams:
old_officer = productionstream.officer
if old_officer == officer:
continue
if productionstream.status in [
constants.PRODUCTION_STREAM_INITIATED,
constants.PROOFS_SOURCE_REQUESTED,
]:
productionstream.status = constants.PROOFS_TASKED
productionstream.officer = officer
productionstream.save()
event = ProductionEvent(
stream=productionstream,
event="assignment",
comments=" tasked Production Officer with proofs production:",
noted_to=officer,
noted_by=request.user.production_user,
)
event.save()
# Update permissions
assign_perm("can_work_for_stream", officer.user, productionstream)
remove_perm("can_work_for_stream", old_officer.user, productionstream)
# Temp fix.
# TODO: Implement proper email
ProductionUtils.load({"request": request, "stream": productionstream})
ProductionUtils.email_assigned_production_officer()
if supervisor := form.cleaned_data["supervisor"]:
for productionstream in form.productionstreams:
old_supervisor = productionstream.supervisor
if old_supervisor == supervisor:
continue
productionstream.supervisor = supervisor
productionstream.save()
event = ProductionEvent(
stream=productionstream,
event="assignment",
comments=" assigned Production Supervisor:",
noted_to=supervisor,
noted_by=request.user.production_user,
)
event.save()
# Update permissions
assign_perm("can_work_for_stream", supervisor.user, productionstream)
assign_perm(
"can_perform_supervisory_actions", supervisor.user, productionstream
)
remove_perm(
"can_work_for_stream", old_supervisor.user, productionstream
)
remove_perm(
"can_perform_supervisory_actions",
old_supervisor.user,
productionstream,
)
# Temp fix.
# TODO: Implement proper email
ProductionUtils.load({"request": request, "stream": productionstream})
ProductionUtils.email_assigned_supervisor()
context = {
"form": form,
}
return render(
request,
"production/_hx_productionstream_actions_bulk_assign_officer.html",
context,
)
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment