mirror of
https://github.com/ae-utbm/sith.git
synced 2025-10-14 16:58:31 +00:00
Compare commits
37 Commits
dependabot
...
refactor-e
Author | SHA1 | Date | |
---|---|---|---|
|
6b5268c87d | ||
|
78da1eebc7 | ||
|
856e872641 | ||
|
13e5edab08 | ||
|
f6c2762a4e | ||
|
b6209dc9b1 | ||
|
308dd4b56f | ||
|
289ffe1109 | ||
|
ff5bb04af1 | ||
ca50e5dc81
|
|||
|
f015bde768 | ||
bb09fd0feb
|
|||
210278440a
|
|||
e041da9cf4
|
|||
54c1957776
|
|||
30356d97f3
|
|||
7eaf25a64f
|
|||
c6e86841b3
|
|||
cbe9887efb
|
|||
|
980952807a | ||
|
0b7c516f18 | ||
|
e186052283 | ||
|
ec80b72a25 | ||
|
6cd3875b2b | ||
ad8b003336
|
|||
|
b4f5a866e3 | ||
d87b069769
|
|||
|
9461b2e5d9 | ||
4701c0804b
|
|||
|
acb6c6ce9c | ||
95e6fff98b
|
|||
|
f1a5a0781c | ||
|
854dd2d9e7 | ||
|
a7c96425c8 | ||
dff23fae7f
|
|||
|
34b0dc3302 | ||
|
ce2ef78a6d |
@@ -83,9 +83,10 @@ TODO : rewrite the pagination used in this template an Alpine one
|
||||
</table>
|
||||
<script type="text/javascript">
|
||||
function formPagination(link){
|
||||
$("form").attr("action", link.href);
|
||||
const form = document.getElementById("form")
|
||||
form.action = link.href;
|
||||
link.href = "javascript:void(0)"; // block link action
|
||||
$("form").submit();
|
||||
form.submit();
|
||||
}
|
||||
</script>
|
||||
{{ paginate(paginated_result, paginator, "formPagination(this)") }}
|
||||
|
@@ -344,7 +344,7 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
||||
form = self.get_form()
|
||||
if form.is_valid():
|
||||
if not len([v for v in form.cleaned_data.values() if v is not None]):
|
||||
qs = Selling.objects.filter(id=-1)
|
||||
qs = Selling.objects.none()
|
||||
if form.cleaned_data["begin_date"]:
|
||||
qs = qs.filter(date__gte=form.cleaned_data["begin_date"])
|
||||
if form.cleaned_data["end_date"]:
|
||||
@@ -362,7 +362,9 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
||||
if len(selected_products) > 0:
|
||||
qs = qs.filter(product__in=selected_products)
|
||||
|
||||
kwargs["result"] = qs.all().order_by("-id")
|
||||
kwargs["result"] = qs.select_related(
|
||||
"counter", "counter__club", "customer", "customer__user", "seller"
|
||||
).order_by("-id")
|
||||
kwargs["total"] = sum([s.quantity * s.unit_price for s in kwargs["result"]])
|
||||
total_quantity = qs.all().aggregate(Sum("quantity"))
|
||||
if total_quantity["quantity__sum"]:
|
||||
|
49
com/static/bundled/com/slideshow-index.ts
Normal file
49
com/static/bundled/com/slideshow-index.ts
Normal file
@@ -0,0 +1,49 @@
|
||||
const INTERVAL = 10;
|
||||
|
||||
interface Poster {
|
||||
url: string; // URL of the poster
|
||||
displayTime: number; // Number of seconds to display that poster
|
||||
}
|
||||
|
||||
document.addEventListener("alpine:init", () => {
|
||||
Alpine.data("slideshow", (posters: Poster[]) => ({
|
||||
posters: posters,
|
||||
progress: 0,
|
||||
elapsed: 0,
|
||||
|
||||
current: 0,
|
||||
previous: 0,
|
||||
|
||||
init() {
|
||||
this.$watch("elapsed", () => {
|
||||
const displayTime = this.posters[this.current].displayTime * 1000;
|
||||
if (this.elapsed > displayTime) {
|
||||
this.previous = this.current;
|
||||
this.current = this.getNext();
|
||||
this.elapsed = 0;
|
||||
}
|
||||
if (displayTime === 0) {
|
||||
this.progress = 100;
|
||||
} else {
|
||||
this.progress = (100 * this.elapsed) / displayTime;
|
||||
}
|
||||
});
|
||||
setInterval(() => {
|
||||
this.elapsed += INTERVAL;
|
||||
}, INTERVAL);
|
||||
},
|
||||
|
||||
getNext() {
|
||||
return (this.current + 1) % this.posters.length;
|
||||
},
|
||||
|
||||
async toggleFullScreen(event: Event) {
|
||||
if (document.fullscreenElement) {
|
||||
await document.exitFullscreen();
|
||||
return;
|
||||
}
|
||||
const target = event.target as HTMLElement;
|
||||
await target.requestFullscreen();
|
||||
},
|
||||
}));
|
||||
});
|
@@ -111,7 +111,7 @@
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: 10;
|
||||
content: "Click to expand";
|
||||
content: attr(hover);
|
||||
color: white;
|
||||
background-color: rgba(black, 0.5);
|
||||
}
|
||||
|
@@ -1,23 +0,0 @@
|
||||
$(document).ready(() => {
|
||||
$("#poster_list #view").click(() => {
|
||||
$("#view").removeClass("active");
|
||||
});
|
||||
|
||||
$("#poster_list .poster .image").click((e) => {
|
||||
let el = $(e.target);
|
||||
if (el.hasClass("image")) {
|
||||
el = el.find("img");
|
||||
}
|
||||
$("#poster_list #view #placeholder").html(el.clone());
|
||||
|
||||
$("#view").addClass("active");
|
||||
});
|
||||
|
||||
$(document).keyup((e) => {
|
||||
if (e.keyCode === 27) {
|
||||
// escape key maps to keycode `27`
|
||||
e.preventDefault();
|
||||
$("#view").removeClass("active");
|
||||
}
|
||||
});
|
||||
});
|
@@ -1,98 +0,0 @@
|
||||
$(document).ready(() => {
|
||||
const transitionTime = 1000;
|
||||
|
||||
let i = 0;
|
||||
const max = $("#slideshow .slide").length;
|
||||
|
||||
function enterFullscreen() {
|
||||
const element = document.getElementById("slideshow");
|
||||
$(element).addClass("fullscreen");
|
||||
if (element.requestFullscreen) {
|
||||
element.requestFullscreen();
|
||||
} else if (element.mozRequestFullScreen) {
|
||||
element.mozRequestFullScreen();
|
||||
} else if (element.webkitRequestFullscreen) {
|
||||
element.webkitRequestFullscreen();
|
||||
} else if (element.msRequestFullscreen) {
|
||||
element.msRequestFullscreen();
|
||||
}
|
||||
}
|
||||
|
||||
function exitFullscreen() {
|
||||
const element = document.getElementById("slideshow");
|
||||
$(element).removeClass("fullscreen");
|
||||
if (document.exitFullscreen) {
|
||||
document.exitFullscreen();
|
||||
} else if (document.webkitExitFullscreen) {
|
||||
document.webkitExitFullscreen();
|
||||
} else if (document.mozCancelFullScreen) {
|
||||
document.mozCancelFullScreen();
|
||||
} else if (document.msExitFullscreen) {
|
||||
document.msExitFullscreen();
|
||||
}
|
||||
}
|
||||
|
||||
function initProgressBar() {
|
||||
$("#slideshow #progress_bar").css("transition", "none");
|
||||
$("#slideshow #progress_bar").removeClass("progress");
|
||||
$("#slideshow #progress_bar").addClass("init");
|
||||
}
|
||||
|
||||
function startProgressBar(displayTime) {
|
||||
$("#slideshow #progress_bar").removeClass("init");
|
||||
$("#slideshow #progress_bar").addClass("progress");
|
||||
$("#slideshow #progress_bar").css("transition", `width ${displayTime}s linear`);
|
||||
}
|
||||
|
||||
function next() {
|
||||
initProgressBar();
|
||||
const slide = $($("#slideshow .slide").get(i % max));
|
||||
slide.removeClass("center");
|
||||
slide.addClass("left");
|
||||
|
||||
const nextSlide = $($("#slideshow .slide").get((i + 1) % max));
|
||||
nextSlide.removeClass("right");
|
||||
nextSlide.addClass("center");
|
||||
const displayTime = nextSlide.attr("display_time") || 2;
|
||||
|
||||
$("#slideshow .bullet").removeClass("active");
|
||||
const bullet = $("#slideshow .bullet")[(i + 1) % max];
|
||||
$(bullet).addClass("active");
|
||||
|
||||
i = (i + 1) % max;
|
||||
|
||||
setTimeout(() => {
|
||||
const othersLeft = $("#slideshow .slide.left");
|
||||
othersLeft.removeClass("left");
|
||||
othersLeft.addClass("right");
|
||||
|
||||
startProgressBar(displayTime);
|
||||
setTimeout(next, displayTime * 1000);
|
||||
}, transitionTime);
|
||||
}
|
||||
|
||||
const displayTime = $("#slideshow .center").attr("display_time");
|
||||
initProgressBar();
|
||||
setTimeout(() => {
|
||||
if (max > 1) {
|
||||
startProgressBar(displayTime);
|
||||
setTimeout(next, displayTime * 1000);
|
||||
}
|
||||
}, 10);
|
||||
|
||||
$("#slideshow").click(() => {
|
||||
if ($("#slideshow").hasClass("fullscreen")) {
|
||||
exitFullscreen();
|
||||
} else {
|
||||
enterFullscreen();
|
||||
}
|
||||
});
|
||||
|
||||
$(document).keyup((e) => {
|
||||
if (e.keyCode === 27) {
|
||||
// escape key maps to keycode `27`
|
||||
e.preventDefault();
|
||||
exitFullscreen();
|
||||
}
|
||||
});
|
||||
});
|
@@ -34,7 +34,7 @@ body{
|
||||
|
||||
z-index: 10;
|
||||
|
||||
content: "Click to expand";
|
||||
content: attr(hover);
|
||||
|
||||
color: white;
|
||||
background-color: rgba(black, 0.5);
|
||||
@@ -43,7 +43,7 @@ body{
|
||||
|
||||
}
|
||||
|
||||
&.fullscreen{
|
||||
&:fullscreen {
|
||||
position: fixed;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
@@ -65,38 +65,59 @@ body{
|
||||
position: relative;
|
||||
height: 100%;
|
||||
overflow: hidden;
|
||||
background-color: grey;
|
||||
|
||||
.slide {
|
||||
position: absolute;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
|
||||
display: inline-flex;
|
||||
display: none;
|
||||
justify-content: center;
|
||||
|
||||
top: 0px;
|
||||
|
||||
background-color: grey;
|
||||
transition: left 1s ease-out;
|
||||
left: 0%;
|
||||
|
||||
img {
|
||||
max-width: 100%;
|
||||
max-height: 100%;
|
||||
object-fit: contain;
|
||||
}
|
||||
|
||||
&.current {
|
||||
display: inline-flex;
|
||||
left: 0%;
|
||||
animation: scrolling-in 1s linear;
|
||||
}
|
||||
|
||||
.slide.left{
|
||||
left: -100%;
|
||||
&.previous {
|
||||
display: inline-flex;
|
||||
animation: scrolling-out 1s linear;
|
||||
opacity: 0;
|
||||
transition: opacity 0.1s;
|
||||
transition-delay: 0.9s;
|
||||
}
|
||||
|
||||
.slide.center{
|
||||
left: 0px;
|
||||
@keyframes scrolling-in {
|
||||
0% {
|
||||
transform: translateX(100%);
|
||||
}
|
||||
|
||||
100% {
|
||||
transform: translateX(0%);
|
||||
}
|
||||
}
|
||||
|
||||
@keyframes scrolling-out {
|
||||
0% {
|
||||
transform: translateX(0%);
|
||||
}
|
||||
|
||||
100% {
|
||||
transform: translateX(-100%);
|
||||
}
|
||||
}
|
||||
|
||||
.slide.right{
|
||||
left: 100%;
|
||||
transition: none;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -129,21 +150,27 @@ body{
|
||||
}
|
||||
}
|
||||
|
||||
#progress_bar{
|
||||
progress {
|
||||
--color: #304c83;
|
||||
|
||||
position: absolute;
|
||||
bottom: 0px;
|
||||
height: 10px;
|
||||
background-color: #304c83;
|
||||
|
||||
&.init{
|
||||
width: 0px;
|
||||
transition: none;
|
||||
}
|
||||
|
||||
&.progress{
|
||||
color: var(--color);
|
||||
width: 100%;
|
||||
transition: width 10s linear;
|
||||
}
|
||||
}
|
||||
margin-bottom: 0px;
|
||||
border: none;
|
||||
|
||||
&::-moz-progress-bar {
|
||||
background: var(--color);
|
||||
}
|
||||
|
||||
&::-webkit-progress-value {
|
||||
background: var(--color);
|
||||
}
|
||||
|
||||
&[value] {
|
||||
background-color: transparent;
|
||||
}
|
||||
}
|
||||
}
|
@@ -1,11 +1,5 @@
|
||||
{% extends "core/base.jinja" %}
|
||||
|
||||
{% block script %}
|
||||
{{ super() }}
|
||||
<script src="{{ static('com/js/poster_list.js') }}"></script>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block title %}
|
||||
{% trans %}Poster{% endtrans %}
|
||||
{% endblock %}
|
||||
@@ -15,7 +9,7 @@
|
||||
{% endblock %}
|
||||
|
||||
{% block content %}
|
||||
<div id="poster_list">
|
||||
<div id="poster_list" x-data="{ active: null }">
|
||||
|
||||
<div id="title">
|
||||
<h3>{% trans %}Posters{% endtrans %}</h3>
|
||||
@@ -38,7 +32,13 @@
|
||||
{% for poster in poster_list %}
|
||||
<div class="poster{% if not poster.is_moderated %} not_moderated{% endif %}">
|
||||
<div class="name">{{ poster.name }}</div>
|
||||
<div class="image"><img src="{{ poster.file.url }}"></img></div>
|
||||
<div
|
||||
class="image"
|
||||
hover="{% trans %}Click to expand{% endtrans %}"
|
||||
@click="active = $el.firstElementChild"
|
||||
>
|
||||
<img src="{{ poster.file.url }}"></img>
|
||||
</div>
|
||||
<div class="dates">
|
||||
<div class="begin">{{ poster.date_begin | localtime | date("d/M/Y H:m") }}</div>
|
||||
<div class="end">{{ poster.date_end | localtime | date("d/M/Y H:m") }}</div>
|
||||
@@ -62,7 +62,14 @@
|
||||
|
||||
</div>
|
||||
|
||||
<div id="view"><div id="placeholder"></div></div>
|
||||
<div
|
||||
id="view"
|
||||
@keyup.escape.window="active = null"
|
||||
@click="active = null"
|
||||
:class="{active: active !== null}"
|
||||
>
|
||||
<div id="placeholder"><img :src="active?.src"></div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
@@ -2,28 +2,44 @@
|
||||
<html lang="fr">
|
||||
<head>
|
||||
<title>{% trans %}Slideshow{% endtrans %}</title>
|
||||
<link rel="shortcut icon" href="{{ static('core/img/favicon.ico') }}">
|
||||
<link href="{{ static('css/slideshow.scss') }}" rel="stylesheet" type="text/css" />
|
||||
<script src="{{ static('bundled/vendored/jquery.min.js') }}"></script>
|
||||
<script src="{{ static('com/js/slideshow.js') }}"></script>
|
||||
<script type="module" src="{{ static('bundled/alpine-index.js') }}"></script>
|
||||
<script type="module" src="{{ static('bundled/com/slideshow-index.ts') }}"></script>
|
||||
</head>
|
||||
<body>
|
||||
<div id="slideshow">
|
||||
<body x-data="slideshow([
|
||||
{% for poster in posters %}
|
||||
{
|
||||
url: '{{ poster.file.url }}',
|
||||
displayTime: {{ poster.display_time }}
|
||||
},
|
||||
{% endfor %}
|
||||
])">
|
||||
<div
|
||||
id="slideshow"
|
||||
@click="toggleFullScreen"
|
||||
hover="{% trans %}Click to expand{% endtrans %}"
|
||||
@keyup.f.window="toggleFullScreen"
|
||||
>
|
||||
|
||||
<div id="slides">
|
||||
{% for poster in posters %}
|
||||
<div class="slide {% if loop.first %}center{% else %}right{% endif %}" display_time="{{ poster.display_time }}">
|
||||
<img src="{{ poster.file.url }}">
|
||||
<template x-for="(poster, index) in posters">
|
||||
<div class="slide" :class="{
|
||||
current: index === current,
|
||||
previous: index !== current && index === previous,
|
||||
}">
|
||||
<img :src="poster.url">
|
||||
</div>
|
||||
{% endfor %}
|
||||
</template>
|
||||
</div>
|
||||
|
||||
<div id="progress_bullets">
|
||||
{% for poster in posters %}
|
||||
<div class="bullet {% if loop.first %}active{% endif %}"></div>
|
||||
{% endfor %}
|
||||
<template x-for="(poster, index) in posters">
|
||||
<div class="bullet" :class="{active: current === index}"></div>
|
||||
</template>
|
||||
</div>
|
||||
|
||||
<div id="progress_bar"></div>
|
||||
<progress :value="progress" max="100" x-show="posters.length > 1 && progress > 0"></progress>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
|
@@ -31,9 +31,7 @@
|
||||
<td>
|
||||
<a href="{{ url('com:weekmail_article_edit', article_id=a.id) }}">{% trans %}Edit{% endtrans %}</a> |
|
||||
<a href="{{ url('com:weekmail_article_delete', article_id=a.id) }}">{% trans %}Delete{% endtrans %}</a> |
|
||||
<a href="?add_article={{ a.id }}">{% trans %}Add to weekmail{% endtrans %}</a> |
|
||||
<a href="?up_article={{ a.id }}">{% trans %}Up{% endtrans %}</a> |
|
||||
<a href="?down_article={{ a.id }}">{% trans %}Down{% endtrans %}</a>
|
||||
<a href="?add_article={{ a.id }}">{% trans %}Add to weekmail{% endtrans %}</a>
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
|
41
com/views.py
41
com/views.py
@@ -28,6 +28,7 @@ from typing import Any
|
||||
|
||||
from dateutil.relativedelta import relativedelta
|
||||
from django.conf import settings
|
||||
from django.contrib import messages
|
||||
from django.contrib.auth.mixins import (
|
||||
PermissionRequiredMixin,
|
||||
)
|
||||
@@ -55,7 +56,7 @@ from core.auth.mixins import (
|
||||
PermissionOrClubBoardRequiredMixin,
|
||||
)
|
||||
from core.models import User
|
||||
from core.views.mixins import QuickNotifMixin, TabedViewMixin
|
||||
from core.views.mixins import TabedViewMixin
|
||||
from core.views.widgets.markdown import MarkdownInput
|
||||
|
||||
# Sith object
|
||||
@@ -333,7 +334,7 @@ class NewsFeed(Feed):
|
||||
# Weekmail
|
||||
|
||||
|
||||
class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, DetailView):
|
||||
class WeekmailPreviewView(ComTabsMixin, CanEditPropMixin, DetailView):
|
||||
model = Weekmail
|
||||
template_name = "com/weekmail_preview.jinja"
|
||||
success_url = reverse_lazy("com:weekmail")
|
||||
@@ -345,12 +346,11 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai
|
||||
|
||||
def post(self, request, *args, **kwargs):
|
||||
self.object = self.get_object()
|
||||
messages.success(self.request, _("Weekmail sent successfully"))
|
||||
if request.POST["send"] == "validate":
|
||||
try:
|
||||
self.object.send()
|
||||
return HttpResponseRedirect(
|
||||
reverse("com:weekmail") + "?qn_weekmail_send_success"
|
||||
)
|
||||
return HttpResponseRedirect(reverse("com:weekmail"))
|
||||
except SMTPRecipientsRefused as e:
|
||||
self.bad_recipients = e.recipients
|
||||
elif request.POST["send"] == "clean":
|
||||
@@ -361,7 +361,6 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai
|
||||
for u in users:
|
||||
u.preferences.receive_weekmail = False
|
||||
u.preferences.save()
|
||||
self.quick_notif_list += ["qn_success"]
|
||||
return super().get(request, *args, **kwargs)
|
||||
|
||||
def get_object(self, queryset=None):
|
||||
@@ -375,7 +374,7 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai
|
||||
return kwargs
|
||||
|
||||
|
||||
class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView):
|
||||
class WeekmailEditView(ComTabsMixin, CanEditPropMixin, UpdateView):
|
||||
model = Weekmail
|
||||
template_name = "com/weekmail.jinja"
|
||||
form_class = modelform_factory(
|
||||
@@ -415,7 +414,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi
|
||||
art.rank, prev_art.rank = prev_art.rank, art.rank
|
||||
art.save()
|
||||
prev_art.save()
|
||||
self.quick_notif_list += ["qn_success"]
|
||||
messages.success(
|
||||
self.request,
|
||||
_("%(title)s moved up in the Weekmail") % {"title": art.title},
|
||||
)
|
||||
if "down_article" in request.GET:
|
||||
art = get_object_or_404(
|
||||
WeekmailArticle, id=request.GET["down_article"], weekmail=self.object
|
||||
@@ -427,7 +429,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi
|
||||
art.rank, next_art.rank = next_art.rank, art.rank
|
||||
art.save()
|
||||
next_art.save()
|
||||
self.quick_notif_list += ["qn_success"]
|
||||
messages.success(
|
||||
self.request,
|
||||
_("%(title)s moved down in the Weekmail") % {"title": art.title},
|
||||
)
|
||||
if "add_article" in request.GET:
|
||||
art = get_object_or_404(
|
||||
WeekmailArticle, id=request.GET["add_article"], weekmail=None
|
||||
@@ -436,7 +441,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi
|
||||
art.rank = self.object.articles.aggregate(Max("rank"))["rank__max"] or 0
|
||||
art.rank += 1
|
||||
art.save()
|
||||
self.quick_notif_list += ["qn_success"]
|
||||
messages.success(
|
||||
self.request,
|
||||
_("%(title)s added to the Weekmail") % {"title": art.title},
|
||||
)
|
||||
if "del_article" in request.GET:
|
||||
art = get_object_or_404(
|
||||
WeekmailArticle, id=request.GET["del_article"], weekmail=self.object
|
||||
@@ -444,7 +452,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi
|
||||
art.weekmail = None
|
||||
art.rank = -1
|
||||
art.save()
|
||||
self.quick_notif_list += ["qn_success"]
|
||||
messages.success(
|
||||
self.request,
|
||||
_("%(title)s removed from the Weekmail") % {"title": art.title},
|
||||
)
|
||||
return super().get(request, *args, **kwargs)
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
@@ -454,9 +465,7 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi
|
||||
return kwargs
|
||||
|
||||
|
||||
class WeekmailArticleEditView(
|
||||
ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView
|
||||
):
|
||||
class WeekmailArticleEditView(ComTabsMixin, CanEditPropMixin, UpdateView):
|
||||
"""Edit an article."""
|
||||
|
||||
model = WeekmailArticle
|
||||
@@ -468,11 +477,10 @@ class WeekmailArticleEditView(
|
||||
pk_url_kwarg = "article_id"
|
||||
template_name = "core/edit.jinja"
|
||||
success_url = reverse_lazy("com:weekmail")
|
||||
quick_notif_url_arg = "qn_weekmail_article_edit"
|
||||
current_tab = "weekmail"
|
||||
|
||||
|
||||
class WeekmailArticleCreateView(QuickNotifMixin, CreateView):
|
||||
class WeekmailArticleCreateView(CreateView):
|
||||
"""Post an article."""
|
||||
|
||||
model = WeekmailArticle
|
||||
@@ -483,7 +491,6 @@ class WeekmailArticleCreateView(QuickNotifMixin, CreateView):
|
||||
)
|
||||
template_name = "core/create.jinja"
|
||||
success_url = reverse_lazy("core:user_tools")
|
||||
quick_notif_url_arg = "qn_weekmail_new_article"
|
||||
|
||||
def get_initial(self):
|
||||
if "club" not in self.request.GET:
|
||||
|
@@ -768,7 +768,7 @@ class Command(BaseCommand):
|
||||
s = Subscription(
|
||||
member=user,
|
||||
subscription_type=subscription_type,
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
)
|
||||
s.subscription_start = s.compute_start(start)
|
||||
s.subscription_end = s.compute_end(
|
||||
|
@@ -1197,6 +1197,18 @@ class NotLocked(LockError):
|
||||
pass
|
||||
|
||||
|
||||
class PageQuerySet(models.QuerySet):
|
||||
def viewable_by(self, user: User) -> Self:
|
||||
if user.is_anonymous:
|
||||
return self.filter(view_groups=settings.SITH_GROUP_PUBLIC_ID)
|
||||
if user.has_perm("core.view_page"):
|
||||
return self.all()
|
||||
groups_ids = [g.id for g in user.cached_groups]
|
||||
if user.is_subscribed:
|
||||
groups_ids.append(settings.SITH_GROUP_SUBSCRIBERS_ID)
|
||||
return self.filter(view_groups__in=groups_ids)
|
||||
|
||||
|
||||
# This function prevents generating migration upon settings change
|
||||
def get_default_owner_group():
|
||||
return settings.SITH_GROUP_ROOT_ID
|
||||
@@ -1266,6 +1278,8 @@ class Page(models.Model):
|
||||
_("lock_timeout"), null=True, blank=True, default=None
|
||||
)
|
||||
|
||||
objects = PageQuerySet.as_manager()
|
||||
|
||||
class Meta:
|
||||
unique_together = ("name", "parent")
|
||||
permissions = (
|
||||
@@ -1275,12 +1289,9 @@ class Page(models.Model):
|
||||
def __str__(self):
|
||||
return self.get_full_name()
|
||||
|
||||
def save(self, *args, **kwargs):
|
||||
def save(self, *args, force_lock: bool = False, **kwargs):
|
||||
"""Performs some needed actions before and after saving a page in database."""
|
||||
locked = kwargs.pop("force_lock", False)
|
||||
if not locked:
|
||||
locked = self.is_locked()
|
||||
if not locked:
|
||||
if not force_lock and not self.is_locked():
|
||||
raise NotLocked("The page is not locked and thus can not be saved")
|
||||
self.full_clean()
|
||||
if not self.id:
|
||||
@@ -1292,7 +1303,7 @@ class Page(models.Model):
|
||||
# It also update all the children to maintain correct names
|
||||
self._full_name = self.get_full_name()
|
||||
for c in self.children.all():
|
||||
c.save()
|
||||
c.save(force_lock=force_lock)
|
||||
super().save(*args, **kwargs)
|
||||
self.unset_lock()
|
||||
|
||||
@@ -1408,14 +1419,14 @@ class Page(models.Model):
|
||||
def need_club_redirection(self):
|
||||
return self.is_club_page and self.name != settings.SITH_CLUB_ROOT_PAGE
|
||||
|
||||
def delete(self):
|
||||
def delete(self, *args, **kwargs):
|
||||
self.unset_lock_recursive()
|
||||
self.set_lock_recursive(User.objects.get(id=0))
|
||||
for child in self.children.all():
|
||||
child.parent = self.parent
|
||||
child.save()
|
||||
child.unset_lock_recursive()
|
||||
super().delete()
|
||||
return super().delete(*args, **kwargs)
|
||||
|
||||
|
||||
class PageRev(models.Model):
|
||||
@@ -1462,9 +1473,12 @@ class PageRev(models.Model):
|
||||
def get_absolute_url(self):
|
||||
return reverse("core:page", kwargs={"page_name": self.page._full_name})
|
||||
|
||||
def can_be_edited_by(self, user):
|
||||
def can_be_edited_by(self, user: User) -> bool:
|
||||
return self.page.can_be_edited_by(user)
|
||||
|
||||
def is_owned_by(self, user: User) -> bool:
|
||||
return any(g.id == self.page.owner_group_id for g in user.cached_groups)
|
||||
|
||||
|
||||
def get_notification_types():
|
||||
return settings.SITH_NOTIFICATIONS
|
||||
|
@@ -1,7 +1,10 @@
|
||||
import { limitedChoices } from "#core:alpine/limited-choices";
|
||||
import { alpinePlugin } from "#core:utils/notifications";
|
||||
import sort from "@alpinejs/sort";
|
||||
import Alpine from "alpinejs";
|
||||
|
||||
Alpine.plugin(sort);
|
||||
Alpine.plugin([sort, limitedChoices]);
|
||||
Alpine.magic("notifications", alpinePlugin);
|
||||
window.Alpine = Alpine;
|
||||
|
||||
window.addEventListener("DOMContentLoaded", () => {
|
||||
|
69
core/static/bundled/alpine/limited-choices.ts
Normal file
69
core/static/bundled/alpine/limited-choices.ts
Normal file
@@ -0,0 +1,69 @@
|
||||
import type { Alpine as AlpineType } from "alpinejs";
|
||||
|
||||
export function limitedChoices(Alpine: AlpineType) {
|
||||
/**
|
||||
* Directive to limit the number of elements
|
||||
* that can be selected in a group of checkboxes.
|
||||
*
|
||||
* When the max numbers of selectable elements is reached,
|
||||
* new elements will still be inserted, but oldest ones will be deselected.
|
||||
* For example, if checkboxes A, B and C have been selected and the max
|
||||
* number of selections is 3, then selecting D will result in having
|
||||
* B, C and D selected.
|
||||
*
|
||||
* # Example in template
|
||||
* ```html
|
||||
* <div x-data="{nbMax: 2}", x-limited-choices="nbMax">
|
||||
* <button @click="nbMax += 1">Click me to increase the limit</button>
|
||||
* <input type="checkbox" value="A" name="foo">
|
||||
* <input type="checkbox" value="B" name="foo">
|
||||
* <input type="checkbox" value="C" name="foo">
|
||||
* <input type="checkbox" value="D" name="foo">
|
||||
* </div>
|
||||
* ```
|
||||
*/
|
||||
Alpine.directive(
|
||||
"limited-choices",
|
||||
(el, { expression }, { evaluateLater, effect }) => {
|
||||
const getMaxChoices = evaluateLater(expression);
|
||||
let maxChoices: number;
|
||||
const inputs: HTMLInputElement[] = Array.from(
|
||||
el.querySelectorAll("input[type='checkbox']"),
|
||||
);
|
||||
const checked = [] as HTMLInputElement[];
|
||||
|
||||
const manageDequeue = () => {
|
||||
if (checked.length <= maxChoices) {
|
||||
// There isn't too many checkboxes selected. Nothing to do
|
||||
return;
|
||||
}
|
||||
const popped = checked.splice(0, checked.length - maxChoices);
|
||||
for (const p of popped) {
|
||||
p.checked = false;
|
||||
}
|
||||
};
|
||||
|
||||
for (const input of inputs) {
|
||||
input.addEventListener("change", (_e) => {
|
||||
if (input.checked) {
|
||||
checked.push(input);
|
||||
} else {
|
||||
checked.splice(checked.indexOf(input), 1);
|
||||
}
|
||||
manageDequeue();
|
||||
});
|
||||
}
|
||||
effect(() => {
|
||||
getMaxChoices((value: string) => {
|
||||
const previousValue = maxChoices;
|
||||
maxChoices = Number.parseInt(value);
|
||||
if (maxChoices < previousValue) {
|
||||
// The maximum number of selectable items has been lowered.
|
||||
// Some currently selected elements may need to be removed
|
||||
manageDequeue();
|
||||
}
|
||||
});
|
||||
});
|
||||
},
|
||||
);
|
||||
}
|
36
core/static/bundled/utils/notifications.ts
Normal file
36
core/static/bundled/utils/notifications.ts
Normal file
@@ -0,0 +1,36 @@
|
||||
export enum NotificationLevel {
|
||||
Error = "error",
|
||||
Warning = "warning",
|
||||
Success = "success",
|
||||
}
|
||||
|
||||
export function createNotification(message: string, level: NotificationLevel) {
|
||||
const element = document.getElementById("quick-notifications");
|
||||
if (element === null) {
|
||||
return false;
|
||||
}
|
||||
return element.dispatchEvent(
|
||||
new CustomEvent("quick-notification-add", {
|
||||
detail: { text: message, tag: level },
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
export function deleteNotifications() {
|
||||
const element = document.getElementById("quick-notifications");
|
||||
if (element === null) {
|
||||
return false;
|
||||
}
|
||||
return element.dispatchEvent(new CustomEvent("quick-notification-delete"));
|
||||
}
|
||||
|
||||
export function alpinePlugin() {
|
||||
return {
|
||||
error: (message: string) => createNotification(message, NotificationLevel.Error),
|
||||
warning: (message: string) =>
|
||||
createNotification(message, NotificationLevel.Warning),
|
||||
success: (message: string) =>
|
||||
createNotification(message, NotificationLevel.Success),
|
||||
clear: () => deleteNotifications(),
|
||||
};
|
||||
}
|
@@ -321,7 +321,6 @@ $hovered-red-text-color: #ff4d4d;
|
||||
|
||||
>#header_notif {
|
||||
box-sizing: border-box;
|
||||
display: none;
|
||||
position: absolute;
|
||||
margin: 0;
|
||||
background-color: whitesmoke;
|
||||
|
@@ -1,38 +0,0 @@
|
||||
$(() => {
|
||||
$("#quick_notif li").click(function () {
|
||||
$(this).hide();
|
||||
});
|
||||
});
|
||||
|
||||
// biome-ignore lint/correctness/noUnusedVariables: used in other scripts
|
||||
function createQuickNotif(msg) {
|
||||
const el = document.createElement("li");
|
||||
el.textContent = msg;
|
||||
el.addEventListener("click", () => el.parentNode.removeChild(el));
|
||||
document.getElementById("quick_notif").appendChild(el);
|
||||
}
|
||||
|
||||
// biome-ignore lint/correctness/noUnusedVariables: used in other scripts
|
||||
function deleteQuickNotifs() {
|
||||
const el = document.getElementById("quick_notif");
|
||||
while (el.firstChild) {
|
||||
el.removeChild(el.firstChild);
|
||||
}
|
||||
}
|
||||
|
||||
// biome-ignore lint/correctness/noUnusedVariables: used in other scripts
|
||||
function displayNotif() {
|
||||
$("#header_notif").toggle().parent().toggleClass("white");
|
||||
}
|
||||
|
||||
// You can't get the csrf token from the template in a widget
|
||||
// We get it from a cookie as a workaround, see this link
|
||||
// https://docs.djangoproject.com/en/2.0/ref/csrf/#ajax
|
||||
// Sadly, getting the cookie is not possible with CSRF_COOKIE_HTTPONLY or CSRF_USE_SESSIONS is True
|
||||
// So, the true workaround is to get the token from the dom
|
||||
// https://docs.djangoproject.com/en/2.0/ref/csrf/#acquiring-the-token-if-csrf-use-sessions-is-true
|
||||
// biome-ignore lint/style/useNamingConvention: can't find it used anywhere but I will not play with the devil
|
||||
// biome-ignore lint/correctness/noUnusedVariables: used in other scripts
|
||||
function getCSRFToken() {
|
||||
return $("[name=csrfmiddlewaretoken]").val();
|
||||
}
|
@@ -270,17 +270,6 @@ body {
|
||||
}
|
||||
|
||||
/*--------------------------------CONTENT------------------------------*/
|
||||
#quick_notif {
|
||||
width: 100%;
|
||||
margin: 0 auto;
|
||||
list-style-type: none;
|
||||
background: $second-color;
|
||||
|
||||
li {
|
||||
padding: 10px;
|
||||
}
|
||||
}
|
||||
|
||||
#content {
|
||||
padding: 1em 1%;
|
||||
box-shadow: $shadow-color 0 5px 10px;
|
||||
|
@@ -32,10 +32,6 @@
|
||||
<script type="module" src="{{ static('bundled/country-flags-index.ts') }}"></script>
|
||||
<script type="module" src="{{ static('bundled/core/tooltips-index.ts') }}"></script>
|
||||
|
||||
<!-- Jquery declared here to be accessible in every django widgets -->
|
||||
<script src="{{ static('bundled/vendored/jquery.min.js') }}"></script>
|
||||
<script src="{{ static('core/js/script.js') }}"></script>
|
||||
|
||||
{% block additional_css %}{% endblock %}
|
||||
{% block additional_js %}{% endblock %}
|
||||
{% endblock %}
|
||||
@@ -74,17 +70,15 @@
|
||||
|
||||
<div id="page">
|
||||
|
||||
<ul id="quick_notif">
|
||||
{% for n in quick_notifs %}
|
||||
<li>{{ n }}</li>
|
||||
{% endfor %}
|
||||
</ul>
|
||||
|
||||
<div id="content">
|
||||
{%- block tabs -%}
|
||||
{% include "core/base/tabs.jinja" %}
|
||||
{%- endblock -%}
|
||||
|
||||
{% block notifications %}
|
||||
{% include "core/base/notifications.jinja" %}
|
||||
{% endblock %}
|
||||
|
||||
{%- block errors -%}
|
||||
{% if error %}
|
||||
{{ error }}
|
||||
@@ -101,16 +95,6 @@
|
||||
{% endblock %}
|
||||
|
||||
{% block script %}
|
||||
<script>
|
||||
document.addEventListener("keydown", (e) => {
|
||||
// Looking at the `s` key when not typing in a form
|
||||
if (e.keyCode !== 83 || ["INPUT", "TEXTAREA", "SELECT"].includes(e.target.nodeName)) {
|
||||
return;
|
||||
}
|
||||
document.getElementById("search").focus();
|
||||
e.preventDefault(); // Don't type the character in the focused search input
|
||||
})
|
||||
</script>
|
||||
{% endblock %}
|
||||
</body>
|
||||
</html>
|
||||
|
@@ -74,9 +74,9 @@
|
||||
{% endif %}
|
||||
></a>
|
||||
</div>
|
||||
<div class="notification">
|
||||
<a href="#" onclick="displayNotif()">
|
||||
<i class="fa-regular fa-bell"></i>
|
||||
<div class="notification" x-data="{display: false}" :class="{white: display}">
|
||||
<a href="#" @click.prevent="display = !display">
|
||||
<i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i>
|
||||
{% set notification_count = user.notifications.filter(viewed=False).count() %}
|
||||
|
||||
{% if notification_count > 0 %}
|
||||
@@ -89,7 +89,7 @@
|
||||
</span>
|
||||
{% endif %}
|
||||
</a>
|
||||
<div id="header_notif">
|
||||
<div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false">
|
||||
<ul>
|
||||
{% if user.notifications.filter(viewed=False).count() > 0 %}
|
||||
{% for n in user.notifications.filter(viewed=False).order_by('-date') %}
|
||||
|
24
core/templates/core/base/notifications.jinja
Normal file
24
core/templates/core/base/notifications.jinja
Normal file
@@ -0,0 +1,24 @@
|
||||
<div id="quick-notifications"
|
||||
x-data="{
|
||||
messages: [
|
||||
{% if messages %}
|
||||
{% for message in messages %}
|
||||
{
|
||||
tag: '{{ message.tags }}',
|
||||
text: '{{ message }}',
|
||||
},
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
]
|
||||
}"
|
||||
@quick-notification-add="(e) => messages.push(e?.detail)"
|
||||
@quick-notification-delete="messages = []">
|
||||
<template x-for="message in messages">
|
||||
<div x-data="{show: true}" class="alert" :class="`alert-${message.tag}`" x-show="show" x-transition>
|
||||
<span class="alert-main" x-text="message.text"></span>
|
||||
<span class="clickable" @click="show = false">
|
||||
<i class="fa fa-close"></i>
|
||||
</span>
|
||||
</div>
|
||||
</template>
|
||||
</div>
|
@@ -15,6 +15,7 @@
|
||||
{{ select_all_checkbox("add_users") }}
|
||||
<hr>
|
||||
{% csrf_token %}
|
||||
{{ form.non_field_errors() }}
|
||||
<label for="{{ form.users_removed.id_for_label }}">{{ form.users_removed.label }} :</label>
|
||||
{{ form.users_removed.errors }}
|
||||
{% for user in form.users_removed %}
|
||||
|
@@ -30,7 +30,11 @@
|
||||
- {{ purchase.date|localtime|time(DATETIME_FORMAT) }}
|
||||
</td>
|
||||
<td>{{ purchase.counter }}</td>
|
||||
{% if not purchase.seller %}
|
||||
<td>{% trans %}Deleted user{% endtrans %}</td>
|
||||
{% else %}
|
||||
<td><a href="{{ purchase.seller.get_absolute_url() }}">{{ purchase.seller.get_display_name() }}</a></td>
|
||||
{% endif %}
|
||||
<td>{{ purchase.label }}</td>
|
||||
<td>{{ purchase.quantity }}</td>
|
||||
<td>{{ purchase.quantity * purchase.unit_price }} €</td>
|
||||
|
58
core/tests/test_page.py
Normal file
58
core/tests/test_page.py
Normal file
@@ -0,0 +1,58 @@
|
||||
import pytest
|
||||
from django.conf import settings
|
||||
from django.contrib.auth.models import Permission
|
||||
from django.test import Client
|
||||
from django.urls import reverse
|
||||
from model_bakery import baker
|
||||
from pytest_django.asserts import assertRedirects
|
||||
|
||||
from core.baker_recipes import board_user, subscriber_user
|
||||
from core.models import AnonymousUser, Page, User
|
||||
from sith.settings import SITH_GROUP_OLD_SUBSCRIBERS_ID, SITH_GROUP_SUBSCRIBERS_ID
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_edit_page(client: Client):
|
||||
user = board_user.make()
|
||||
page = baker.prepare(Page)
|
||||
page.save(force_lock=True)
|
||||
page.view_groups.add(user.groups.first())
|
||||
client.force_login(user)
|
||||
|
||||
url = reverse("core:page_edit", kwargs={"page_name": page._full_name})
|
||||
res = client.get(url)
|
||||
assert res.status_code == 200
|
||||
|
||||
res = client.post(url, data={"content": "Hello World"})
|
||||
assertRedirects(res, reverse("core:page", kwargs={"page_name": page._full_name}))
|
||||
revision = page.revisions.last()
|
||||
assert revision.content == "Hello World"
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_viewable_by():
|
||||
# remove existing pages to prevent side effect
|
||||
Page.objects.all().delete()
|
||||
view_groups = [
|
||||
[settings.SITH_GROUP_PUBLIC_ID],
|
||||
[settings.SITH_GROUP_PUBLIC_ID, SITH_GROUP_SUBSCRIBERS_ID],
|
||||
[SITH_GROUP_SUBSCRIBERS_ID],
|
||||
[SITH_GROUP_SUBSCRIBERS_ID, SITH_GROUP_OLD_SUBSCRIBERS_ID],
|
||||
[],
|
||||
]
|
||||
pages = baker.make(Page, _quantity=len(view_groups), _bulk_create=True)
|
||||
for page, groups in zip(pages, view_groups, strict=True):
|
||||
page.view_groups.set(groups)
|
||||
|
||||
viewable = Page.objects.viewable_by(AnonymousUser()).values_list("id", flat=True)
|
||||
assert set(viewable) == {pages[0].id, pages[1].id}
|
||||
|
||||
subscriber = subscriber_user.make()
|
||||
viewable = Page.objects.viewable_by(subscriber).values_list("id", flat=True)
|
||||
assert set(viewable) == {p.id for p in pages[0:4]}
|
||||
|
||||
root_user = baker.make(
|
||||
User, user_permissions=[Permission.objects.get(codename="view_page")]
|
||||
)
|
||||
viewable = Page.objects.viewable_by(root_user).values_list("id", flat=True)
|
||||
assert set(viewable) == {p.id for p in pages}
|
@@ -20,7 +20,8 @@ from core.baker_recipes import (
|
||||
)
|
||||
from core.models import Group, User
|
||||
from core.views import UserTabsMixin
|
||||
from counter.models import Counter, Refilling, Selling
|
||||
from counter.baker_recipes import sale_recipe
|
||||
from counter.models import Counter, Customer, Refilling, Selling
|
||||
from eboutic.models import Invoice, InvoiceItem
|
||||
|
||||
|
||||
@@ -129,6 +130,31 @@ def test_user_account_not_found(client: Client):
|
||||
assert res.status_code == 404
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_is_deleted_barman_shown_as_deleted(client: Client):
|
||||
customer = baker.make(Customer)
|
||||
date = now()
|
||||
sale_recipe.make(
|
||||
seller=iter([None, baker.make(User)]),
|
||||
customer=customer,
|
||||
date=date,
|
||||
_quantity=2,
|
||||
_bulk_create=True,
|
||||
)
|
||||
client.force_login(customer.user)
|
||||
res = client.get(
|
||||
reverse(
|
||||
"core:user_account_detail",
|
||||
kwargs={
|
||||
"user_id": customer.user.id,
|
||||
"year": date.year,
|
||||
"month": date.month,
|
||||
},
|
||||
)
|
||||
)
|
||||
assert res.status_code == 200
|
||||
|
||||
|
||||
class TestFilterInactive(TestCase):
|
||||
@classmethod
|
||||
def setUpTestData(cls):
|
||||
|
@@ -2,7 +2,6 @@ import copy
|
||||
import inspect
|
||||
from typing import Any, ClassVar, LiteralString, Protocol, Unpack
|
||||
|
||||
from django.conf import settings
|
||||
from django.core.exceptions import ImproperlyConfigured
|
||||
from django.http import HttpRequest, HttpResponse
|
||||
from django.template.loader import render_to_string
|
||||
@@ -41,36 +40,6 @@ class TabedViewMixin(View):
|
||||
return kwargs
|
||||
|
||||
|
||||
class QuickNotifMixin:
|
||||
quick_notif_list = []
|
||||
|
||||
def dispatch(self, request, *arg, **kwargs):
|
||||
# In some cases, the class can stay instanciated, so we need to reset the list
|
||||
self.quick_notif_list = []
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
|
||||
def get_success_url(self):
|
||||
ret = super().get_success_url()
|
||||
if hasattr(self, "quick_notif_url_arg"):
|
||||
if "?" in ret:
|
||||
ret += "&" + self.quick_notif_url_arg
|
||||
else:
|
||||
ret += "?" + self.quick_notif_url_arg
|
||||
return ret
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
"""Add quick notifications to context."""
|
||||
kwargs = super().get_context_data(**kwargs)
|
||||
kwargs["quick_notifs"] = []
|
||||
for n in self.quick_notif_list:
|
||||
kwargs["quick_notifs"].append(settings.SITH_QUICK_NOTIF[n])
|
||||
for key, val in settings.SITH_QUICK_NOTIF.items():
|
||||
for gk in self.request.GET:
|
||||
if key == gk:
|
||||
kwargs["quick_notifs"].append(val)
|
||||
return kwargs
|
||||
|
||||
|
||||
class AllowFragment:
|
||||
"""Add `is_fragment` to templates. It's only True if the request is emitted by htmx"""
|
||||
|
||||
|
@@ -43,11 +43,14 @@ class CanEditPagePropMixin(CanEditPropMixin):
|
||||
return res
|
||||
|
||||
|
||||
class PageListView(CanViewMixin, ListView):
|
||||
class PageListView(ListView):
|
||||
model = Page
|
||||
template_name = "core/page_list.jinja"
|
||||
queryset = (
|
||||
Page.objects.annotate(
|
||||
|
||||
def get_queryset(self):
|
||||
return (
|
||||
Page.objects.viewable_by(self.request.user)
|
||||
.annotate(
|
||||
display_name=Coalesce(
|
||||
Subquery(
|
||||
PageRev.objects.filter(page=OuterRef("id"))
|
||||
@@ -57,7 +60,6 @@ class PageListView(CanViewMixin, ListView):
|
||||
F("name"),
|
||||
)
|
||||
)
|
||||
.prefetch_related("view_groups")
|
||||
.select_related("parent")
|
||||
)
|
||||
|
||||
@@ -184,7 +186,7 @@ class PageEditViewBase(CanEditMixin, UpdateView):
|
||||
)
|
||||
template_name = "core/pagerev_edit.jinja"
|
||||
|
||||
def get_object(self):
|
||||
def get_object(self, *args, **kwargs):
|
||||
self.page = Page.get_page_by_full_name(self.kwargs["page_name"])
|
||||
return self._get_revision()
|
||||
|
||||
|
@@ -65,7 +65,7 @@ from core.views.forms import (
|
||||
UserGroupsForm,
|
||||
UserProfileForm,
|
||||
)
|
||||
from core.views.mixins import QuickNotifMixin, TabedViewMixin, UseFragmentsMixin
|
||||
from core.views.mixins import TabedViewMixin, UseFragmentsMixin
|
||||
from counter.models import Counter, Refilling, Selling
|
||||
from eboutic.models import Invoice
|
||||
from subscription.models import Subscription
|
||||
@@ -564,7 +564,7 @@ class UserUpdateGroupView(UserTabsMixin, CanEditPropMixin, UpdateView):
|
||||
current_tab = "groups"
|
||||
|
||||
|
||||
class UserToolsView(LoginRequiredMixin, QuickNotifMixin, UserTabsMixin, TemplateView):
|
||||
class UserToolsView(LoginRequiredMixin, UserTabsMixin, TemplateView):
|
||||
"""Displays the logged user's tools."""
|
||||
|
||||
template_name = "core/user_tools.jinja"
|
||||
|
@@ -4,7 +4,6 @@
|
||||
heading_level: 3
|
||||
members:
|
||||
- TabedViewMixin
|
||||
- QuickNotifMixin
|
||||
- AllowFragment
|
||||
- FragmentMixin
|
||||
- UseFragmentsMixin
|
@@ -31,12 +31,5 @@
|
||||
</div>
|
||||
|
||||
<br>
|
||||
|
||||
{% if messages %}
|
||||
{% for message in messages %}
|
||||
<div class="alert alert-{{ message.tags }}">
|
||||
{{ message }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
{% include "core/base/notifications.jinja" %}
|
||||
</div>
|
||||
|
@@ -1,5 +1,9 @@
|
||||
{% extends "core/base.jinja" %}
|
||||
|
||||
{% block notifications %}
|
||||
{# Notifications are moved inside the billing info fragment #}
|
||||
{% endblock %}
|
||||
|
||||
{% block title %}
|
||||
{% trans %}Basket state{% endtrans %}
|
||||
{% endblock %}
|
||||
|
@@ -22,14 +22,6 @@
|
||||
{% block content %}
|
||||
<h1 id="eboutic-title">{% trans %}Eboutic{% endtrans %}</h1>
|
||||
|
||||
{% if messages %}
|
||||
{% for message in messages %}
|
||||
<div class="alert alert-{{ message.tags }}">
|
||||
{{ message }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
|
||||
<div id="eboutic" x-data="basket({{ last_purchase_time }})">
|
||||
<div id="basket">
|
||||
<h3>Panier</h3>
|
||||
|
@@ -4,14 +4,6 @@
|
||||
<h3>{% trans %}Eboutic{% endtrans %}</h3>
|
||||
|
||||
<div>
|
||||
{% if messages %}
|
||||
{% for message in messages %}
|
||||
<div class="alert alert-{{ message.tags }}">
|
||||
{{ message }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
|
||||
{% if success %}
|
||||
{% trans %}Payment successful{% endtrans %}
|
||||
{% else %}
|
||||
|
155
election/forms.py
Normal file
155
election/forms.py
Normal file
@@ -0,0 +1,155 @@
|
||||
from django import forms
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
|
||||
from core.models import User
|
||||
from core.views.forms import SelectDateTime
|
||||
from core.views.widgets.ajax_select import (
|
||||
AutoCompleteSelect,
|
||||
AutoCompleteSelectMultipleGroup,
|
||||
AutoCompleteSelectUser,
|
||||
)
|
||||
from core.views.widgets.markdown import MarkdownInput
|
||||
from election.models import Candidature, Election, ElectionList, Role
|
||||
|
||||
|
||||
class LimitedCheckboxField(forms.ModelMultipleChoiceField):
|
||||
"""A `ModelMultipleChoiceField`, with a max limit of selectable inputs."""
|
||||
|
||||
def __init__(self, queryset, max_choice, **kwargs):
|
||||
self.max_choice = max_choice
|
||||
super().__init__(queryset, **kwargs)
|
||||
|
||||
def clean(self, value):
|
||||
qs = super().clean(value)
|
||||
self.validate(qs)
|
||||
return qs
|
||||
|
||||
def validate(self, qs):
|
||||
if qs.count() > self.max_choice:
|
||||
raise forms.ValidationError(
|
||||
_("You have selected too many candidates."), code="invalid"
|
||||
)
|
||||
|
||||
|
||||
class CandidateForm(forms.ModelForm):
|
||||
"""Form to candidate."""
|
||||
|
||||
required_css_class = "required"
|
||||
|
||||
class Meta:
|
||||
model = Candidature
|
||||
fields = ["user", "role", "program", "election_list"]
|
||||
labels = {
|
||||
"user": _("User to candidate"),
|
||||
}
|
||||
widgets = {
|
||||
"program": MarkdownInput,
|
||||
"user": AutoCompleteSelectUser,
|
||||
"role": AutoCompleteSelect,
|
||||
"election_list": AutoCompleteSelect,
|
||||
}
|
||||
|
||||
def __init__(self, *args, election: Election, can_edit: bool = False, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
self.fields["role"].queryset = election.roles.select_related("election")
|
||||
self.fields["election_list"].queryset = election.election_lists.all()
|
||||
if not can_edit:
|
||||
self.fields["user"].widget = forms.HiddenInput()
|
||||
|
||||
|
||||
class VoteForm(forms.Form):
|
||||
def __init__(self, election: Election, user: User, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
if not election.can_vote(user):
|
||||
return
|
||||
for role in election.roles.all():
|
||||
cand = role.candidatures
|
||||
if role.max_choice > 1:
|
||||
self.fields[role.title] = LimitedCheckboxField(
|
||||
cand, role.max_choice, required=False
|
||||
)
|
||||
else:
|
||||
self.fields[role.title] = forms.ModelChoiceField(
|
||||
cand,
|
||||
required=False,
|
||||
widget=forms.RadioSelect(),
|
||||
empty_label=_("Blank vote"),
|
||||
)
|
||||
|
||||
|
||||
class RoleForm(forms.ModelForm):
|
||||
"""Form for creating a role."""
|
||||
|
||||
class Meta:
|
||||
model = Role
|
||||
fields = ["title", "election", "description", "max_choice"]
|
||||
widgets = {"election": AutoCompleteSelect}
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
election_id = kwargs.pop("election_id", None)
|
||||
super().__init__(*args, **kwargs)
|
||||
if election_id:
|
||||
self.fields["election"].queryset = Election.objects.filter(
|
||||
id=election_id
|
||||
).all()
|
||||
|
||||
def clean(self):
|
||||
cleaned_data = super().clean()
|
||||
title = cleaned_data.get("title")
|
||||
election = cleaned_data.get("election")
|
||||
if Role.objects.filter(title=title, election=election).exists():
|
||||
raise forms.ValidationError(
|
||||
_("This role already exists for this election"), code="invalid"
|
||||
)
|
||||
|
||||
|
||||
class ElectionListForm(forms.ModelForm):
|
||||
class Meta:
|
||||
model = ElectionList
|
||||
fields = ("title", "election")
|
||||
widgets = {"election": AutoCompleteSelect}
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
election_id = kwargs.pop("election_id", None)
|
||||
super().__init__(*args, **kwargs)
|
||||
if election_id:
|
||||
self.fields["election"].queryset = Election.objects.filter(
|
||||
id=election_id
|
||||
).all()
|
||||
|
||||
|
||||
class ElectionForm(forms.ModelForm):
|
||||
class Meta:
|
||||
model = Election
|
||||
fields = [
|
||||
"title",
|
||||
"description",
|
||||
"archived",
|
||||
"start_candidature",
|
||||
"end_candidature",
|
||||
"start_date",
|
||||
"end_date",
|
||||
"edit_groups",
|
||||
"view_groups",
|
||||
"vote_groups",
|
||||
"candidature_groups",
|
||||
]
|
||||
widgets = {
|
||||
"edit_groups": AutoCompleteSelectMultipleGroup,
|
||||
"view_groups": AutoCompleteSelectMultipleGroup,
|
||||
"vote_groups": AutoCompleteSelectMultipleGroup,
|
||||
"candidature_groups": AutoCompleteSelectMultipleGroup,
|
||||
}
|
||||
|
||||
start_date = forms.DateTimeField(
|
||||
label=_("Start date"), widget=SelectDateTime, required=True
|
||||
)
|
||||
end_date = forms.DateTimeField(
|
||||
label=_("End date"), widget=SelectDateTime, required=True
|
||||
)
|
||||
start_candidature = forms.DateTimeField(
|
||||
label=_("Start candidature"), widget=SelectDateTime, required=True
|
||||
)
|
||||
end_candidature = forms.DateTimeField(
|
||||
label=_("End candidature"), widget=SelectDateTime, required=True
|
||||
)
|
@@ -0,0 +1,30 @@
|
||||
# Generated by Django 4.2.20 on 2025-03-14 18:18
|
||||
|
||||
import django.db.models.deletion
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
dependencies = [
|
||||
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||
("election", "0004_auto_20191006_0049"),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name="candidature",
|
||||
name="program",
|
||||
field=models.TextField(blank=True, default="", verbose_name="description"),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name="candidature",
|
||||
name="user",
|
||||
field=models.ForeignKey(
|
||||
on_delete=django.db.models.deletion.CASCADE,
|
||||
related_name="candidates",
|
||||
to=settings.AUTH_USER_MODEL,
|
||||
verbose_name="user",
|
||||
),
|
||||
),
|
||||
]
|
@@ -1,5 +1,7 @@
|
||||
from django.db import models
|
||||
from django.db.models import Count
|
||||
from django.utils import timezone
|
||||
from django.utils.functional import cached_property
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
from ordered_model.models import OrderedModel
|
||||
|
||||
@@ -22,21 +24,18 @@ class Election(models.Model):
|
||||
verbose_name=_("edit groups"),
|
||||
blank=True,
|
||||
)
|
||||
|
||||
view_groups = models.ManyToManyField(
|
||||
Group,
|
||||
related_name="viewable_elections",
|
||||
verbose_name=_("view groups"),
|
||||
blank=True,
|
||||
)
|
||||
|
||||
vote_groups = models.ManyToManyField(
|
||||
Group,
|
||||
related_name="votable_elections",
|
||||
verbose_name=_("vote groups"),
|
||||
blank=True,
|
||||
)
|
||||
|
||||
candidature_groups = models.ManyToManyField(
|
||||
Group,
|
||||
related_name="candidate_elections",
|
||||
@@ -45,7 +44,7 @@ class Election(models.Model):
|
||||
)
|
||||
|
||||
voters = models.ManyToManyField(
|
||||
User, verbose_name=("voters"), related_name="voted_elections"
|
||||
User, verbose_name=_("voters"), related_name="voted_elections"
|
||||
)
|
||||
archived = models.BooleanField(_("archived"), default=False)
|
||||
|
||||
@@ -55,20 +54,20 @@ class Election(models.Model):
|
||||
@property
|
||||
def is_vote_active(self):
|
||||
now = timezone.now()
|
||||
return bool(now <= self.end_date and now >= self.start_date)
|
||||
return self.start_date <= now <= self.end_date
|
||||
|
||||
@property
|
||||
def is_vote_finished(self):
|
||||
return bool(timezone.now() > self.end_date)
|
||||
return timezone.now() > self.end_date
|
||||
|
||||
@property
|
||||
def is_candidature_active(self):
|
||||
now = timezone.now()
|
||||
return bool(now <= self.end_candidature and now >= self.start_candidature)
|
||||
return self.start_candidature <= now <= self.end_candidature
|
||||
|
||||
@property
|
||||
def is_vote_editable(self):
|
||||
return bool(timezone.now() <= self.end_candidature)
|
||||
return timezone.now() <= self.end_candidature
|
||||
|
||||
def can_candidate(self, user):
|
||||
for group_id in self.candidature_groups.values_list("pk", flat=True):
|
||||
@@ -87,7 +86,7 @@ class Election(models.Model):
|
||||
def has_voted(self, user):
|
||||
return self.voters.filter(id=user.id).exists()
|
||||
|
||||
@property
|
||||
@cached_property
|
||||
def results(self):
|
||||
results = {}
|
||||
total_vote = self.voters.count()
|
||||
@@ -95,12 +94,6 @@ class Election(models.Model):
|
||||
results[role.title] = role.results(total_vote)
|
||||
return results
|
||||
|
||||
def delete(self, *args, **kwargs):
|
||||
self.election_lists.all().delete()
|
||||
super().delete(*args, **kwargs)
|
||||
|
||||
# Permissions
|
||||
|
||||
|
||||
class Role(OrderedModel):
|
||||
"""This class allows to create a new role avaliable for a candidature."""
|
||||
@@ -115,23 +108,27 @@ class Role(OrderedModel):
|
||||
description = models.TextField(_("description"), null=True, blank=True)
|
||||
max_choice = models.IntegerField(_("max choice"), default=1)
|
||||
|
||||
def results(self, total_vote):
|
||||
results = {}
|
||||
def __str__(self):
|
||||
return f"{self.title} - {self.election.title}"
|
||||
|
||||
def results(self, total_vote: int) -> dict[str, dict[str, int | float]]:
|
||||
if total_vote == 0:
|
||||
candidates = self.candidatures.values_list("user__username")
|
||||
return {
|
||||
key: {"vote": 0, "percent": 0} for key in ["blank_votes", *candidates]
|
||||
}
|
||||
total_vote *= self.max_choice
|
||||
results = {"total vote": total_vote}
|
||||
non_blank = 0
|
||||
for candidature in self.candidatures.all():
|
||||
cand_results = {}
|
||||
cand_results["vote"] = self.votes.filter(candidature=candidature).count()
|
||||
if total_vote == 0:
|
||||
cand_results["percent"] = 0
|
||||
else:
|
||||
cand_results["percent"] = cand_results["vote"] * 100 / total_vote
|
||||
non_blank += cand_results["vote"]
|
||||
results[candidature.user.username] = cand_results
|
||||
results["total vote"] = total_vote
|
||||
if total_vote == 0:
|
||||
results["blank vote"] = {"vote": 0, "percent": 0}
|
||||
else:
|
||||
candidatures = self.candidatures.annotate(nb_votes=Count("votes")).values(
|
||||
"nb_votes", "user__username"
|
||||
)
|
||||
for candidature in candidatures:
|
||||
non_blank += candidature["nb_votes"]
|
||||
results[candidature["user__username"]] = {
|
||||
"vote": candidature["nb_votes"],
|
||||
"percent": candidature["nb_votes"] * 100 / total_vote,
|
||||
}
|
||||
results["blank vote"] = {
|
||||
"vote": total_vote - non_blank,
|
||||
"percent": (total_vote - non_blank) * 100 / total_vote,
|
||||
@@ -142,9 +139,6 @@ class Role(OrderedModel):
|
||||
def edit_groups(self):
|
||||
return self.election.edit_groups
|
||||
|
||||
def __str__(self):
|
||||
return ("%s : %s") % (self.election.title, self.title)
|
||||
|
||||
|
||||
class ElectionList(models.Model):
|
||||
"""To allow per list vote."""
|
||||
@@ -163,11 +157,6 @@ class ElectionList(models.Model):
|
||||
def can_be_edited_by(self, user):
|
||||
return user.can_edit(self.election)
|
||||
|
||||
def delete(self, *args, **kwargs):
|
||||
for candidature in self.candidatures.all():
|
||||
candidature.delete()
|
||||
super().delete(*args, **kwargs)
|
||||
|
||||
|
||||
class Candidature(models.Model):
|
||||
"""This class is a component of responsability."""
|
||||
@@ -182,10 +171,9 @@ class Candidature(models.Model):
|
||||
User,
|
||||
verbose_name=_("user"),
|
||||
related_name="candidates",
|
||||
blank=True,
|
||||
on_delete=models.CASCADE,
|
||||
)
|
||||
program = models.TextField(_("description"), null=True, blank=True)
|
||||
program = models.TextField(_("description"), default="", blank=True)
|
||||
election_list = models.ForeignKey(
|
||||
ElectionList,
|
||||
related_name="candidatures",
|
||||
@@ -196,13 +184,10 @@ class Candidature(models.Model):
|
||||
def __str__(self):
|
||||
return f"{self.role.title} : {self.user.username}"
|
||||
|
||||
def delete(self):
|
||||
for vote in self.votes.all():
|
||||
vote.delete()
|
||||
super().delete()
|
||||
|
||||
def can_be_edited_by(self, user):
|
||||
return (user == self.user) or user.can_edit(self.role.election)
|
||||
return (
|
||||
(user == self.user) or user.can_edit(self.role.election)
|
||||
) and self.role.election.is_vote_editable
|
||||
|
||||
|
||||
class Vote(models.Model):
|
||||
|
@@ -31,7 +31,7 @@
|
||||
<time datetime="{{ election.end_date }}">{{ election.end_date|localtime|date(DATETIME_FORMAT)}}</time>
|
||||
{% trans %} at {% endtrans %}<time>{{ election.end_date|localtime|time(DATETIME_FORMAT)}}</time>
|
||||
</p>
|
||||
{%- if election.has_voted(user) %}
|
||||
{%- if user_has_voted %}
|
||||
<p class="election__elector-infos">
|
||||
{%- if election.is_vote_active %}
|
||||
<span>{% trans %}You already have submitted your vote.{% endtrans %}</span>
|
||||
@@ -45,12 +45,11 @@
|
||||
<form action="{{ url('election:vote', election.id) }}" method="post" class="election__vote-form" name="vote-form" id="vote-form">
|
||||
{% csrf_token %}
|
||||
<table class="election_table">
|
||||
{%- set election_lists = election.election_lists.all() -%}
|
||||
<thead class="lists">
|
||||
<tr>
|
||||
<th class="column" style="width: {{ 100 / (election_lists.count() + 1) }}%">{% trans %}Blank vote{% endtrans %}</th>
|
||||
<th class="column" style="width: {{ 100 / (election_lists|length + 1) }}%">{% trans %}Blank vote{% endtrans %}</th>
|
||||
{%- for election_list in election_lists %}
|
||||
<th class="column" style="width: {{ 100 / (election_lists.count() + 1) }}%">
|
||||
<th class="column" style="width: {{ 100 / (election_lists|length + 1) }}%">
|
||||
<span>{{ election_list.title }}</span>
|
||||
{% if user.can_edit(election_list) and election.is_vote_editable -%}
|
||||
<a href="{{ url('election:delete_list', list_id=election_list.id) }}"><i class="fa-regular fa-trash-can delete-action"></i></a>
|
||||
@@ -59,18 +58,26 @@
|
||||
{%- endfor %}
|
||||
</tr>
|
||||
</thead>
|
||||
{%- set role_list = election.roles.order_by('order').all() %}
|
||||
{%- for role in role_list %}
|
||||
{%- set count = [0] %}
|
||||
{%- for role in election_roles %}
|
||||
{%- set role_data = election_form.data.getlist(role.title) if role.title in election_form.data else [] %}
|
||||
<tbody data-max-choice="{{role.max_choice}}" class="role{{ ' role_error' if role.title in election_form.errors else '' }}{{ ' role__multiple-choices' if role.max_choice > 1 else ''}}">
|
||||
|
||||
<tbody
|
||||
{% if role.max_choice > 1 -%}
|
||||
x-data x-limited-choices="{{ role.max_choice }}"
|
||||
{%- endif %}
|
||||
class="role {% if role.title in election_form.errors %}role_error{% endif %}"
|
||||
>
|
||||
<tr>
|
||||
<td class="role_title">
|
||||
<div class="role_text">
|
||||
<h4>{{ role.title }}</h4>
|
||||
<p class="role_description" show-more="300">{{ role.description }}</p>
|
||||
{%- if role.max_choice > 1 and not election.has_voted(user) and election.can_vote(user) %}
|
||||
<strong>{% trans %}You may choose up to{% endtrans %} {{ role.max_choice }} {% trans %}people.{% endtrans %}</strong>
|
||||
{%- if role.max_choice > 1 and show_vote_buttons %}
|
||||
<strong>
|
||||
{% trans trimmed nb_choices=role.max_choice %}
|
||||
You may choose up to {{ nb_choices }} people.
|
||||
{% endtrans %}
|
||||
</strong>
|
||||
{%- endif %}
|
||||
|
||||
{%- if election_form.errors[role.title] is defined %}
|
||||
@@ -81,36 +88,40 @@
|
||||
</div>
|
||||
{% if user.can_edit(role) and election.is_vote_editable -%}
|
||||
<div class="role_buttons">
|
||||
<a href="{{url('election:update_role', role_id=role.id)}}">️<i class="fa-regular fa-pen-to-square edit-action"></i></a>
|
||||
<a href="{{url('election:delete_role', role_id=role.id)}}"><i class="fa-regular fa-trash-can delete-action"></i></a>
|
||||
{%- if role == role_list.last() %}
|
||||
<a href="{{ url('election:update_role', role_id=role.id) }}">️
|
||||
<i class="fa-regular fa-pen-to-square edit-action"></i>
|
||||
</a>
|
||||
<a href="{{ url('election:delete_role', role_id=role.id) }}">
|
||||
<i class="fa-regular fa-trash-can delete-action"></i>
|
||||
</a>
|
||||
{%- if loop.last -%}
|
||||
<button disabled><i class="fa fa-arrow-down"></i></button>
|
||||
<button disabled><i class="fa fa-caret-down"></i></button>
|
||||
{%- else %}
|
||||
{%- else -%}
|
||||
<button type="button" onclick="window.location.replace('?role={{ role.id }}&action=bottom');"><i class="fa fa-arrow-down"></i></button>
|
||||
<button type="button" onclick="window.location.replace('?role={{ role.id }}&action=down');"><i class="fa fa-caret-down"></i></button>
|
||||
{%- endif %}
|
||||
{% if role == role_list.first() %}
|
||||
{%- endif -%}
|
||||
{%- if loop.first -%}
|
||||
<button disabled><i class="fa fa-caret-up"></i></button>
|
||||
<button disabled><i class="fa fa-arrow-up"></i></button>
|
||||
{% else %}
|
||||
{%- else -%}
|
||||
<button type="button" onclick="window.location.replace('?role={{ role.id }}&action=up');"><i class="fa fa-caret-up"></i></button>
|
||||
<button type="button" onclick="window.location.replace('?role={{ role.id }}&action=top');"><i class="fa fa-arrow-up"></i></button>
|
||||
{% endif %}
|
||||
{%- endif -%}
|
||||
</div>
|
||||
{%- endif -%}
|
||||
</td>
|
||||
</tr>
|
||||
<tr class="role_candidates">
|
||||
<td class="list_per_role" style="width: 100%; max-width: {{ 100 / (election_lists.count() + 1) }}%">
|
||||
{%- if role.max_choice == 1 and election.can_vote(user) %}
|
||||
<td class="list_per_role" style="width: 100%; max-width: {{ 100 / (election_lists|length + 1) }}%">
|
||||
{%- if role.max_choice == 1 and show_vote_buttons %}
|
||||
<div class="radio-btn">
|
||||
<input id="id_{{ role.title }}_{{ count[0] }}" type="radio" name="{{ role.title }}" value {{ '' if role_data in election_form else 'checked' }} {{ 'disabled' if election.has_voted(user) else '' }}>
|
||||
<label for="id_{{ role.title }}_{{ count[0] }}">
|
||||
{% set input_id = "blank_vote_" + role.id|string %}
|
||||
<input id="{{ input_id }}" type="radio" name="{{ role.title }}">
|
||||
<label for="{{ input_id }}">
|
||||
<span>{% trans %}Choose blank vote{% endtrans %}</span>
|
||||
</label>
|
||||
</div>
|
||||
{%- set _ = count.append(count.pop() + 1) %}
|
||||
{%- endif %}
|
||||
{%- if election.is_vote_finished %}
|
||||
{%- set results = election_results[role.title]['blank vote'] %}
|
||||
@@ -120,13 +131,14 @@
|
||||
{%- endif %}
|
||||
</td>
|
||||
{%- for election_list in election_lists %}
|
||||
<td class="list_per_role" style="width: 100%; max-width: {{ 100 / (election_lists.count() + 1) }}%">
|
||||
<td class="list_per_role" style="width: 100%; max-width: {{ 100 / (election_lists|length + 1) }}%">
|
||||
<ul class="candidates">
|
||||
{%- for candidature in election_list.candidatures.filter(role=role) %}
|
||||
{%- for candidature in election_list.candidatures.select_related("user", "user__profile_pict").filter(role=role) %}
|
||||
<li class="candidate">
|
||||
{%- if election.can_vote(user) %}
|
||||
<input id="id_{{ role.title }}_{{ count[0] }}" type="{{ 'checkbox' if role.max_choice > 1 else 'radio' }}" {{ 'checked' if candidature.id|string in role_data else '' }} {{ 'disabled' if election.has_voted(user) else '' }} name="{{ role.title }}" value="{{ candidature.id }}">
|
||||
<label for="id_{{ role.title }}_{{ count[0] }}">
|
||||
{%- if show_vote_buttons %}
|
||||
{% set input_id = "candidature_" + candidature.id|string %}
|
||||
<input id="{{ input_id }}" type="{{ 'checkbox' if role.max_choice > 1 else 'radio' }}" {{ 'checked' if candidature.id|string in role_data else '' }} {{ 'disabled' if user_has_voted else '' }} name="{{ role.title }}" value="{{ candidature.id }}">
|
||||
<label for="{{ input_id }}">
|
||||
{%- endif %}
|
||||
<figure>
|
||||
{%- if user.is_subscriber_viewable %}
|
||||
@@ -140,7 +152,7 @@
|
||||
<h5>{{ candidature.user.first_name }} <em>{{candidature.user.nick_name or ''}} </em>{{ candidature.user.last_name }}</h5>
|
||||
{%- if not election.is_vote_finished %}
|
||||
<q class="candidate_program" show-more="200">
|
||||
{{ candidature.program|markdown or '' }}
|
||||
{{ candidature.program|markdown }}
|
||||
</q>
|
||||
{%- endif %}
|
||||
</figcaption>
|
||||
@@ -153,9 +165,8 @@
|
||||
{%- endif -%}
|
||||
{%- endif -%}
|
||||
</figure>
|
||||
{%- if election.can_vote(user) %}
|
||||
{%- if show_vote_buttons %}
|
||||
</label>
|
||||
{%- set _ = count.append(count.pop() + 1) %}
|
||||
{%- endif %}
|
||||
{%- if election.is_vote_finished %}
|
||||
{%- set results = election_results[role.title][candidature.user.username] %}
|
||||
@@ -191,36 +202,9 @@
|
||||
<a class="button" href="{{ url('election:delete', election_id=object.id) }}">{% trans %}Delete{% endtrans %}</a>
|
||||
{%- endif %}
|
||||
</section>
|
||||
{%- if not election.has_voted(user) and election.can_vote(user) %}
|
||||
{%- if show_vote_buttons %}
|
||||
<section class="buttons">
|
||||
<button class="button button_send" form="vote-form">{% trans %}Submit the vote !{% endtrans %}</button>
|
||||
</section>
|
||||
{%- endif %}
|
||||
{% endblock %}
|
||||
|
||||
{% block script %}
|
||||
{{ super() }}
|
||||
<script type="text/javascript">
|
||||
document.querySelectorAll('.role__multiple-choices').forEach(setupRestrictions);
|
||||
|
||||
function setupRestrictions(role) {
|
||||
var selectedChoices = [];
|
||||
role.querySelectorAll('input').forEach(setupRestriction);
|
||||
|
||||
function setupRestriction(choice) {
|
||||
if (choice.checked)
|
||||
selectedChoices.push(choice);
|
||||
choice.addEventListener('change', onChange);
|
||||
|
||||
function onChange() {
|
||||
if (choice.checked)
|
||||
selectedChoices.push(choice);
|
||||
else
|
||||
selectedChoices.splice(selectedChoices.indexOf(choice), 1);
|
||||
while (selectedChoices.length > role.dataset.maxChoice)
|
||||
selectedChoices.shift().checked = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
@@ -1,9 +1,15 @@
|
||||
from django.conf import settings
|
||||
from django.test import TestCase
|
||||
from django.urls import reverse
|
||||
from datetime import timedelta
|
||||
|
||||
import pytest
|
||||
from django.conf import settings
|
||||
from django.test import Client, TestCase
|
||||
from django.urls import reverse
|
||||
from django.utils.timezone import now
|
||||
from model_bakery import baker
|
||||
|
||||
from core.baker_recipes import subscriber_user
|
||||
from core.models import Group, User
|
||||
from election.models import Election
|
||||
from election.models import Candidature, Election, ElectionList, Role, Vote
|
||||
|
||||
|
||||
class TestElection(TestCase):
|
||||
@@ -12,8 +18,7 @@ class TestElection(TestCase):
|
||||
cls.election = Election.objects.first()
|
||||
cls.public_group = Group.objects.get(id=settings.SITH_GROUP_PUBLIC_ID)
|
||||
cls.sli = User.objects.get(username="sli")
|
||||
cls.subscriber = User.objects.get(username="subscriber")
|
||||
cls.public = User.objects.get(username="public")
|
||||
cls.public = baker.make(User)
|
||||
|
||||
|
||||
class TestElectionDetail(TestElection):
|
||||
@@ -36,7 +41,7 @@ class TestElectionDetail(TestElection):
|
||||
|
||||
class TestElectionUpdateView(TestElection):
|
||||
def test_permission_denied(self):
|
||||
self.client.force_login(self.subscriber)
|
||||
self.client.force_login(subscriber_user.make())
|
||||
response = self.client.get(
|
||||
reverse("election:update", args=str(self.election.id))
|
||||
)
|
||||
@@ -45,3 +50,68 @@ class TestElectionUpdateView(TestElection):
|
||||
reverse("election:update", args=str(self.election.id))
|
||||
)
|
||||
assert response.status_code == 403
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_election_create_list_permission(client: Client):
|
||||
election = baker.make(Election, end_candidature=now() + timedelta(hours=1))
|
||||
groups = [
|
||||
Group.objects.get(pk=settings.SITH_GROUP_SUBSCRIBERS_ID),
|
||||
baker.make(Group),
|
||||
]
|
||||
election.candidature_groups.add(groups[0])
|
||||
election.edit_groups.add(groups[1])
|
||||
url = reverse("election:create_list", kwargs={"election_id": election.id})
|
||||
for user in subscriber_user.make(), baker.make(User, groups=[groups[1]]):
|
||||
client.force_login(user)
|
||||
assert client.get(url).status_code == 200
|
||||
# the post is a 200 instead of a 302, because we don't give form data,
|
||||
# but we don't care as we only test permissions here
|
||||
assert client.post(url).status_code == 200
|
||||
client.force_login(baker.make(User))
|
||||
assert client.get(url).status_code == 403
|
||||
assert client.post(url).status_code == 403
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_election_results():
|
||||
election = baker.make(
|
||||
Election, voters=baker.make(User, _quantity=50, _bulk_create=True)
|
||||
)
|
||||
lists = baker.make(ElectionList, election=election, _quantity=2, _bulk_create=True)
|
||||
roles = baker.make(
|
||||
Role, election=election, max_choice=iter([1, 2]), _quantity=2, _bulk_create=True
|
||||
)
|
||||
users = baker.make(User, _quantity=4, _bulk_create=True)
|
||||
cand = [
|
||||
baker.make(Candidature, role=roles[0], user=users[0], election_list=lists[0]),
|
||||
baker.make(Candidature, role=roles[0], user=users[1], election_list=lists[1]),
|
||||
baker.make(Candidature, role=roles[1], user=users[2], election_list=lists[0]),
|
||||
baker.make(Candidature, role=roles[1], user=users[3], election_list=lists[1]),
|
||||
]
|
||||
votes = [
|
||||
baker.make(Vote, role=roles[0], _quantity=20, _bulk_create=True),
|
||||
baker.make(Vote, role=roles[0], _quantity=25, _bulk_create=True),
|
||||
baker.make(Vote, role=roles[1], _quantity=20, _bulk_create=True),
|
||||
baker.make(Vote, role=roles[1], _quantity=35, _bulk_create=True),
|
||||
baker.make(Vote, role=roles[1], _quantity=10, _bulk_create=True),
|
||||
]
|
||||
cand[0].votes.set(votes[0])
|
||||
cand[1].votes.set(votes[1])
|
||||
cand[2].votes.set([*votes[2], *votes[4]])
|
||||
cand[3].votes.set([*votes[3], *votes[4]])
|
||||
|
||||
assert election.results == {
|
||||
roles[0].title: {
|
||||
cand[0].user.username: {"percent": 40.0, "vote": 20},
|
||||
cand[1].user.username: {"percent": 50.0, "vote": 25},
|
||||
"blank vote": {"percent": 10.0, "vote": 5},
|
||||
"total vote": 50,
|
||||
},
|
||||
roles[1].title: {
|
||||
cand[2].user.username: {"percent": 30.0, "vote": 30},
|
||||
cand[3].user.username: {"percent": 45.0, "vote": 45},
|
||||
"blank vote": {"percent": 25.0, "vote": 25},
|
||||
"total vote": 100,
|
||||
},
|
||||
}
|
||||
|
@@ -1,183 +1,34 @@
|
||||
from typing import TYPE_CHECKING
|
||||
|
||||
from django import forms
|
||||
from django.contrib.auth.mixins import LoginRequiredMixin, PermissionRequiredMixin
|
||||
from cryptography.utils import cached_property
|
||||
from django.conf import settings
|
||||
from django.contrib.auth.mixins import (
|
||||
LoginRequiredMixin,
|
||||
PermissionRequiredMixin,
|
||||
UserPassesTestMixin,
|
||||
)
|
||||
from django.core.exceptions import PermissionDenied
|
||||
from django.db import transaction
|
||||
from django.db.models.query import QuerySet
|
||||
from django.shortcuts import get_object_or_404, redirect
|
||||
from django.db.models import QuerySet
|
||||
from django.shortcuts import get_object_or_404
|
||||
from django.urls import reverse, reverse_lazy
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
from django.views.generic import DetailView, ListView
|
||||
from django.views.generic.edit import CreateView, DeleteView, FormView, UpdateView
|
||||
|
||||
from core.auth.mixins import CanCreateMixin, CanEditMixin, CanViewMixin
|
||||
from core.views.forms import SelectDateTime
|
||||
from core.views.widgets.ajax_select import (
|
||||
AutoCompleteSelect,
|
||||
AutoCompleteSelectMultipleGroup,
|
||||
AutoCompleteSelectUser,
|
||||
from core.auth.mixins import CanEditMixin, CanViewMixin
|
||||
from election.forms import (
|
||||
CandidateForm,
|
||||
ElectionForm,
|
||||
ElectionListForm,
|
||||
RoleForm,
|
||||
VoteForm,
|
||||
)
|
||||
from core.views.widgets.markdown import MarkdownInput
|
||||
from election.models import Candidature, Election, ElectionList, Role, Vote
|
||||
|
||||
if TYPE_CHECKING:
|
||||
from core.models import User
|
||||
|
||||
|
||||
# Custom form field
|
||||
|
||||
|
||||
class LimitedCheckboxField(forms.ModelMultipleChoiceField):
|
||||
"""A `ModelMultipleChoiceField`, with a max limit of selectable inputs."""
|
||||
|
||||
def __init__(self, queryset, max_choice, **kwargs):
|
||||
self.max_choice = max_choice
|
||||
super().__init__(queryset, **kwargs)
|
||||
|
||||
def clean(self, value):
|
||||
qs = super().clean(value)
|
||||
self.validate(qs)
|
||||
return qs
|
||||
|
||||
def validate(self, qs):
|
||||
if qs.count() > self.max_choice:
|
||||
raise forms.ValidationError(
|
||||
_("You have selected too much candidates."), code="invalid"
|
||||
)
|
||||
|
||||
|
||||
# Forms
|
||||
|
||||
|
||||
class CandidateForm(forms.ModelForm):
|
||||
"""Form to candidate."""
|
||||
|
||||
class Meta:
|
||||
model = Candidature
|
||||
fields = ["user", "role", "program", "election_list"]
|
||||
labels = {
|
||||
"user": _("User to candidate"),
|
||||
}
|
||||
widgets = {
|
||||
"program": MarkdownInput,
|
||||
"user": AutoCompleteSelectUser,
|
||||
"role": AutoCompleteSelect,
|
||||
"election_list": AutoCompleteSelect,
|
||||
}
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
election_id = kwargs.pop("election_id", None)
|
||||
can_edit = kwargs.pop("can_edit", False)
|
||||
super().__init__(*args, **kwargs)
|
||||
if election_id:
|
||||
self.fields["role"].queryset = Role.objects.filter(
|
||||
election__id=election_id
|
||||
).all()
|
||||
self.fields["election_list"].queryset = ElectionList.objects.filter(
|
||||
election__id=election_id
|
||||
).all()
|
||||
if not can_edit:
|
||||
self.fields["user"].widget = forms.HiddenInput()
|
||||
|
||||
|
||||
class VoteForm(forms.Form):
|
||||
def __init__(self, election, user, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
if not election.has_voted(user):
|
||||
for role in election.roles.all():
|
||||
cand = role.candidatures
|
||||
if role.max_choice > 1:
|
||||
self.fields[role.title] = LimitedCheckboxField(
|
||||
cand, role.max_choice, required=False
|
||||
)
|
||||
else:
|
||||
self.fields[role.title] = forms.ModelChoiceField(
|
||||
cand,
|
||||
required=False,
|
||||
widget=forms.RadioSelect(),
|
||||
empty_label=_("Blank vote"),
|
||||
)
|
||||
|
||||
|
||||
class RoleForm(forms.ModelForm):
|
||||
"""Form for creating a role."""
|
||||
|
||||
class Meta:
|
||||
model = Role
|
||||
fields = ["title", "election", "description", "max_choice"]
|
||||
widgets = {"election": AutoCompleteSelect}
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
election_id = kwargs.pop("election_id", None)
|
||||
super().__init__(*args, **kwargs)
|
||||
if election_id:
|
||||
self.fields["election"].queryset = Election.objects.filter(
|
||||
id=election_id
|
||||
).all()
|
||||
|
||||
def clean(self):
|
||||
cleaned_data = super().clean()
|
||||
title = cleaned_data.get("title")
|
||||
election = cleaned_data.get("election")
|
||||
if Role.objects.filter(title=title, election=election).exists():
|
||||
raise forms.ValidationError(
|
||||
_("This role already exists for this election"), code="invalid"
|
||||
)
|
||||
|
||||
|
||||
class ElectionListForm(forms.ModelForm):
|
||||
class Meta:
|
||||
model = ElectionList
|
||||
fields = ("title", "election")
|
||||
widgets = {"election": AutoCompleteSelect}
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
election_id = kwargs.pop("election_id", None)
|
||||
super().__init__(*args, **kwargs)
|
||||
if election_id:
|
||||
self.fields["election"].queryset = Election.objects.filter(
|
||||
id=election_id
|
||||
).all()
|
||||
|
||||
|
||||
class ElectionForm(forms.ModelForm):
|
||||
class Meta:
|
||||
model = Election
|
||||
fields = [
|
||||
"title",
|
||||
"description",
|
||||
"archived",
|
||||
"start_candidature",
|
||||
"end_candidature",
|
||||
"start_date",
|
||||
"end_date",
|
||||
"edit_groups",
|
||||
"view_groups",
|
||||
"vote_groups",
|
||||
"candidature_groups",
|
||||
]
|
||||
widgets = {
|
||||
"edit_groups": AutoCompleteSelectMultipleGroup,
|
||||
"view_groups": AutoCompleteSelectMultipleGroup,
|
||||
"vote_groups": AutoCompleteSelectMultipleGroup,
|
||||
"candidature_groups": AutoCompleteSelectMultipleGroup,
|
||||
}
|
||||
|
||||
start_date = forms.DateTimeField(
|
||||
label=_("Start date"), widget=SelectDateTime, required=True
|
||||
)
|
||||
end_date = forms.DateTimeField(
|
||||
label=_("End date"), widget=SelectDateTime, required=True
|
||||
)
|
||||
start_candidature = forms.DateTimeField(
|
||||
label=_("Start candidature"), widget=SelectDateTime, required=True
|
||||
)
|
||||
end_candidature = forms.DateTimeField(
|
||||
label=_("End candidature"), widget=SelectDateTime, required=True
|
||||
)
|
||||
|
||||
|
||||
# Display elections
|
||||
|
||||
|
||||
@@ -185,25 +36,21 @@ class ElectionsListView(CanViewMixin, ListView):
|
||||
"""A list of all non archived elections visible."""
|
||||
|
||||
model = Election
|
||||
queryset = model.objects.filter(archived=False)
|
||||
ordering = ["-id"]
|
||||
paginate_by = 10
|
||||
template_name = "election/election_list.jinja"
|
||||
|
||||
def get_queryset(self):
|
||||
return super().get_queryset().filter(archived=False).all()
|
||||
|
||||
|
||||
class ElectionListArchivedView(CanViewMixin, ListView):
|
||||
"""A list of all archived elections visible."""
|
||||
|
||||
model = Election
|
||||
queryset = model.objects.filter(archived=True)
|
||||
ordering = ["-id"]
|
||||
paginate_by = 10
|
||||
template_name = "election/election_list.jinja"
|
||||
|
||||
def get_queryset(self):
|
||||
return super().get_queryset().filter(archived=True).all()
|
||||
|
||||
|
||||
class ElectionDetailView(CanViewMixin, DetailView):
|
||||
"""Details an election responsability by responsability."""
|
||||
@@ -212,46 +59,67 @@ class ElectionDetailView(CanViewMixin, DetailView):
|
||||
template_name = "election/election_detail.jinja"
|
||||
pk_url_kwarg = "election_id"
|
||||
|
||||
@staticmethod
|
||||
def _reorder_votes(action: str, role: int):
|
||||
role = Role.objects.filter(id=role).first()
|
||||
if not role:
|
||||
return
|
||||
if action == "up":
|
||||
role.up()
|
||||
elif action == "down":
|
||||
role.down()
|
||||
elif action == "bottom":
|
||||
role.bottom()
|
||||
elif action == "top":
|
||||
role.top()
|
||||
|
||||
def get(self, request, *arg, **kwargs):
|
||||
response = super().get(request, *arg, **kwargs)
|
||||
election: Election = self.get_object()
|
||||
if request.user.can_edit(election) and election.is_vote_editable:
|
||||
if election.is_vote_editable and request.user.can_edit(election):
|
||||
action = request.GET.get("action", None)
|
||||
role = request.GET.get("role", None)
|
||||
if action and role and Role.objects.filter(id=role).exists():
|
||||
if action == "up":
|
||||
Role.objects.get(id=role).up()
|
||||
elif action == "down":
|
||||
Role.objects.get(id=role).down()
|
||||
elif action == "bottom":
|
||||
Role.objects.get(id=role).bottom()
|
||||
elif action == "top":
|
||||
Role.objects.get(id=role).top()
|
||||
return redirect(
|
||||
reverse("election:detail", kwargs={"election_id": election.id})
|
||||
)
|
||||
return response
|
||||
if action and role and role.isdigit():
|
||||
self._reorder_votes(action, int(role))
|
||||
return super().get(request, *arg, **kwargs)
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
"""Add additionnal data to the template."""
|
||||
kwargs = super().get_context_data(**kwargs)
|
||||
kwargs["election_form"] = VoteForm(self.object, self.request.user)
|
||||
kwargs["election_results"] = self.object.results
|
||||
return kwargs
|
||||
user: User = self.request.user
|
||||
return super().get_context_data(**kwargs) | {
|
||||
"election_form": VoteForm(self.object, user),
|
||||
"show_vote_buttons": self.object.can_vote(user),
|
||||
"user_has_voted": self.object.has_voted(user),
|
||||
"election_results": (
|
||||
self.object.results if self.object.is_vote_finished else None
|
||||
),
|
||||
"election_lists": list(self.object.election_lists.all()),
|
||||
"election_roles": list(self.object.roles.order_by("order")),
|
||||
}
|
||||
|
||||
|
||||
# Form view
|
||||
|
||||
|
||||
class VoteFormView(CanCreateMixin, FormView):
|
||||
class VoteFormView(LoginRequiredMixin, UserPassesTestMixin, FormView):
|
||||
"""Alows users to vote."""
|
||||
|
||||
form_class = VoteForm
|
||||
template_name = "election/election_detail.jinja"
|
||||
|
||||
def dispatch(self, request, *arg, **kwargs):
|
||||
self.election = get_object_or_404(Election, pk=kwargs["election_id"])
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
@cached_property
|
||||
def election(self):
|
||||
return get_object_or_404(Election, pk=self.kwargs["election_id"])
|
||||
|
||||
def test_func(self):
|
||||
groups = set(self.election.vote_groups.values_list("id", flat=True))
|
||||
if (
|
||||
settings.SITH_GROUP_SUBSCRIBERS_ID in groups
|
||||
and self.request.user.is_subscribed
|
||||
):
|
||||
# the subscriber group isn't truly attached to users,
|
||||
# so it must be dealt with separately
|
||||
return True
|
||||
return self.request.user.groups.filter(id__in=groups).exists()
|
||||
|
||||
def vote(self, election_data):
|
||||
with transaction.atomic():
|
||||
@@ -271,20 +139,16 @@ class VoteFormView(CanCreateMixin, FormView):
|
||||
self.election.voters.add(self.request.user)
|
||||
|
||||
def get_form_kwargs(self):
|
||||
kwargs = super().get_form_kwargs()
|
||||
kwargs["election"] = self.election
|
||||
kwargs["user"] = self.request.user
|
||||
return kwargs
|
||||
return super().get_form_kwargs() | {
|
||||
"election": self.election,
|
||||
"user": self.request.user,
|
||||
}
|
||||
|
||||
def form_valid(self, form):
|
||||
"""Verify that the user is part in a vote group."""
|
||||
data = form.clean()
|
||||
res = super(FormView, self).form_valid(form)
|
||||
for grp_id in self.election.vote_groups.values_list("pk", flat=True):
|
||||
if self.request.user.is_in_group(pk=grp_id):
|
||||
self.vote(data)
|
||||
return res
|
||||
return res
|
||||
return super().form_valid(form)
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy("election:detail", kwargs={"election_id": self.election.id})
|
||||
@@ -310,26 +174,22 @@ class CandidatureCreateView(LoginRequiredMixin, CreateView):
|
||||
|
||||
def dispatch(self, request, *arg, **kwargs):
|
||||
self.election = get_object_or_404(Election, pk=kwargs["election_id"])
|
||||
self.can_edit = self.request.user.can_edit(self.election)
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
|
||||
def get_initial(self):
|
||||
init = {}
|
||||
self.can_edit = self.request.user.can_edit(self.election)
|
||||
init["user"] = self.request.user.id
|
||||
return init
|
||||
return {"user": self.request.user.id}
|
||||
|
||||
def get_form_kwargs(self):
|
||||
kwargs = super().get_form_kwargs()
|
||||
kwargs["election_id"] = self.election.id
|
||||
kwargs["can_edit"] = self.can_edit
|
||||
return kwargs
|
||||
return super().get_form_kwargs() | {
|
||||
"election": self.election,
|
||||
"can_edit": self.can_edit,
|
||||
}
|
||||
|
||||
def form_valid(self, form):
|
||||
def form_valid(self, form: CandidateForm):
|
||||
"""Verify that the selected user is in candidate group."""
|
||||
obj = form.instance
|
||||
obj.election = self.election
|
||||
if not hasattr(obj, "user"):
|
||||
obj.user = self.request.user
|
||||
if (obj.election.can_candidate(obj.user)) and (
|
||||
obj.user == self.request.user or self.can_edit
|
||||
):
|
||||
@@ -337,9 +197,7 @@ class CandidatureCreateView(LoginRequiredMixin, CreateView):
|
||||
raise PermissionDenied
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
kwargs = super().get_context_data(**kwargs)
|
||||
kwargs["election"] = self.election
|
||||
return kwargs
|
||||
return super().get_context_data(**kwargs) | {"election": self.election}
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy("election:detail", kwargs={"election_id": self.election.id})
|
||||
@@ -355,80 +213,79 @@ class ElectionCreateView(PermissionRequiredMixin, CreateView):
|
||||
return reverse("election:detail", kwargs={"election_id": self.object.id})
|
||||
|
||||
|
||||
class RoleCreateView(CanCreateMixin, CreateView):
|
||||
class RoleCreateView(LoginRequiredMixin, UserPassesTestMixin, CreateView):
|
||||
model = Role
|
||||
form_class = RoleForm
|
||||
template_name = "core/create.jinja"
|
||||
|
||||
def dispatch(self, request, *arg, **kwargs):
|
||||
self.election = get_object_or_404(Election, pk=kwargs["election_id"])
|
||||
@cached_property
|
||||
def election(self):
|
||||
return get_object_or_404(Election, pk=self.kwargs["election_id"])
|
||||
|
||||
def test_func(self):
|
||||
if not self.election.is_vote_editable:
|
||||
raise PermissionDenied
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
return False
|
||||
if self.request.user.has_perm("election.add_role"):
|
||||
return True
|
||||
groups = set(self.election.edit_groups.values_list("id", flat=True))
|
||||
if (
|
||||
settings.SITH_GROUP_SUBSCRIBERS_ID in groups
|
||||
and self.request.user.is_subscribed
|
||||
):
|
||||
# the subscriber group isn't truly attached to users,
|
||||
# so it must be dealt with separately
|
||||
return True
|
||||
return self.request.user.groups.filter(id__in=groups).exists()
|
||||
|
||||
def get_initial(self):
|
||||
init = {}
|
||||
init["election"] = self.election
|
||||
return init
|
||||
|
||||
def form_valid(self, form):
|
||||
"""Verify that the user can edit properly."""
|
||||
obj: Role = form.instance
|
||||
user: User = self.request.user
|
||||
if obj.election:
|
||||
for grp_id in obj.election.edit_groups.values_list("pk", flat=True):
|
||||
if user.is_in_group(pk=grp_id):
|
||||
return super(CreateView, self).form_valid(form)
|
||||
raise PermissionDenied
|
||||
return {"election": self.election}
|
||||
|
||||
def get_form_kwargs(self):
|
||||
kwargs = super().get_form_kwargs()
|
||||
kwargs["election_id"] = self.election.id
|
||||
return kwargs
|
||||
return super().get_form_kwargs() | {"election_id": self.election.id}
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy(
|
||||
"election:detail", kwargs={"election_id": self.object.election.id}
|
||||
return reverse(
|
||||
"election:detail", kwargs={"election_id": self.object.election_id}
|
||||
)
|
||||
|
||||
|
||||
class ElectionListCreateView(CanCreateMixin, CreateView):
|
||||
class ElectionListCreateView(LoginRequiredMixin, UserPassesTestMixin, CreateView):
|
||||
model = ElectionList
|
||||
form_class = ElectionListForm
|
||||
template_name = "core/create.jinja"
|
||||
|
||||
def dispatch(self, request, *arg, **kwargs):
|
||||
self.election = get_object_or_404(Election, pk=kwargs["election_id"])
|
||||
@cached_property
|
||||
def election(self):
|
||||
return get_object_or_404(Election, pk=self.kwargs["election_id"])
|
||||
|
||||
def test_func(self):
|
||||
if not self.election.is_vote_editable:
|
||||
raise PermissionDenied
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
return False
|
||||
if self.request.user.has_perm("election.add_electionlist"):
|
||||
return True
|
||||
groups = set(
|
||||
self.election.candidature_groups.values("id")
|
||||
.union(self.election.edit_groups.values("id"))
|
||||
.values_list("id", flat=True)
|
||||
)
|
||||
if (
|
||||
settings.SITH_GROUP_SUBSCRIBERS_ID in groups
|
||||
and self.request.user.is_subscribed
|
||||
):
|
||||
# the subscriber group isn't truly attached to users,
|
||||
# so it must be dealt with separately
|
||||
return True
|
||||
return self.request.user.groups.filter(id__in=groups).exists()
|
||||
|
||||
def get_initial(self):
|
||||
init = {}
|
||||
init["election"] = self.election
|
||||
return init
|
||||
return {"election": self.election}
|
||||
|
||||
def get_form_kwargs(self):
|
||||
kwargs = super().get_form_kwargs()
|
||||
kwargs["election_id"] = self.election.id
|
||||
return kwargs
|
||||
|
||||
def form_valid(self, form):
|
||||
"""Verify that the user can vote on this election."""
|
||||
obj: ElectionList = form.instance
|
||||
user: User = self.request.user
|
||||
if obj.election:
|
||||
for grp_id in obj.election.candidature_groups.values_list("pk", flat=True):
|
||||
if user.is_in_group(pk=grp_id):
|
||||
return super(CreateView, self).form_valid(form)
|
||||
for grp_id in obj.election.edit_groups.values_list("pk", flat=True):
|
||||
if user.is_in_group(pk=grp_id):
|
||||
return super(CreateView, self).form_valid(form)
|
||||
raise PermissionDenied
|
||||
return super().get_form_kwargs() | {"election_id": self.election.id}
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy(
|
||||
"election:detail", kwargs={"election_id": self.object.election.id}
|
||||
return reverse(
|
||||
"election:detail", kwargs={"election_id": self.object.election_id}
|
||||
)
|
||||
|
||||
|
||||
@@ -457,45 +314,23 @@ class ElectionUpdateView(CanEditMixin, UpdateView):
|
||||
return reverse_lazy("election:detail", kwargs={"election_id": self.object.id})
|
||||
|
||||
|
||||
class CandidatureUpdateView(CanEditMixin, UpdateView):
|
||||
class CandidatureUpdateView(LoginRequiredMixin, CanEditMixin, UpdateView):
|
||||
model = Candidature
|
||||
form_class = CandidateForm
|
||||
template_name = "core/edit.jinja"
|
||||
pk_url_kwarg = "candidature_id"
|
||||
|
||||
def dispatch(self, request, *arg, **kwargs):
|
||||
self.object = self.get_object()
|
||||
if not self.object.role.election.is_vote_editable:
|
||||
raise PermissionDenied
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
|
||||
def remove_fields(self):
|
||||
self.form.fields.pop("role", None)
|
||||
|
||||
def get(self, request, *args, **kwargs):
|
||||
self.form = self.get_form()
|
||||
self.remove_fields()
|
||||
return self.render_to_response(self.get_context_data(form=self.form))
|
||||
|
||||
def post(self, request, *args, **kwargs):
|
||||
self.form = self.get_form()
|
||||
self.remove_fields()
|
||||
if (
|
||||
request.user.is_authenticated
|
||||
and request.user.can_edit(self.object)
|
||||
and self.form.is_valid()
|
||||
):
|
||||
return super().form_valid(self.form)
|
||||
return self.form_invalid(self.form)
|
||||
def get_form(self, *args, **kwargs):
|
||||
form = super().get_form(*args, **kwargs)
|
||||
form.fields.pop("role", None)
|
||||
return form
|
||||
|
||||
def get_form_kwargs(self):
|
||||
kwargs = super().get_form_kwargs()
|
||||
kwargs["election_id"] = self.object.role.election.id
|
||||
return kwargs
|
||||
return super().get_form_kwargs() | {"election": self.object.role.election}
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy(
|
||||
"election:detail", kwargs={"election_id": self.object.role.election.id}
|
||||
return reverse(
|
||||
"election:detail", kwargs={"election_id": self.object.role.election_id}
|
||||
)
|
||||
|
||||
|
||||
@@ -546,18 +381,12 @@ class RoleUpdateView(CanEditMixin, UpdateView):
|
||||
# Delete Views
|
||||
|
||||
|
||||
class ElectionDeleteView(DeleteView):
|
||||
class ElectionDeleteView(PermissionRequiredMixin, DeleteView):
|
||||
model = Election
|
||||
template_name = "core/delete_confirm.jinja"
|
||||
pk_url_kwarg = "election_id"
|
||||
|
||||
def dispatch(self, request, *args, **kwargs):
|
||||
if request.user.is_root:
|
||||
return super().dispatch(request, *args, **kwargs)
|
||||
raise PermissionDenied
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy("election:list")
|
||||
permission_required = "election.delete_election"
|
||||
success_url = reverse_lazy("election:list")
|
||||
|
||||
|
||||
class CandidatureDeleteView(CanEditMixin, DeleteView):
|
||||
@@ -573,7 +402,7 @@ class CandidatureDeleteView(CanEditMixin, DeleteView):
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy("election:detail", kwargs={"election_id": self.election.id})
|
||||
return reverse("election:detail", kwargs={"election_id": self.election.id})
|
||||
|
||||
|
||||
class RoleDeleteView(CanEditMixin, DeleteView):
|
||||
@@ -589,7 +418,7 @@ class RoleDeleteView(CanEditMixin, DeleteView):
|
||||
return super().dispatch(request, *arg, **kwargs)
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy("election:detail", kwargs={"election_id": self.election.id})
|
||||
return reverse("election:detail", kwargs={"election_id": self.election.id})
|
||||
|
||||
|
||||
class ElectionListDeleteView(CanEditMixin, DeleteView):
|
||||
@@ -605,4 +434,4 @@ class ElectionListDeleteView(CanEditMixin, DeleteView):
|
||||
return super().dispatch(request, *args, **kwargs)
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
return reverse_lazy("election:detail", kwargs={"election_id": self.election.id})
|
||||
return reverse("election:detail", kwargs={"election_id": self.election.id})
|
||||
|
@@ -6,7 +6,7 @@
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Report-Msgid-Bugs-To: \n"
|
||||
"POT-Creation-Date: 2025-09-19 17:22+0200\n"
|
||||
"POT-Creation-Date: 2025-09-25 15:33+0200\n"
|
||||
"PO-Revision-Date: 2016-07-18\n"
|
||||
"Last-Translator: Maréchal <thomas.girod@utbm.fr\n"
|
||||
"Language-Team: AE info <ae.info@utbm.fr>\n"
|
||||
@@ -141,7 +141,7 @@ msgstr "vous devez spécifier au moins un utilisateur ou une adresse email"
|
||||
msgid "Begin date"
|
||||
msgstr "Date de début"
|
||||
|
||||
#: club/forms.py com/forms.py counter/forms.py election/views.py
|
||||
#: club/forms.py com/forms.py counter/forms.py election/forms.py
|
||||
#: subscription/forms.py
|
||||
msgid "End date"
|
||||
msgstr "Date de fin"
|
||||
@@ -679,7 +679,7 @@ msgstr "Listes de diffusion"
|
||||
msgid "Format: 16:9 | Resolution: 1920x1080"
|
||||
msgstr "Format : 16:9 | Résolution : 1920x1080"
|
||||
|
||||
#: com/forms.py election/views.py subscription/forms.py
|
||||
#: com/forms.py election/forms.py subscription/forms.py
|
||||
msgid "Start date"
|
||||
msgstr "Date de début"
|
||||
|
||||
@@ -1103,6 +1103,10 @@ msgstr "Modération"
|
||||
msgid "No posters"
|
||||
msgstr "Aucune affiche"
|
||||
|
||||
#: com/templates/com/poster_list.jinja com/templates/com/screen_slideshow.jinja
|
||||
msgid "Click to expand"
|
||||
msgstr "Cliquez pour agrandir"
|
||||
|
||||
#: com/templates/com/poster_moderate.jinja
|
||||
msgid "Posters - moderation"
|
||||
msgstr "Affiches - modération"
|
||||
@@ -1160,14 +1164,6 @@ msgstr "Contenu"
|
||||
msgid "Add to weekmail"
|
||||
msgstr "Ajouter au Weekmail"
|
||||
|
||||
#: com/templates/com/weekmail.jinja
|
||||
msgid "Up"
|
||||
msgstr "Monter"
|
||||
|
||||
#: com/templates/com/weekmail.jinja
|
||||
msgid "Down"
|
||||
msgstr "Descendre"
|
||||
|
||||
#: com/templates/com/weekmail.jinja
|
||||
msgid "Articles included the next weekmail"
|
||||
msgstr "Article inclus dans le prochain Weekmail"
|
||||
@@ -1176,6 +1172,14 @@ msgstr "Article inclus dans le prochain Weekmail"
|
||||
msgid "Delete from weekmail"
|
||||
msgstr "Supprimer du Weekmail"
|
||||
|
||||
#: com/templates/com/weekmail.jinja
|
||||
msgid "Up"
|
||||
msgstr "Monter"
|
||||
|
||||
#: com/templates/com/weekmail.jinja
|
||||
msgid "Down"
|
||||
msgstr "Descendre"
|
||||
|
||||
#: com/templates/com/weekmail_preview.jinja
|
||||
#: core/templates/core/user_account_detail.jinja
|
||||
#: pedagogy/templates/pedagogy/uv_detail.jinja
|
||||
@@ -1257,6 +1261,10 @@ msgstr "Liste d'écrans"
|
||||
msgid "All incoming events"
|
||||
msgstr "Tous les événements à venir"
|
||||
|
||||
#: com/views.py
|
||||
msgid "Weekmail sent successfully"
|
||||
msgstr "Weekmail envoyé avec succès"
|
||||
|
||||
#: com/views.py
|
||||
msgid "Delete and save to regenerate"
|
||||
msgstr "Supprimer et sauver pour régénérer"
|
||||
@@ -1265,6 +1273,26 @@ msgstr "Supprimer et sauver pour régénérer"
|
||||
msgid "Weekmail of the "
|
||||
msgstr "Weekmail du "
|
||||
|
||||
#: com/views.py
|
||||
#, python-format
|
||||
msgid "%(title)s moved up in the Weekmail"
|
||||
msgstr "%(title)s monté dans le Weekmail"
|
||||
|
||||
#: com/views.py
|
||||
#, python-format
|
||||
msgid "%(title)s moved down in the Weekmail"
|
||||
msgstr "%(title)s descendu dans le Weekmail"
|
||||
|
||||
#: com/views.py
|
||||
#, python-format
|
||||
msgid "%(title)s added to the Weekmail"
|
||||
msgstr "%(title)s ajouté dans Weekmail"
|
||||
|
||||
#: com/views.py
|
||||
#, python-format
|
||||
msgid "%(title)s removed from the Weekmail"
|
||||
msgstr "%(title)s retiré du Weekmail"
|
||||
|
||||
#: com/views.py
|
||||
msgid ""
|
||||
"You must be a board member of the selected club to post in the Weekmail."
|
||||
@@ -2340,6 +2368,10 @@ msgstr "Etickets"
|
||||
msgid "User has no account"
|
||||
msgstr "L'utilisateur n'a pas de compte"
|
||||
|
||||
#: core/templates/core/user_account_detail.jinja
|
||||
msgid "Deleted user"
|
||||
msgstr "Utilisateur supprimé"
|
||||
|
||||
#: core/templates/core/user_account_detail.jinja
|
||||
#: counter/templates/counter/last_ops.jinja
|
||||
#: counter/templates/counter/refilling_list.jinja
|
||||
@@ -3898,6 +3930,30 @@ msgstr ""
|
||||
msgid "You can't buy a refilling with sith money"
|
||||
msgstr "Vous ne pouvez pas acheter un rechargement avec de l'argent du sith"
|
||||
|
||||
#: election/forms.py
|
||||
msgid "You have selected too many candidates."
|
||||
msgstr "Vous avez sélectionné trop de candidats."
|
||||
|
||||
#: election/forms.py
|
||||
msgid "User to candidate"
|
||||
msgstr "Utilisateur se présentant"
|
||||
|
||||
#: election/forms.py election/templates/election/election_detail.jinja
|
||||
msgid "Blank vote"
|
||||
msgstr "Vote blanc"
|
||||
|
||||
#: election/forms.py
|
||||
msgid "This role already exists for this election"
|
||||
msgstr "Ce rôle existe déjà pour cette élection"
|
||||
|
||||
#: election/forms.py
|
||||
msgid "Start candidature"
|
||||
msgstr "Début des candidatures"
|
||||
|
||||
#: election/forms.py
|
||||
msgid "End candidature"
|
||||
msgstr "Fin des candidatures"
|
||||
|
||||
#: election/models.py
|
||||
msgid "start candidature"
|
||||
msgstr "début des candidatures"
|
||||
@@ -3922,6 +3978,10 @@ msgstr "groupe de vote"
|
||||
msgid "candidature groups"
|
||||
msgstr "groupe de candidature"
|
||||
|
||||
#: election/models.py
|
||||
msgid "voters"
|
||||
msgstr "électeurs"
|
||||
|
||||
#: election/models.py
|
||||
msgid "election"
|
||||
msgstr "élection"
|
||||
@@ -3977,17 +4037,10 @@ msgstr "Vous avez déjà soumis votre vote."
|
||||
msgid "You have voted in this election."
|
||||
msgstr "Vous avez déjà voté pour cette élection."
|
||||
|
||||
#: election/templates/election/election_detail.jinja election/views.py
|
||||
msgid "Blank vote"
|
||||
msgstr "Vote blanc"
|
||||
|
||||
#: election/templates/election/election_detail.jinja
|
||||
msgid "You may choose up to"
|
||||
msgstr "Vous pouvez choisir jusqu'à"
|
||||
|
||||
#: election/templates/election/election_detail.jinja
|
||||
msgid "people."
|
||||
msgstr "personne(s)"
|
||||
#, python-format
|
||||
msgid "You may choose up to %(nb_choices)s people."
|
||||
msgstr "Vous pouvez choisir jusqu'à %(nb_choices)s personnes."
|
||||
|
||||
#: election/templates/election/election_detail.jinja
|
||||
msgid "Choose blank vote"
|
||||
@@ -4029,26 +4082,6 @@ msgstr "au"
|
||||
msgid "Polls open from"
|
||||
msgstr "Votes ouverts du"
|
||||
|
||||
#: election/views.py
|
||||
msgid "You have selected too much candidates."
|
||||
msgstr "Vous avez sélectionné trop de candidats."
|
||||
|
||||
#: election/views.py
|
||||
msgid "User to candidate"
|
||||
msgstr "Utilisateur se présentant"
|
||||
|
||||
#: election/views.py
|
||||
msgid "This role already exists for this election"
|
||||
msgstr "Ce rôle existe déjà pour cette élection"
|
||||
|
||||
#: election/views.py
|
||||
msgid "Start candidature"
|
||||
msgstr "Début des candidatures"
|
||||
|
||||
#: election/views.py
|
||||
msgid "End candidature"
|
||||
msgstr "Fin des candidatures"
|
||||
|
||||
#: forum/models.py
|
||||
msgid "is a category"
|
||||
msgstr "est une catégorie"
|
||||
@@ -4540,22 +4573,6 @@ msgstr "Signaler ce commentaire"
|
||||
msgid "Edit UE"
|
||||
msgstr "Éditer l'UE"
|
||||
|
||||
#: pedagogy/templates/pedagogy/uv_edit.jinja
|
||||
msgid "Import from UTBM"
|
||||
msgstr "Importer depuis l'UTBM"
|
||||
|
||||
#: pedagogy/templates/pedagogy/uv_edit.jinja
|
||||
msgid "Unknown UE code"
|
||||
msgstr "Code d'UE inconnu"
|
||||
|
||||
#: pedagogy/templates/pedagogy/uv_edit.jinja
|
||||
msgid "Successful autocomplete"
|
||||
msgstr "Autocomplétion réussite"
|
||||
|
||||
#: pedagogy/templates/pedagogy/uv_edit.jinja
|
||||
msgid "An error occurred: "
|
||||
msgstr "Une erreur est survenue : "
|
||||
|
||||
#: rootplace/forms.py
|
||||
msgid "User that will be kept"
|
||||
msgstr "Utilisateur qui sera conservé"
|
||||
@@ -4819,8 +4836,8 @@ msgid "N/A"
|
||||
msgstr "N/A"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "Transfert"
|
||||
msgstr "Virement"
|
||||
msgid "AE account"
|
||||
msgstr "Compte AE"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "Belfort"
|
||||
@@ -5108,26 +5125,6 @@ msgstr "Vous avez acheté %s"
|
||||
msgid "You have a notification"
|
||||
msgstr "Vous avez une notification"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "Success!"
|
||||
msgstr "Succès !"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "Fail!"
|
||||
msgstr "Échec !"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "You successfully posted an article in the Weekmail"
|
||||
msgstr "Article posté avec succès dans le Weekmail"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "You successfully edited an article in the Weekmail"
|
||||
msgstr "Article édité avec succès dans le Weekmail"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "You successfully sent the Weekmail"
|
||||
msgstr "Weekmail envoyé avec succès"
|
||||
|
||||
#: sith/settings.py
|
||||
msgid "AE tee-shirt"
|
||||
msgstr "Tee-shirt AE"
|
||||
@@ -5168,6 +5165,14 @@ msgstr "lieu"
|
||||
msgid "You can not subscribe many time for the same period"
|
||||
msgstr "Vous ne pouvez pas cotiser plusieurs fois pour la même période"
|
||||
|
||||
#: subscription/templates/subscription/forms/create_existing_user.jinja
|
||||
msgid ""
|
||||
"If the subscription is done using the AE account, you must also click it on "
|
||||
"the AE counter."
|
||||
msgstr ""
|
||||
"Si la cotisation est faite en utilisant le compte AE, vous devez également "
|
||||
"la cliquer sur le comptoir AE."
|
||||
|
||||
#: subscription/templates/subscription/fragments/creation_success.jinja
|
||||
#, python-format
|
||||
msgid "Subscription created for %(user)s"
|
||||
@@ -5431,10 +5436,38 @@ msgstr "Mes photos"
|
||||
msgid "Admin tools"
|
||||
msgstr "Admin Trombi"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Trombi modified"
|
||||
msgstr "Trombi modifié"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "User added to the trombi"
|
||||
msgstr "Utilisateur ajouté au trombi"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "User couldn't be added to the trombi"
|
||||
msgstr "L'utilisateur n'a pas pu être ajouté au trombi"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "User removed from the trombi"
|
||||
msgstr "Utilisateur retiré du trombi"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Explain why you rejected the comment"
|
||||
msgstr "Expliquez pourquoi vous refusez le commentaire"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Comment accepted"
|
||||
msgstr "Commentaire accepté"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Comment rejected"
|
||||
msgstr "Commentaire rejeté"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Comment removed"
|
||||
msgstr "Commentaire retiré"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Rejected comment"
|
||||
msgstr "Commentaire rejeté"
|
||||
@@ -5475,6 +5508,10 @@ msgstr ""
|
||||
"pouvez vous inscrire qu'à un seul Trombi, donc ne jouez pas avec cet option "
|
||||
"ou vous encourerez la colère des admins!"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "User modified"
|
||||
msgstr "Utilisateur modifié"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Personal email (not UTBM)"
|
||||
msgstr "Email personnel (pas UTBM)"
|
||||
@@ -5487,6 +5524,14 @@ msgstr "Téléphone"
|
||||
msgid "Native town"
|
||||
msgstr "Ville d'origine"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "User removed from trombi"
|
||||
msgstr "Utilisateur retiré du trombi"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid "Comment added"
|
||||
msgstr "Commentaire ajouté"
|
||||
|
||||
#: trombi/views.py
|
||||
msgid ""
|
||||
"You can not yet write comment, you must wait for the subscription deadline "
|
||||
|
25
package-lock.json
generated
25
package-lock.json
generated
@@ -30,7 +30,6 @@
|
||||
"easymde": "^2.19.0",
|
||||
"glob": "^11.0.0",
|
||||
"htmx.org": "^2.0.3",
|
||||
"jquery": "^3.7.1",
|
||||
"js-cookie": "^3.0.5",
|
||||
"lit-html": "^3.3.0",
|
||||
"native-file-system-adapter": "^3.0.1",
|
||||
@@ -47,7 +46,6 @@
|
||||
"@types/alpinejs": "^3.13.10",
|
||||
"@types/cytoscape-cxtmenu": "^3.4.4",
|
||||
"@types/cytoscape-klay": "^3.1.4",
|
||||
"@types/jquery": "^3.5.31",
|
||||
"@types/js-cookie": "^3.0.6",
|
||||
"typescript": "^5.8.3",
|
||||
"vite": "^6.3.6",
|
||||
@@ -2889,16 +2887,6 @@
|
||||
"integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
|
||||
"license": "MIT"
|
||||
},
|
||||
"node_modules/@types/jquery": {
|
||||
"version": "3.5.33",
|
||||
"resolved": "https://registry.npmjs.org/@types/jquery/-/jquery-3.5.33.tgz",
|
||||
"integrity": "sha512-SeyVJXlCZpEki5F0ghuYe+L+PprQta6nRZqhONt9F13dWBtR/ftoaIbdRQ7cis7womE+X2LKhsDdDtkkDhJS6g==",
|
||||
"dev": true,
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"@types/sizzle": "*"
|
||||
}
|
||||
},
|
||||
"node_modules/@types/js-cookie": {
|
||||
"version": "3.0.6",
|
||||
"resolved": "https://registry.npmjs.org/@types/js-cookie/-/js-cookie-3.0.6.tgz",
|
||||
@@ -2919,13 +2907,6 @@
|
||||
"integrity": "sha512-a79Yc3TOk6dGdituy8hmTTJXjOkZ7zsFYV10L337ttq/rec8lRMDBpV7fL3uLx6TgbFCa5DU/h8FmIBQPSbU0w==",
|
||||
"license": "MIT"
|
||||
},
|
||||
"node_modules/@types/sizzle": {
|
||||
"version": "2.3.10",
|
||||
"resolved": "https://registry.npmjs.org/@types/sizzle/-/sizzle-2.3.10.tgz",
|
||||
"integrity": "sha512-TC0dmN0K8YcWEAEfiPi5gJP14eJe30TTGjkvek3iM/1NdHHsdCA/Td6GvNndMOo/iSnIsZ4HuuhrYPDAmbxzww==",
|
||||
"dev": true,
|
||||
"license": "MIT"
|
||||
},
|
||||
"node_modules/@types/tern": {
|
||||
"version": "0.23.9",
|
||||
"resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.9.tgz",
|
||||
@@ -4384,12 +4365,6 @@
|
||||
"jiti": "lib/jiti-cli.mjs"
|
||||
}
|
||||
},
|
||||
"node_modules/jquery": {
|
||||
"version": "3.7.1",
|
||||
"resolved": "https://registry.npmjs.org/jquery/-/jquery-3.7.1.tgz",
|
||||
"integrity": "sha512-m4avr8yL8kmFN8psrbFFFmB/If14iN5o9nw/NgnnM+kybDJpRsAynV2BsfpTYrTRysYUdADVD7CkUUizgkpLfg==",
|
||||
"license": "MIT"
|
||||
},
|
||||
"node_modules/js-cookie": {
|
||||
"version": "3.0.5",
|
||||
"resolved": "https://registry.npmjs.org/js-cookie/-/js-cookie-3.0.5.tgz",
|
||||
|
@@ -32,7 +32,6 @@
|
||||
"@types/alpinejs": "^3.13.10",
|
||||
"@types/cytoscape-cxtmenu": "^3.4.4",
|
||||
"@types/cytoscape-klay": "^3.1.4",
|
||||
"@types/jquery": "^3.5.31",
|
||||
"@types/js-cookie": "^3.0.6",
|
||||
"typescript": "^5.8.3",
|
||||
"vite": "^6.3.6",
|
||||
@@ -61,7 +60,6 @@
|
||||
"easymde": "^2.19.0",
|
||||
"glob": "^11.0.0",
|
||||
"htmx.org": "^2.0.3",
|
||||
"jquery": "^3.7.1",
|
||||
"js-cookie": "^3.0.5",
|
||||
"lit-html": "^3.3.0",
|
||||
"native-file-system-adapter": "^3.0.1",
|
||||
|
@@ -13,8 +13,7 @@
|
||||
{% block content %}
|
||||
<div class="pedagogy">
|
||||
<div id="uv_detail">
|
||||
<p id="return_noscript"><a href="{{ url('pedagogy:guide') }}">{% trans %}Back{% endtrans %}</a></p>
|
||||
<button id="return_js" onclick='(function(){
|
||||
<button onclick='(function(){
|
||||
// If comes from the guide page, go back with history
|
||||
if (document.referrer.replace(/\?(.+)/gm,"").endsWith(`{{ url("pedagogy:guide") }}`)){
|
||||
window.history.back();
|
||||
@@ -217,9 +216,4 @@
|
||||
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script type="text/javascript">
|
||||
$("#return_noscript").hide();
|
||||
$("#return_js").show();
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
@@ -21,11 +21,6 @@
|
||||
{{ field.errors }}
|
||||
<label for="{{ field.name }}">{{ field.label }}</label>
|
||||
{{ field }}
|
||||
|
||||
|
||||
{% if field.name == 'code' %}
|
||||
<button type="button" id="autofill">{% trans %}Import from UTBM{% endtrans %}</button>
|
||||
{% endif %}
|
||||
</p>
|
||||
{% endif %}
|
||||
|
||||
@@ -36,48 +31,3 @@
|
||||
<p><input type="submit" value="{% trans %}Update{% endtrans %}" /></p>
|
||||
</form>
|
||||
{% endblock %}
|
||||
|
||||
{% block script %}
|
||||
{{ super() }}
|
||||
|
||||
<script type="text/javascript">
|
||||
document.addEventListener('DOMContentLoaded', function() {
|
||||
const autofillBtn = document.getElementById('autofill')
|
||||
const codeInput = document.querySelector('input[name="code"]')
|
||||
|
||||
autofillBtn.addEventListener('click', () => {
|
||||
const url = `/api/uv/${codeInput.value}`;
|
||||
deleteQuickNotifs()
|
||||
|
||||
$.ajax({
|
||||
dataType: "json",
|
||||
url: url,
|
||||
success: function(data, _, xhr) {
|
||||
if (xhr.status !== 200) {
|
||||
createQuickNotif("{% trans %}Unknown UE code{% endtrans %}")
|
||||
return
|
||||
}
|
||||
Object.entries(data)
|
||||
.filter(([_, val]) => !!val) // skip entries with null or undefined value
|
||||
.map(([key, val]) => { // convert keys to DOM elements
|
||||
return [document.querySelector('[name="' + key + '"]'), val];
|
||||
})
|
||||
.filter(([elem, _]) => !!elem) // skip non-existing DOM elements
|
||||
.forEach(([elem, val]) => { // write the value in the form field
|
||||
if (elem.tagName === 'TEXTAREA') {
|
||||
// MD editor text input
|
||||
elem.parentNode.querySelector('.CodeMirror').CodeMirror.setValue(val);
|
||||
} else {
|
||||
elem.value = val;
|
||||
}
|
||||
});
|
||||
createQuickNotif('{% trans %}Successful autocomplete{% endtrans %}')
|
||||
},
|
||||
error: function(_, _, statusMessage) {
|
||||
createQuickNotif('{% trans %}An error occurred: {% endtrans %}' + statusMessage)
|
||||
},
|
||||
})
|
||||
})
|
||||
})
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
@@ -25,7 +25,7 @@ dependencies = [
|
||||
"Pillow<12.0.0,>=11.1.0",
|
||||
"mistune<4.0.0,>=3.1.3",
|
||||
"django-jinja<3.0.0,>=2.11.0",
|
||||
"cryptography>=45.0.3,<47.0.0",
|
||||
"cryptography>=45.0.3,<46.0.0",
|
||||
"django-phonenumber-field<9.0.0,>=8.1.0",
|
||||
"phonenumbers>=9.0.2,<10.0.0",
|
||||
"reportlab<5.0.0,>=4.3.1",
|
||||
|
@@ -309,6 +309,7 @@ exportToHtml("loadViewer", (config: ViewerConfig) => {
|
||||
// Clear selection and cache of retrieved user so they can be filtered again
|
||||
widget.clear(false);
|
||||
widget.clearOptions();
|
||||
widget.setTextboxValue("");
|
||||
},
|
||||
|
||||
/**
|
||||
|
@@ -421,18 +421,11 @@ SITH_PROFILE_DEPARTMENTS = [
|
||||
("NA", _("N/A")),
|
||||
]
|
||||
|
||||
SITH_ACCOUNTING_PAYMENT_METHOD = [
|
||||
("CHECK", _("Check")),
|
||||
("CASH", _("Cash")),
|
||||
("TRANSFERT", _("Transfert")),
|
||||
("CARD", _("Credit card")),
|
||||
]
|
||||
|
||||
SITH_SUBSCRIPTION_PAYMENT_METHOD = [
|
||||
("CHECK", _("Check")),
|
||||
("CARD", _("Credit card")),
|
||||
("CASH", _("Cash")),
|
||||
("EBOUTIC", _("Eboutic")),
|
||||
("AE_ACCOUNT", _("AE account")),
|
||||
("OTHER", _("Other")),
|
||||
]
|
||||
|
||||
@@ -441,6 +434,7 @@ SITH_SUBSCRIPTION_LOCATIONS = [
|
||||
("SEVENANS", _("Sevenans")),
|
||||
("MONTBELIARD", _("Montbéliard")),
|
||||
("EBOUTIC", _("Eboutic")),
|
||||
("OTHER", _("Other")),
|
||||
]
|
||||
|
||||
SITH_COUNTER_BARS = [(1, "MDE"), (2, "Foyer"), (35, "La Gommette")]
|
||||
@@ -691,14 +685,6 @@ SITH_PERMANENT_NOTIFICATIONS = {
|
||||
"SAS_MODERATION": "sas.models.sas_notification_callback",
|
||||
}
|
||||
|
||||
SITH_QUICK_NOTIF = {
|
||||
"qn_success": _("Success!"),
|
||||
"qn_fail": _("Fail!"),
|
||||
"qn_weekmail_new_article": _("You successfully posted an article in the Weekmail"),
|
||||
"qn_weekmail_article_edit": _("You successfully edited an article in the Weekmail"),
|
||||
"qn_weekmail_send_success": _("You successfully sent the Weekmail"),
|
||||
}
|
||||
|
||||
# Mailing related settings
|
||||
|
||||
SITH_MAILING_DOMAIN = "utbm.fr"
|
||||
|
@@ -2,6 +2,7 @@ import secrets
|
||||
from typing import Any
|
||||
|
||||
from django import forms
|
||||
from django.conf import settings
|
||||
from django.core.exceptions import ValidationError
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
|
||||
@@ -23,6 +24,13 @@ class SelectionDateForm(forms.Form):
|
||||
|
||||
|
||||
class SubscriptionForm(forms.ModelForm):
|
||||
allowed_payment_methods = ["CARD", "CASH", "AE_ACCOUNT"]
|
||||
|
||||
class Meta:
|
||||
model = Subscription
|
||||
fields = ["subscription_type", "payment_method", "location"]
|
||||
widgets = {"payment_method": forms.RadioSelect}
|
||||
|
||||
def __init__(self, *args, initial=None, **kwargs):
|
||||
initial = initial or {}
|
||||
if "subscription_type" not in initial:
|
||||
@@ -30,6 +38,14 @@ class SubscriptionForm(forms.ModelForm):
|
||||
if "payment_method" not in initial:
|
||||
initial["payment_method"] = "CARD"
|
||||
super().__init__(*args, initial=initial, **kwargs)
|
||||
self.fields["payment_method"].choices = [
|
||||
m
|
||||
for m in settings.SITH_SUBSCRIPTION_PAYMENT_METHOD
|
||||
if m[0] in self.allowed_payment_methods
|
||||
]
|
||||
self.fields["location"].choices = [
|
||||
m for m in settings.SITH_SUBSCRIPTION_LOCATIONS if m[0] != "EBOUTIC"
|
||||
]
|
||||
|
||||
def save(self, *args, **kwargs):
|
||||
if self.errors:
|
||||
@@ -61,7 +77,8 @@ class SubscriptionNewUserForm(SubscriptionForm):
|
||||
assert user.is_subscribed
|
||||
"""
|
||||
|
||||
template_name = "subscription/forms/create_new_user.html"
|
||||
allowed_payment_methods = ["CARD", "CASH"]
|
||||
template_name = "subscription/forms/create_new_user.jinja"
|
||||
|
||||
__user_fields = forms.fields_for_model(
|
||||
User,
|
||||
@@ -73,10 +90,6 @@ class SubscriptionNewUserForm(SubscriptionForm):
|
||||
email = __user_fields["email"]
|
||||
date_of_birth = __user_fields["date_of_birth"]
|
||||
|
||||
class Meta:
|
||||
model = Subscription
|
||||
fields = ["subscription_type", "payment_method", "location"]
|
||||
|
||||
field_order = [
|
||||
"first_name",
|
||||
"last_name",
|
||||
@@ -130,7 +143,7 @@ class SubscriptionNewUserForm(SubscriptionForm):
|
||||
class SubscriptionExistingUserForm(SubscriptionForm):
|
||||
"""Form to add a subscription to an existing user."""
|
||||
|
||||
template_name = "subscription/forms/create_existing_user.html"
|
||||
template_name = "subscription/forms/create_existing_user.jinja"
|
||||
required_css_class = "required"
|
||||
|
||||
birthdate = forms.fields_for_model(
|
||||
@@ -140,10 +153,9 @@ class SubscriptionExistingUserForm(SubscriptionForm):
|
||||
help_texts={"date_of_birth": _("This user didn't fill its birthdate yet.")},
|
||||
)["date_of_birth"]
|
||||
|
||||
class Meta:
|
||||
model = Subscription
|
||||
fields = ["member", "subscription_type", "payment_method", "location"]
|
||||
widgets = {"member": AutoCompleteSelectUser}
|
||||
class Meta(SubscriptionForm.Meta):
|
||||
fields = ["member", *SubscriptionForm.Meta.fields]
|
||||
widgets = SubscriptionForm.Meta.widgets | {"member": AutoCompleteSelectUser}
|
||||
|
||||
field_order = [
|
||||
"member",
|
||||
|
@@ -0,0 +1,56 @@
|
||||
# Generated by Django 5.2.3 on 2025-09-08 05:38
|
||||
|
||||
from django.db import migrations, models
|
||||
from django.db.migrations.state import StateApps
|
||||
|
||||
|
||||
def rename_enums(apps: StateApps, schema_editor):
|
||||
Subscription = apps.get_model("subscription", "Subscription")
|
||||
Subscription.objects.filter(subscription_type="EBOUTIC").update(
|
||||
subscription_type="AE_ACCOUNT"
|
||||
)
|
||||
|
||||
|
||||
def rename_enums_reverse(apps: StateApps, schema_editor):
|
||||
Subscription = apps.get_model("subscription", "Subscription")
|
||||
Subscription.objects.filter(subscription_type="AE_ACCOUNT").update(
|
||||
subscription_type="EBOUTIC"
|
||||
)
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
dependencies = [("subscription", "0014_auto_20201207_2323")]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name="subscription",
|
||||
name="location",
|
||||
field=models.CharField(
|
||||
choices=[
|
||||
("BELFORT", "Belfort"),
|
||||
("SEVENANS", "Sevenans"),
|
||||
("MONTBELIARD", "Montbéliard"),
|
||||
("EBOUTIC", "Eboutic"),
|
||||
("OTHER", "Other"),
|
||||
],
|
||||
max_length=20,
|
||||
verbose_name="location",
|
||||
),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name="subscription",
|
||||
name="payment_method",
|
||||
field=models.CharField(
|
||||
choices=[
|
||||
("CHECK", "Check"),
|
||||
("CARD", "Credit card"),
|
||||
("CASH", "Cash"),
|
||||
("AE_ACCOUNT", "AE account"),
|
||||
("OTHER", "Other"),
|
||||
],
|
||||
max_length=255,
|
||||
verbose_name="payment method",
|
||||
),
|
||||
),
|
||||
migrations.RunPython(rename_enums, reverse_code=rename_enums_reverse),
|
||||
]
|
@@ -1,14 +0,0 @@
|
||||
{% load static %}
|
||||
{% load i18n %}
|
||||
|
||||
|
||||
<div x-data="existing_user_subscription_form" class="form-content existing-user">
|
||||
<fieldset>
|
||||
{{ form.as_p }}
|
||||
</fieldset>
|
||||
<div
|
||||
id="subscription-form-user-mini-profile"
|
||||
x-html="profileFragment"
|
||||
:aria-busy="loading"
|
||||
></div>
|
||||
</div>
|
@@ -0,0 +1,28 @@
|
||||
{% load static %}
|
||||
{% load i18n %}
|
||||
|
||||
|
||||
<div x-data="existing_user_subscription_form" class="form-content existing-user">
|
||||
<fieldset>
|
||||
{{ errors }}
|
||||
{% for field, errors in fields %}
|
||||
<p{% with classes=field.css_classes %}{% if classes %} class="{{ classes }}"{% endif %}{% endwith %}>
|
||||
{{ field.label_tag }}
|
||||
{{ field }}
|
||||
{% if field.help_text %}
|
||||
<span class="helptext">{{ field.help_text }}</span>
|
||||
{% endif %}
|
||||
</p>
|
||||
{% if field.name == "payment_method" %}
|
||||
<i>
|
||||
{% blocktranslate %}If the subscription is done using the AE account, you must also click it on the AE counter.{% endblocktranslate %}
|
||||
</i>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</fieldset>
|
||||
<div
|
||||
id="subscription-form-user-mini-profile"
|
||||
x-html="profileFragment"
|
||||
:aria-busy="loading"
|
||||
></div>
|
||||
</div>
|
@@ -90,7 +90,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=self.user,
|
||||
subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2017, 8, 29)
|
||||
s.subscription_end = s.compute_end(duration=0.166, start=s.subscription_start)
|
||||
@@ -101,7 +101,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=self.user,
|
||||
subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2017, 8, 29)
|
||||
s.subscription_end = s.compute_end(duration=0.333, start=s.subscription_start)
|
||||
@@ -112,7 +112,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=self.user,
|
||||
subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2017, 8, 29)
|
||||
s.subscription_end = s.compute_end(
|
||||
@@ -126,7 +126,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=self.user,
|
||||
subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2017, 8, 29)
|
||||
s.subscription_end = s.compute_end(duration=0.5, start=s.subscription_start)
|
||||
@@ -137,7 +137,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=self.user,
|
||||
subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2017, 8, 29)
|
||||
s.subscription_end = s.compute_end(duration=0.67, start=s.subscription_start)
|
||||
@@ -148,7 +148,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=self.user,
|
||||
subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2018, 9, 1)
|
||||
s.subscription_end = s.compute_end(duration=0.23, start=s.subscription_start)
|
||||
@@ -160,7 +160,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=user,
|
||||
subscription_type="deux-semestres",
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2015, 8, 29)
|
||||
s.subscription_end = s.compute_end(
|
||||
@@ -181,7 +181,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=user,
|
||||
subscription_type="deux-mois-essai",
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2015, 8, 29)
|
||||
s.subscription_end = s.compute_end(
|
||||
@@ -202,7 +202,7 @@ class TestSubscriptionIntegration(TestCase):
|
||||
s = Subscription(
|
||||
member=user,
|
||||
subscription_type="deux-mois-essai",
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0],
|
||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
||||
)
|
||||
s.subscription_start = date(2015, 8, 29)
|
||||
s.subscription_end = s.compute_end(
|
||||
|
@@ -38,7 +38,7 @@ def test_form_existing_user_valid(
|
||||
"birthdate": user.date_of_birth,
|
||||
"subscription_type": "deux-semestres",
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
}
|
||||
form = SubscriptionExistingUserForm(data)
|
||||
assert form.is_valid()
|
||||
@@ -55,7 +55,7 @@ def test_form_existing_user_with_birthdate(settings: SettingsWrapper):
|
||||
"member": user,
|
||||
"subscription_type": "deux-semestres",
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
}
|
||||
form = SubscriptionExistingUserForm(data)
|
||||
assert not form.is_valid()
|
||||
@@ -81,7 +81,7 @@ def test_form_existing_user_invalid(settings: SettingsWrapper):
|
||||
"member": user,
|
||||
"subscription_type": "deux-semestres",
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
}
|
||||
form = SubscriptionExistingUserForm(data)
|
||||
|
||||
@@ -99,7 +99,7 @@ def test_form_new_user(settings: SettingsWrapper):
|
||||
"date_of_birth": localdate() - relativedelta(years=18),
|
||||
"subscription_type": "deux-semestres",
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
}
|
||||
form = SubscriptionNewUserForm(data)
|
||||
assert form.is_valid()
|
||||
@@ -130,7 +130,7 @@ def test_form_set_new_user_as_student(settings: SettingsWrapper, subscription_ty
|
||||
"date_of_birth": localdate() - relativedelta(years=18),
|
||||
"subscription_type": subscription_type,
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
}
|
||||
form = SubscriptionNewUserForm(data)
|
||||
assert form.is_valid()
|
||||
@@ -180,7 +180,7 @@ def test_submit_form_existing_user(client: Client, settings: SettingsWrapper):
|
||||
"birthdate": user.date_of_birth,
|
||||
"subscription_type": "deux-semestres",
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
},
|
||||
)
|
||||
user.refresh_from_db()
|
||||
@@ -212,7 +212,7 @@ def test_submit_form_new_user(client: Client, settings: SettingsWrapper):
|
||||
"date_of_birth": localdate() - relativedelta(years=18),
|
||||
"subscription_type": "deux-semestres",
|
||||
"location": settings.SITH_SUBSCRIPTION_LOCATIONS[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0],
|
||||
"payment_method": settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0],
|
||||
},
|
||||
)
|
||||
user = User.objects.get(email="jdoe@utbm.fr")
|
||||
|
@@ -26,7 +26,9 @@ from datetime import date
|
||||
|
||||
from django import forms
|
||||
from django.conf import settings
|
||||
from django.contrib import messages
|
||||
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||
from django.contrib.messages.views import SuccessMessageMixin
|
||||
from django.core.exceptions import PermissionDenied
|
||||
from django.db import IntegrityError
|
||||
from django.forms.models import modelform_factory
|
||||
@@ -46,7 +48,7 @@ from core.auth.mixins import (
|
||||
)
|
||||
from core.models import User
|
||||
from core.views.forms import SelectDate
|
||||
from core.views.mixins import QuickNotifMixin, TabedViewMixin
|
||||
from core.views.mixins import TabedViewMixin
|
||||
from core.views.widgets.ajax_select import AutoCompleteSelectUser
|
||||
from trombi.models import Trombi, TrombiClubMembership, TrombiComment, TrombiUser
|
||||
|
||||
@@ -134,15 +136,15 @@ class TrombiCreateView(CanCreateMixin, CreateView):
|
||||
return self.form_invalid(form)
|
||||
|
||||
|
||||
class TrombiEditView(CanEditPropMixin, TrombiTabsMixin, UpdateView):
|
||||
class TrombiEditView(
|
||||
CanEditPropMixin, TrombiTabsMixin, SuccessMessageMixin, UpdateView
|
||||
):
|
||||
model = Trombi
|
||||
form_class = TrombiForm
|
||||
template_name = "core/edit.jinja"
|
||||
pk_url_kwarg = "trombi_id"
|
||||
current_tab = "admin_tools"
|
||||
|
||||
def get_success_url(self):
|
||||
return super().get_success_url() + "?qn_success"
|
||||
success_message = _("Trombi modified")
|
||||
|
||||
|
||||
class AddUserForm(forms.Form):
|
||||
@@ -155,7 +157,7 @@ class AddUserForm(forms.Form):
|
||||
)
|
||||
|
||||
|
||||
class TrombiDetailView(CanEditMixin, QuickNotifMixin, TrombiTabsMixin, DetailView):
|
||||
class TrombiDetailView(CanEditMixin, TrombiTabsMixin, DetailView):
|
||||
model = Trombi
|
||||
template_name = "trombi/detail.jinja"
|
||||
pk_url_kwarg = "trombi_id"
|
||||
@@ -167,9 +169,9 @@ class TrombiDetailView(CanEditMixin, QuickNotifMixin, TrombiTabsMixin, DetailVie
|
||||
if form.is_valid():
|
||||
try:
|
||||
TrombiUser(user=form.cleaned_data["user"], trombi=self.object).save()
|
||||
self.quick_notif_list.append("qn_success")
|
||||
messages.success(self.request, _("User added to the trombi"))
|
||||
except IntegrityError: # We don't care about duplicate keys
|
||||
self.quick_notif_list.append("qn_fail")
|
||||
messages.error(self.request, _("User couldn't be added to the trombi"))
|
||||
return super().get(request, *args, **kwargs)
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
@@ -185,22 +187,20 @@ class TrombiExportView(CanEditMixin, TrombiTabsMixin, DetailView):
|
||||
current_tab = "admin_tools"
|
||||
|
||||
|
||||
class TrombiDeleteUserView(CanEditPropMixin, TrombiTabsMixin, DeleteView):
|
||||
class TrombiDeleteUserView(
|
||||
CanEditPropMixin, TrombiTabsMixin, SuccessMessageMixin, DeleteView
|
||||
):
|
||||
model = TrombiUser
|
||||
pk_url_kwarg = "user_id"
|
||||
template_name = "core/delete_confirm.jinja"
|
||||
current_tab = "admin_tools"
|
||||
success_message = _("User removed from the trombi")
|
||||
|
||||
def get_success_url(self):
|
||||
return (
|
||||
reverse("trombi:detail", kwargs={"trombi_id": self.object.trombi.id})
|
||||
+ "?qn_success"
|
||||
)
|
||||
return reverse("trombi:detail", kwargs={"trombi_id": self.object.trombi.id})
|
||||
|
||||
|
||||
class TrombiModerateCommentsView(
|
||||
CanEditPropMixin, QuickNotifMixin, TrombiTabsMixin, DetailView
|
||||
):
|
||||
class TrombiModerateCommentsView(CanEditPropMixin, TrombiTabsMixin, DetailView):
|
||||
model = Trombi
|
||||
template_name = "trombi/comment_moderation.jinja"
|
||||
pk_url_kwarg = "trombi_id"
|
||||
@@ -235,16 +235,18 @@ class TrombiModerateCommentView(DetailView):
|
||||
if request.POST["action"] == "accept":
|
||||
self.object.is_moderated = True
|
||||
self.object.save()
|
||||
messages.success(self.request, _("Comment accepted"))
|
||||
return redirect(
|
||||
reverse(
|
||||
"trombi:moderate_comments",
|
||||
kwargs={"trombi_id": self.object.author.trombi.id},
|
||||
)
|
||||
+ "?qn_success"
|
||||
)
|
||||
elif request.POST["action"] == "reject":
|
||||
messages.success(self.request, _("Comment rejected"))
|
||||
return super().get(request, *args, **kwargs)
|
||||
elif request.POST["action"] == "delete" and "reason" in request.POST:
|
||||
messages.success(self.request, _("Comment removed"))
|
||||
self.object.author.user.email_user(
|
||||
subject="[%s] %s" % (settings.SITH_NAME, _("Rejected comment")),
|
||||
message=_(
|
||||
@@ -265,7 +267,6 @@ class TrombiModerateCommentView(DetailView):
|
||||
"trombi:moderate_comments",
|
||||
kwargs={"trombi_id": self.object.author.trombi.id},
|
||||
)
|
||||
+ "?qn_success"
|
||||
)
|
||||
raise Http404
|
||||
|
||||
@@ -299,9 +300,7 @@ class UserTrombiForm(forms.Form):
|
||||
)
|
||||
|
||||
|
||||
class UserTrombiToolsView(
|
||||
LoginRequiredMixin, QuickNotifMixin, TrombiTabsMixin, TemplateView
|
||||
):
|
||||
class UserTrombiToolsView(LoginRequiredMixin, TrombiTabsMixin, TemplateView):
|
||||
"""Display a user's trombi tools."""
|
||||
|
||||
template_name = "trombi/user_tools.jinja"
|
||||
@@ -318,7 +317,6 @@ class UserTrombiToolsView(
|
||||
user=request.user, trombi=self.form.cleaned_data["trombi"]
|
||||
)
|
||||
trombi_user.save()
|
||||
self.quick_notif_list += ["qn_success"]
|
||||
return super().get(request, *args, **kwargs)
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
@@ -335,21 +333,24 @@ class UserTrombiToolsView(
|
||||
return kwargs
|
||||
|
||||
|
||||
class UserTrombiEditPicturesView(TrombiTabsMixin, UserIsInATrombiMixin, UpdateView):
|
||||
class UserTrombiEditPicturesView(
|
||||
TrombiTabsMixin, UserIsInATrombiMixin, SuccessMessageMixin, UpdateView
|
||||
):
|
||||
model = TrombiUser
|
||||
fields = ["profile_pict", "scrub_pict"]
|
||||
template_name = "core/edit.jinja"
|
||||
current_tab = "pictures"
|
||||
success_message = _("User modified")
|
||||
|
||||
def get_object(self):
|
||||
return self.request.user.trombi_user
|
||||
|
||||
def get_success_url(self):
|
||||
return reverse("trombi:user_tools") + "?qn_success"
|
||||
return reverse("trombi:user_tools")
|
||||
|
||||
|
||||
class UserTrombiEditProfileView(
|
||||
QuickNotifMixin, TrombiTabsMixin, UserIsInATrombiMixin, UpdateView
|
||||
TrombiTabsMixin, UserIsInATrombiMixin, SuccessMessageMixin, UpdateView
|
||||
):
|
||||
model = User
|
||||
form_class = modelform_factory(
|
||||
@@ -370,16 +371,20 @@ class UserTrombiEditProfileView(
|
||||
)
|
||||
template_name = "trombi/edit_profile.jinja"
|
||||
current_tab = "profile"
|
||||
success_message = _("User modified")
|
||||
|
||||
def get_object(self):
|
||||
return self.request.user
|
||||
|
||||
def get_success_url(self):
|
||||
return reverse("trombi:user_tools") + "?qn_success"
|
||||
return reverse("trombi:user_tools")
|
||||
|
||||
|
||||
class UserTrombiResetClubMembershipsView(UserIsInATrombiMixin, RedirectView):
|
||||
class UserTrombiResetClubMembershipsView(
|
||||
UserIsInATrombiMixin, SuccessMessageMixin, RedirectView
|
||||
):
|
||||
permanent = False
|
||||
success_message = _("User modified")
|
||||
|
||||
def get(self, request, *args, **kwargs):
|
||||
user = self.request.user.trombi_user
|
||||
@@ -387,18 +392,18 @@ class UserTrombiResetClubMembershipsView(UserIsInATrombiMixin, RedirectView):
|
||||
return redirect(self.get_success_url())
|
||||
|
||||
def get_success_url(self):
|
||||
return reverse("trombi:profile") + "?qn_success"
|
||||
return reverse("trombi:profile")
|
||||
|
||||
|
||||
class UserTrombiDeleteMembershipView(TrombiTabsMixin, CanEditMixin, DeleteView):
|
||||
class UserTrombiDeleteMembershipView(
|
||||
TrombiTabsMixin, CanEditMixin, SuccessMessageMixin, DeleteView
|
||||
):
|
||||
model = TrombiClubMembership
|
||||
pk_url_kwarg = "membership_id"
|
||||
template_name = "core/delete_confirm.jinja"
|
||||
success_url = reverse_lazy("trombi:profile")
|
||||
current_tab = "profile"
|
||||
|
||||
def get_success_url(self):
|
||||
return super().get_success_url() + "?qn_success"
|
||||
success_message = _("User removed from trombi")
|
||||
|
||||
|
||||
# Used by admins when someone does not have every club in his list
|
||||
@@ -428,15 +433,18 @@ class UserTrombiAddMembershipView(TrombiTabsMixin, CreateView):
|
||||
)
|
||||
|
||||
|
||||
class UserTrombiEditMembershipView(CanEditMixin, TrombiTabsMixin, UpdateView):
|
||||
class UserTrombiEditMembershipView(
|
||||
CanEditMixin, TrombiTabsMixin, SuccessMessageMixin, UpdateView
|
||||
):
|
||||
model = TrombiClubMembership
|
||||
pk_url_kwarg = "membership_id"
|
||||
fields = ["role", "start", "end"]
|
||||
template_name = "core/edit.jinja"
|
||||
current_tab = "profile"
|
||||
success_message = _("User modified")
|
||||
|
||||
def get_success_url(self):
|
||||
return super().get_success_url() + "?qn_success"
|
||||
return super().get_success_url()
|
||||
|
||||
|
||||
class UserTrombiProfileView(TrombiTabsMixin, DetailView):
|
||||
@@ -461,12 +469,13 @@ class UserTrombiProfileView(TrombiTabsMixin, DetailView):
|
||||
return super().get(request, *args, **kwargs)
|
||||
|
||||
|
||||
class TrombiCommentFormView(LoginRequiredMixin, View):
|
||||
class TrombiCommentFormView(LoginRequiredMixin, SuccessMessageMixin, View):
|
||||
"""Create/edit a trombi comment."""
|
||||
|
||||
model = TrombiComment
|
||||
fields = ["content"]
|
||||
template_name = "trombi/comment.jinja"
|
||||
success_message = _("Comment added")
|
||||
|
||||
def get_form_class(self):
|
||||
self.trombi = self.request.user.trombi_user.trombi
|
||||
@@ -496,7 +505,7 @@ class TrombiCommentFormView(LoginRequiredMixin, View):
|
||||
)
|
||||
|
||||
def get_success_url(self):
|
||||
return reverse("trombi:user_tools") + "?qn_success"
|
||||
return reverse("trombi:user_tools")
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
kwargs = super().get_context_data(**kwargs)
|
||||
|
@@ -11,7 +11,7 @@
|
||||
"allowSyntheticDefaultImports": true,
|
||||
"esModuleInterop": true,
|
||||
"resolveJsonModule": true,
|
||||
"types": ["jquery", "alpinejs"],
|
||||
"types": ["alpinejs"],
|
||||
"paths": {
|
||||
"#openapi": ["./staticfiles/generated/openapi/client/index.ts"],
|
||||
"#openapi:*": ["./staticfiles/generated/openapi/client/*"],
|
||||
|
@@ -4,7 +4,6 @@ import inject from "@rollup/plugin-inject";
|
||||
import { glob } from "glob";
|
||||
import { type AliasOptions, type UserConfig, defineConfig } from "vite";
|
||||
import type { Rollup } from "vite";
|
||||
import { viteStaticCopy } from "vite-plugin-static-copy";
|
||||
import tsconfig from "./tsconfig.json";
|
||||
|
||||
const outDir = resolve(__dirname, "./staticfiles/generated/bundled");
|
||||
@@ -87,17 +86,6 @@ export default defineConfig((config: UserConfig) => {
|
||||
Alpine: "alpinejs",
|
||||
htmx: "htmx.org",
|
||||
}),
|
||||
viteStaticCopy({
|
||||
targets: [
|
||||
{
|
||||
src: resolve(nodeModules, "jquery/dist/jquery.min.js"),
|
||||
dest: vendored,
|
||||
},
|
||||
],
|
||||
}),
|
||||
],
|
||||
optimizeDeps: {
|
||||
include: ["jquery"],
|
||||
},
|
||||
} satisfies UserConfig;
|
||||
});
|
||||
|
Reference in New Issue
Block a user