Add basic proposal acceptance admin view

This commit is contained in:
Stefan Wehrmeyer 2022-03-18 19:04:12 +01:00
parent 7248b40a60
commit b22cdef2b8
5 changed files with 291 additions and 3 deletions

View file

@ -1,16 +1,21 @@
from django.contrib import admin, auth
from django.contrib.auth.models import Group
from django.urls import reverse_lazy
from django.shortcuts import get_object_or_404, redirect, render
from django.urls import path, reverse, reverse_lazy
from django.utils.translation import gettext_lazy as _
from adminsortable2.admin import SortableAdminMixin
from froide.follow.admin import FollowerAdmin
from froide.helper.admin_utils import make_choose_object_action
from froide.helper.admin_utils import make_choose_object_action, make_emptyfilter
from froide.helper.widgets import TagAutocompleteWidget
from froide.organization.models import Organization
from .forms import GovernmentPlanForm, GovernmentPlanUpdateForm
from .forms import (
GovernmentPlanForm,
GovernmentPlanUpdateAcceptProposalForm,
GovernmentPlanUpdateForm,
)
from .models import (
Government,
GovernmentPlan,
@ -113,6 +118,17 @@ class GovernmentPlanAdmin(admin.ModelAdmin):
actions.update(admin_actions)
return actions
def get_urls(self):
urls = super().get_urls()
my_urls = [
path(
"<int:pk>/accept-proposal/",
self.admin_site.admin_view(self.accept_proposal),
name="froide_govplan-plan_accept_proposal",
),
]
return my_urls + urls
def get_list_display(self, request):
list_display = [
"title",
@ -135,6 +151,9 @@ class GovernmentPlanAdmin(admin.ModelAdmin):
if not has_limited_access(request.user):
list_filter.extend(
[
make_emptyfilter(
"proposals", _("Has change proposals"), empty_value=None
),
"organization",
"group",
"government",
@ -173,6 +192,59 @@ class GovernmentPlanAdmin(admin.ModelAdmin):
make_public.short_description = _("Make public")
def accept_proposal(self, request, pk):
obj = get_object_or_404(self.get_queryset(request), pk=pk)
plan_url = reverse(
"admin:froide_govplan_governmentplan_change",
args=(obj.pk,),
current_app=self.admin_site.name,
)
if not obj.proposals:
return redirect(plan_url)
if request.method == "POST":
proposals = obj.proposals or {}
proposal_id = request.POST.get("proposal_id")
data = proposals[proposal_id]["data"]
form = GovernmentPlanUpdateAcceptProposalForm(data=data, plan=obj)
if form.is_valid():
update = form.save(
delete_unconfirmed=request.POST.get("delete", "0") == "1",
delete_reason=request.POST.get("delete_reason", ""),
proposal_id=proposal_id,
delete_proposals=request.POST.getlist("proposal_delete"),
)
if update is None:
self.message_user(request, _("The proposal has been deleted."))
return redirect(plan_url)
self.message_user(
request,
_("An unpublished update has been created."),
)
update_url = reverse(
"admin:froide_govplan_governmentplanupdate_change",
args=(update.pk,),
current_app=self.admin_site.name,
)
return redirect(update_url)
else:
form = GovernmentPlanUpdateAcceptProposalForm(plan=obj)
opts = self.model._meta
context = {
"form": form,
"proposals": form.get_proposals(),
"object": obj,
"app_label": opts.app_label,
"opts": opts,
}
return render(
request,
"froide_govplan/admin/accept_proposal.html",
context,
)
class GovernmentPlanUpdateAdmin(admin.ModelAdmin):
form = GovernmentPlanUpdateForm

View file

@ -1,4 +1,5 @@
from django import forms
from django.contrib.auth import get_user_model
from django.utils import timezone
from django.utils.safestring import mark_safe
from django.utils.translation import gettext_lazy as _
@ -132,3 +133,95 @@ class GovernmentPlanUpdateProposalForm(forms.ModelForm):
}
plan.save(update_fields=["proposals"])
return plan
class GovernmentPlanUpdateAcceptProposalForm(GovernmentPlanUpdateProposalForm):
def __init__(self, *args, **kwargs):
self.plan = kwargs.pop("plan")
super().__init__(*args, **kwargs)
def get_proposals(self):
data = dict(self.plan.proposals)
user_ids = self.plan.proposals.keys()
user_map = {
str(u.id): u for u in get_user_model().objects.filter(id__in=user_ids)
}
status_dict = dict(PlanStatus.choices)
rating_dict = dict(PlanRating.choices)
for user_id, v in data.items():
v["user"] = user_map[user_id]
data[user_id]["data"]["rating_label"] = rating_dict.get(
data[user_id]["data"]["rating"]
)
data[user_id]["data"]["status_label"] = status_dict.get(
data[user_id]["data"]["status"]
)
return data
def save(
self,
proposal_id=None,
delete_proposals=None,
delete_unconfirmed=False,
delete_reason="",
):
import ipdb
ipdb.set_trace()
update = super(forms.ModelForm, self).save(commit=False)
update.plan = self.plan
if delete_proposals is None:
delete_proposals = []
if proposal_id:
proposals = self.get_proposals()
proposal_user = proposals[proposal_id]["user"]
update.user = proposal_user
# if proposal_user != user:
# proposal_user.send_mail(
# _("Changes to public body “{}” have been applied").format(pb.name),
# _(
# "Hello,\n\nYou can find the changed public body here:"
# "\n\n{url}\n\nAll the Best,\n{site_name}"
# ).format(
# url=pb.get_absolute_domain_url(), site_name=settings.SITE_NAME
# ),
# priority=False,
# )
delete_proposals.append(proposal_id)
for pid in delete_proposals:
if pid in self.plan.proposals:
del self.plan.proposals[pid]
if not self.plan.proposals:
self.plan.proposals = None
self.plan.save(update_fields=["proposals"])
# if delete_unconfirmed:
# self.delete_proposal(pb, user, delete_reason)
# return None
update.save()
# PublicBody.change_proposal_accepted.send(sender=pb, user=user)
return update
# def delete_proposal(self, pb, user, delete_reason=""):
# LogEntry.objects.log_action(
# user_id=user.id,
# content_type_id=ContentType.objects.get_for_model(pb).pk,
# object_id=pb.pk,
# object_repr=str(pb),
# action_flag=DELETION,
# )
# creator = pb.created_by
# if creator:
# creator.send_mail(
# _("Your public body proposal “%s” was rejected") % pb.name,
# _(
# "Hello,\n\nA moderator has rejected your proposal for a new "
# "public body.\n\n{delete_reason}\n\nAll the Best,\n{site_name}"
# ).format(delete_reason=delete_reason, site_name=settings.SITE_NAME),
# priority=False,
# )
# PublicBody.proposal_rejected.send(sender=pb, user=user)
# pb.delete()

View file

@ -0,0 +1,13 @@
{% extends "admin/change_form.html" %}
{% load i18n %}
{% block object-tools-items %}
{% if original.pk and original.proposals %}
<li>
<a href="{% url 'admin:froide_govplan-plan_accept_proposal' pk=original.pk %}">
{% trans "See update proposals" %}
</a>
</li>
{% endif %}
{{ block.super }}
{% endblock %}

View file

@ -0,0 +1,77 @@
{% load i18n %}
{% load form_helper %}
{% load permission_helper %}
<div class="table-responsive">
<table class="table">
<thead>
<tr>
<th>{% trans "Field" %}</th>
{% for key, proposal in proposals.items %}
<th>
<span title="{{ proposal.timestamp }}">
{% if request.user.is_staff and request.user|has_perm:"account.view_user" %}
{{ proposal.user.get_full_name }} ({{ proposal.user.email }})
{% else %}
{{ proposal.timestamp | date:"SHORT_DATETIME" }}
{% endif %}
</span>
</th>
{% endfor %}
</tr>
</thead>
<tbody>
{% if proposals %}
<tr>
<td>
{% trans "Proposal" %}
</td>
{% for key, proposal in proposals.items %}
<td>
<label>
<input type="radio" name="proposal_id" class="proposal" value="{{ key }}" required>
{% trans "Turn into update draft" %}
</label>
</td>
{% endfor %}
</tr>
{% endif %}
{% for field in form %}
<tr>
<td>{{ field.label }}</td>
{% for key, proposal in proposals.items %}
{% with val=proposal.data|get_item_by_key:field.name %}
<td>
{% with label_name=field.name|add:"_label" %}
{% if proposal.data|get_item_by_key:label_name %}
{{ proposal.data|get_item_by_key:label_name }}
{% else %}
{{ val|urlize|linebreaksbr }}
{% endif %}
{% endwith %}
</td>
{% endwith %}
{% endfor %}
</tr>
{{ form.field }}
{% endfor %}
{% if proposals %}
<tr>
<td>
{% trans "Delete proposal" %}
</td>
{% for key, proposal in proposals.items %}
<td>
<label>
<input type="checkbox" name="proposal_delete" value="{{ proposal.user.id }}">
{% trans "Delete" %}
</label>
</td>
{% endfor %}
</tr>
{% endif %}
</tbody>
</table>
</div>

View file

@ -0,0 +1,33 @@
{% extends "admin/change_form.html" %}
{% load i18n %}
{% block title %}{{ object.title }} - {{ block.super }}{% endblock %}
{% block content %}<div id="content-main">
<h3>{{ object.title }}</h3>
<form action="" method="post">
{% csrf_token %}
<fieldset>
{% include "froide_govplan/admin/_proposal.html" %}
</fieldset>
<div class="submit-row">
<input type="submit" value="{% trans 'Create update draft' %}" style="float:left"/>
</div>
<div class="card border-danger mt-3">
<div class="card-body">
<h3>{% trans "Delete proposal" %}</h3>
<p><label for="delete_reason">
{% trans "Please give a reason for not accepting this proposal. It will be send to the user who made the proposal." %}
</label>
<p>
<p><textarea id="delete_reason" class="form-control" name="delete_reason" placeholder="{% trans 'We could not accept this update because...' %}"></textarea></p>
<p><button class="mt-2 btn btn-danger float-right" type="submit" name="delete" value="1">
{% trans "Delete proposed update" %}
</button></p>
</div>
</div>
</form>
{% endblock %}