* introduces BaseDetailView * refactors detail views for EIV, KOM, OEK, EMA from function based to class based * refactors already class based HomeView to inherit from new BaseView
233 lines
7.4 KiB
Python
233 lines
7.4 KiB
Python
"""
|
|
Author: Michel Peltriaux
|
|
Organization: Struktur- und Genehmigungsdirektion Nord, Rhineland-Palatinate, Germany
|
|
Contact: ksp-servicestelle@sgdnord.rlp.de
|
|
Created on: 19.08.22
|
|
|
|
"""
|
|
from django.contrib import messages
|
|
from django.contrib.auth.decorators import login_required
|
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
|
from django.http import HttpRequest
|
|
from django.shortcuts import get_object_or_404, redirect, render
|
|
from django.urls import reverse
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from ema.forms import NewEmaForm, EditEmaForm
|
|
from ema.models import Ema
|
|
from ema.tables import EmaTable
|
|
from konova.contexts import BaseContext
|
|
from konova.decorators import shared_access_required, conservation_office_group_required, login_required_modal
|
|
from konova.forms import SimpleGeomForm
|
|
from konova.forms.modals import RemoveModalForm
|
|
from konova.sub_settings.context_settings import TAB_TITLE_IDENTIFIER
|
|
from konova.utils.message_templates import RECORDED_BLOCKS_EDIT, IDENTIFIER_REPLACED, FORM_INVALID, \
|
|
GEOMETRY_SIMPLIFIED, GEOMETRIES_IGNORED_TEMPLATE
|
|
from konova.views.base import BaseIndexView, BaseIdentifierGeneratorView
|
|
from konova.views.detail import BaseDetailView
|
|
|
|
|
|
class EmaIndexView(LoginRequiredMixin, BaseIndexView):
|
|
_TAB_TITLE = _("EMAs - Overview")
|
|
_INDEX_TABLE_CLS = EmaTable
|
|
|
|
def _get_queryset(self):
|
|
qs = Ema.objects.filter(
|
|
deleted=None,
|
|
).order_by(
|
|
"-modified__timestamp"
|
|
)
|
|
return qs
|
|
|
|
|
|
@login_required
|
|
@conservation_office_group_required
|
|
def new_view(request: HttpRequest):
|
|
"""
|
|
Renders a view for a new eco account creation
|
|
|
|
Args:
|
|
request (HttpRequest): The incoming request
|
|
|
|
Returns:
|
|
|
|
"""
|
|
template = "ema/form/view.html"
|
|
data_form = NewEmaForm(request.POST or None)
|
|
geom_form = SimpleGeomForm(request.POST or None, read_only=False)
|
|
if request.method == "POST":
|
|
if data_form.is_valid() and geom_form.is_valid():
|
|
generated_identifier = data_form.cleaned_data.get("identifier", None)
|
|
ema = data_form.save(request.user, geom_form)
|
|
if generated_identifier != ema.identifier:
|
|
messages.info(
|
|
request,
|
|
IDENTIFIER_REPLACED.format(
|
|
generated_identifier,
|
|
ema.identifier
|
|
)
|
|
)
|
|
messages.success(request, _("EMA {} added").format(ema.identifier))
|
|
if geom_form.has_geometry_simplified():
|
|
messages.info(
|
|
request,
|
|
GEOMETRY_SIMPLIFIED
|
|
)
|
|
num_ignored_geometries = geom_form.get_num_geometries_ignored()
|
|
if num_ignored_geometries > 0:
|
|
messages.info(
|
|
request,
|
|
GEOMETRIES_IGNORED_TEMPLATE.format(num_ignored_geometries)
|
|
)
|
|
|
|
return redirect("ema:detail", id=ema.id)
|
|
else:
|
|
messages.error(request, FORM_INVALID, extra_tags="danger",)
|
|
else:
|
|
# For clarification: nothing in this case
|
|
pass
|
|
context = {
|
|
"form": data_form,
|
|
"geom_form": geom_form,
|
|
TAB_TITLE_IDENTIFIER: _("New EMA"),
|
|
}
|
|
context = BaseContext(request, context).context
|
|
return render(request, template, context)
|
|
|
|
|
|
class EmaIdentifierGeneratorView(LoginRequiredMixin, BaseIdentifierGeneratorView):
|
|
_MODEL_CLS = Ema
|
|
_REDIRECT_URL = "ema:index"
|
|
|
|
def _user_has_permission(self, user):
|
|
return user.is_ets_user()
|
|
|
|
|
|
class EmaDetailView(BaseDetailView):
|
|
_MODEL_CLS = Ema
|
|
_TEMPLATE = "ema/detail/view.html"
|
|
|
|
def _get_object(self, id: str):
|
|
""" Fetch object for detail view
|
|
|
|
Args:
|
|
id (str): The record's id'
|
|
|
|
Returns:
|
|
|
|
"""
|
|
ema = get_object_or_404(Ema, id=id, deleted=None)
|
|
return ema
|
|
|
|
def _get_detail_context(self, obj: Ema):
|
|
""" Generate object specific detail context for view
|
|
|
|
Args:
|
|
obj (): The record
|
|
|
|
Returns:
|
|
|
|
"""
|
|
# Order states according to surface
|
|
before_states = obj.before_states.all().order_by("-surface")
|
|
after_states = obj.after_states.all().order_by("-surface")
|
|
|
|
# Precalculate logical errors between before- and after-states
|
|
# Sum() returns None in case of no states, so we catch that and replace it with 0 for easier handling
|
|
sum_before_states = obj.get_surface_before_states()
|
|
sum_after_states = obj.get_surface_after_states()
|
|
diff_states = abs(sum_before_states - sum_after_states)
|
|
|
|
context = {
|
|
"before_states": before_states,
|
|
"after_states": after_states,
|
|
"sum_before_states": sum_before_states,
|
|
"sum_after_states": sum_after_states,
|
|
"diff_states": diff_states,
|
|
"has_finished_deadlines": obj.get_finished_deadlines().exists(),
|
|
}
|
|
return context
|
|
|
|
|
|
@login_required
|
|
@conservation_office_group_required
|
|
@shared_access_required(Ema, "id")
|
|
def edit_view(request: HttpRequest, id: str):
|
|
"""
|
|
Renders a view for editing compensations
|
|
|
|
Args:
|
|
request (HttpRequest): The incoming request
|
|
|
|
Returns:
|
|
|
|
"""
|
|
template = "compensation/form/view.html"
|
|
# Get object from db
|
|
ema = get_object_or_404(Ema, id=id)
|
|
if ema.is_recorded:
|
|
messages.info(
|
|
request,
|
|
RECORDED_BLOCKS_EDIT
|
|
)
|
|
return redirect("ema:detail", id=id)
|
|
|
|
# Create forms, initialize with values from db/from POST request
|
|
data_form = EditEmaForm(request.POST or None, instance=ema)
|
|
geom_form = SimpleGeomForm(request.POST or None, read_only=False, instance=ema)
|
|
if request.method == "POST":
|
|
if data_form.is_valid() and geom_form.is_valid():
|
|
# The data form takes the geom form for processing, as well as the performing user
|
|
ema = data_form.save(request.user, geom_form)
|
|
messages.success(request, _("EMA {} edited").format(ema.identifier))
|
|
if geom_form.has_geometry_simplified():
|
|
messages.info(
|
|
request,
|
|
GEOMETRY_SIMPLIFIED
|
|
)
|
|
|
|
num_ignored_geometries = geom_form.get_num_geometries_ignored()
|
|
if num_ignored_geometries > 0:
|
|
messages.info(
|
|
request,
|
|
GEOMETRIES_IGNORED_TEMPLATE.format(num_ignored_geometries)
|
|
)
|
|
|
|
return redirect("ema:detail", id=ema.id)
|
|
else:
|
|
messages.error(request, FORM_INVALID, extra_tags="danger",)
|
|
else:
|
|
# For clarification: nothing in this case
|
|
pass
|
|
context = {
|
|
"form": data_form,
|
|
"geom_form": geom_form,
|
|
TAB_TITLE_IDENTIFIER: _("Edit {}").format(ema.identifier),
|
|
}
|
|
context = BaseContext(request, context).context
|
|
return render(request, template, context)
|
|
|
|
|
|
@login_required_modal
|
|
@login_required
|
|
@conservation_office_group_required
|
|
@shared_access_required(Ema, "id")
|
|
def remove_view(request: HttpRequest, id: str):
|
|
""" Renders a modal view for removing the EMA
|
|
|
|
Args:
|
|
request (HttpRequest): The incoming request
|
|
id (str): The EMA's id
|
|
|
|
Returns:
|
|
|
|
"""
|
|
ema = get_object_or_404(Ema, id=id)
|
|
form = RemoveModalForm(request.POST or None, instance=ema, request=request)
|
|
return form.process_request(
|
|
request=request,
|
|
msg_success=_("EMA removed"),
|
|
redirect_url=reverse("ema:index"),
|
|
)
|
|
|