mirror of
https://github.com/ae-utbm/sith.git
synced 2025-12-13 19:01:20 +00:00
Compare commits
72 Commits
subscripti
...
dependabot
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3e258f5940 | ||
|
|
c42aea26d7 | ||
|
|
570836190c | ||
|
|
163ef21ace | ||
|
|
a8f7a8865c | ||
|
|
8cd28fda9c | ||
|
|
7665d7efb4 | ||
|
|
722ef67450 | ||
|
|
f027464d0e | ||
|
|
d940e32dac | ||
|
|
a0015eb65f | ||
|
|
812a761690 | ||
|
|
f0b1e8af4a | ||
|
|
5697b4e9c8 | ||
|
|
7f504d9ee2 | ||
|
|
49b0a13dbd | ||
|
|
edd31d5d56 | ||
|
|
3ea2d2aaf2 | ||
|
|
6b27542210 | ||
|
|
e26851beb3 | ||
|
|
285bd71371 | ||
|
|
9c22e061f5 | ||
|
|
4fdc13fb1c | ||
|
|
415193972c | ||
|
|
bf45b95d88 | ||
|
|
9a311d8cee | ||
|
|
7209801511 | ||
|
|
742ac504dc | ||
|
|
3b56d2c22b | ||
|
|
9c64dae7fe | ||
|
917a2b50cc
|
|||
|
|
118a08372f | ||
|
b8429a510f
|
|||
|
|
49a9149a90 | ||
|
|
ed12da222f | ||
|
|
459edc1b6e | ||
| a760a0b75d | |||
|
|
fc615e90b2 | ||
|
76eebaf54e
|
|||
|
|
9407f4b341 | ||
|
|
8bd82c9d7c | ||
|
|
957441ceb1 | ||
|
|
3bcd417ad0 | ||
|
|
453e13d54b | ||
|
|
dbd86b66cc | ||
|
|
dcf799b352 | ||
|
|
d815f7da97 | ||
|
|
dac52db434 | ||
|
|
f398c9901c | ||
|
|
5b91fe2145 | ||
|
|
abd905c24d | ||
|
|
42b53a39f3 | ||
|
|
5306001f6f | ||
|
|
83a4ac2a7e | ||
|
|
30fd4f6926 | ||
|
|
1b1ef18531 | ||
|
|
bcf5d30d8f | ||
|
|
4b44e50780 | ||
|
|
40c3276c3c | ||
|
|
543a424258 | ||
|
|
8ff25e6034 | ||
|
fa8772ede2
|
|||
|
|
2a30f30a31 | ||
|
|
80545e682b | ||
|
|
a7adb4bba3 | ||
|
|
e75e7e697a | ||
|
|
9d99976bee | ||
|
|
4103dce1bb | ||
|
|
126fcbaaa1 | ||
|
|
8a27214801 | ||
|
|
e82f3649e5 | ||
|
|
d3444f6bea |
@@ -1,7 +1,7 @@
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
# Ruff version.
|
# Ruff version.
|
||||||
rev: v0.11.13
|
rev: v0.14.4
|
||||||
hooks:
|
hooks:
|
||||||
- id: ruff-check # just check the code, and print the errors
|
- id: ruff-check # just check the code, and print the errors
|
||||||
- id: ruff-check # actually fix the fixable errors, but print nothing
|
- id: ruff-check # actually fix the fixable errors, but print nothing
|
||||||
@@ -14,7 +14,7 @@ repos:
|
|||||||
- id: biome-check
|
- id: biome-check
|
||||||
additional_dependencies: ["@biomejs/biome@1.9.4"]
|
additional_dependencies: ["@biomejs/biome@1.9.4"]
|
||||||
- repo: https://github.com/rtts/djhtml
|
- repo: https://github.com/rtts/djhtml
|
||||||
rev: 3.0.7
|
rev: 3.0.10
|
||||||
hooks:
|
hooks:
|
||||||
- id: djhtml
|
- id: djhtml
|
||||||
name: format templates
|
name: format templates
|
||||||
|
|||||||
10
club/api.py
10
club/api.py
@@ -1,7 +1,5 @@
|
|||||||
from typing import Annotated
|
|
||||||
|
|
||||||
from annotated_types import MinLen
|
|
||||||
from django.db.models import Prefetch
|
from django.db.models import Prefetch
|
||||||
|
from ninja import Query
|
||||||
from ninja.security import SessionAuth
|
from ninja.security import SessionAuth
|
||||||
from ninja_extra import ControllerBase, api_controller, paginate, route
|
from ninja_extra import ControllerBase, api_controller, paginate, route
|
||||||
from ninja_extra.pagination import PageNumberPaginationExtra
|
from ninja_extra.pagination import PageNumberPaginationExtra
|
||||||
@@ -10,7 +8,7 @@ from ninja_extra.schemas import PaginatedResponseSchema
|
|||||||
from api.auth import ApiKeyAuth
|
from api.auth import ApiKeyAuth
|
||||||
from api.permissions import CanAccessLookup, HasPerm
|
from api.permissions import CanAccessLookup, HasPerm
|
||||||
from club.models import Club, Membership
|
from club.models import Club, Membership
|
||||||
from club.schemas import ClubSchema, SimpleClubSchema
|
from club.schemas import ClubSchema, ClubSearchFilterSchema, SimpleClubSchema
|
||||||
|
|
||||||
|
|
||||||
@api_controller("/club")
|
@api_controller("/club")
|
||||||
@@ -23,8 +21,8 @@ class ClubController(ControllerBase):
|
|||||||
url_name="search_club",
|
url_name="search_club",
|
||||||
)
|
)
|
||||||
@paginate(PageNumberPaginationExtra, page_size=50)
|
@paginate(PageNumberPaginationExtra, page_size=50)
|
||||||
def search_club(self, search: Annotated[str, MinLen(1)]):
|
def search_club(self, filters: Query[ClubSearchFilterSchema]):
|
||||||
return Club.objects.filter(name__icontains=search).values()
|
return filters.filter(Club.objects.all())
|
||||||
|
|
||||||
@route.get(
|
@route.get(
|
||||||
"/{int:club_id}",
|
"/{int:club_id}",
|
||||||
|
|||||||
@@ -252,7 +252,7 @@ class ClubAddMemberForm(ClubMemberForm):
|
|||||||
Board members can attribute roles lower than their own.
|
Board members can attribute roles lower than their own.
|
||||||
Other users cannot attribute roles with this form
|
Other users cannot attribute roles with this form
|
||||||
"""
|
"""
|
||||||
if self.request_user.has_perm("club.add_subscription"):
|
if self.request_user.has_perm("club.add_membership"):
|
||||||
return settings.SITH_CLUB_ROLES_ID["President"]
|
return settings.SITH_CLUB_ROLES_ID["President"]
|
||||||
membership = self.request_user_membership
|
membership = self.request_user_membership
|
||||||
if membership is None or membership.role <= settings.SITH_MAXIMUM_FREE_ROLE:
|
if membership is None or membership.role <= settings.SITH_MAXIMUM_FREE_ROLE:
|
||||||
|
|||||||
@@ -1,9 +1,26 @@
|
|||||||
from ninja import ModelSchema
|
from typing import Annotated
|
||||||
|
|
||||||
|
from annotated_types import MinLen
|
||||||
|
from django.db.models import Q
|
||||||
|
from ninja import Field, FilterSchema, ModelSchema
|
||||||
|
|
||||||
from club.models import Club, Membership
|
from club.models import Club, Membership
|
||||||
from core.schemas import SimpleUserSchema
|
from core.schemas import SimpleUserSchema
|
||||||
|
|
||||||
|
|
||||||
|
class ClubSearchFilterSchema(FilterSchema):
|
||||||
|
search: Annotated[str, MinLen(1)] | None = Field(None, q="name__icontains")
|
||||||
|
is_active: bool | None = None
|
||||||
|
parent_id: int | None = None
|
||||||
|
parent_name: str | None = Field(None, q="parent__name__icontains")
|
||||||
|
exclude_ids: set[int] | None = None
|
||||||
|
|
||||||
|
def filter_exclude_ids(self, value: set[int] | None):
|
||||||
|
if value is None:
|
||||||
|
return Q()
|
||||||
|
return ~Q(id__in=value)
|
||||||
|
|
||||||
|
|
||||||
class SimpleClubSchema(ModelSchema):
|
class SimpleClubSchema(ModelSchema):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Club
|
model = Club
|
||||||
|
|||||||
@@ -6,11 +6,11 @@ because it works with a somewhat dynamic form,
|
|||||||
but was written before Alpine was introduced in the project.
|
but was written before Alpine was introduced in the project.
|
||||||
TODO : rewrite the pagination used in this template an Alpine one
|
TODO : rewrite the pagination used in this template an Alpine one
|
||||||
#}
|
#}
|
||||||
{% macro paginate(page_obj, paginator, js_action) %}
|
{% macro paginate(page_obj, paginator) %}
|
||||||
{% set js = js_action|default('') %}
|
{% set js = "formPagination(this)" %}
|
||||||
{% if page_obj.has_previous() or page_obj.has_next() %}
|
{% if page_obj.has_previous() or page_obj.has_next() %}
|
||||||
{% if page_obj.has_previous() %}
|
{% if page_obj.has_previous() %}
|
||||||
<a {% if js %} type="submit" onclick="{{ js }}" {% endif %} href="?page={{ page_obj.previous_page_number() }}">{% trans %}Previous{% endtrans %}</a>
|
<a type="submit" onclick="{{ js }}" href="?page={{ page_obj.previous_page_number() }}">{% trans %}Previous{% endtrans %}</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<span class="disabled">{% trans %}Previous{% endtrans %}</span>
|
<span class="disabled">{% trans %}Previous{% endtrans %}</span>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@@ -18,11 +18,11 @@ TODO : rewrite the pagination used in this template an Alpine one
|
|||||||
{% if page_obj.number == i %}
|
{% if page_obj.number == i %}
|
||||||
<span class="active">{{ i }} <span class="sr-only">({% trans %}current{% endtrans %})</span></span>
|
<span class="active">{{ i }} <span class="sr-only">({% trans %}current{% endtrans %})</span></span>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a {% if js %} type="submit" onclick="{{ js }}" {% endif %} href="?page={{ i }}">{{ i }}</a>
|
<a type="submit" onclick="{{ js }}" href="?page={{ i }}">{{ i }}</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% if page_obj.has_next() %}
|
{% if page_obj.has_next() %}
|
||||||
<a {% if js %} type="submit" onclick="{{ js }}" {% endif %} href="?page={{ page_obj.next_page_number() }}">{% trans %}Next{% endtrans %}</a>
|
<a type="submit" onclick="{{ js }}" href="?page={{ page_obj.next_page_number() }}">{% trans %}Next{% endtrans %}</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<span class="disabled">{% trans %}Next{% endtrans %}</span>
|
<span class="disabled">{% trans %}Next{% endtrans %}</span>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@@ -81,6 +81,10 @@ TODO : rewrite the pagination used in this template an Alpine one
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
{{ paginate(paginated_result, paginator) }}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block script %}
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
function formPagination(link){
|
function formPagination(link){
|
||||||
const form = document.getElementById("form")
|
const form = document.getElementById("form")
|
||||||
@@ -89,7 +93,6 @@ TODO : rewrite the pagination used in this template an Alpine one
|
|||||||
form.submit();
|
form.submit();
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
{{ paginate(paginated_result, paginator, "formPagination(this)") }}
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
from datetime import date, timedelta
|
from datetime import date, timedelta
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from django.test import Client
|
from django.contrib.auth.models import Permission
|
||||||
|
from django.test import Client, TestCase
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from model_bakery import baker
|
from model_bakery import baker
|
||||||
from model_bakery.recipe import Recipe
|
from model_bakery.recipe import Recipe
|
||||||
@@ -9,6 +10,54 @@ from pytest_django.asserts import assertNumQueries
|
|||||||
|
|
||||||
from club.models import Club, Membership
|
from club.models import Club, Membership
|
||||||
from core.baker_recipes import subscriber_user
|
from core.baker_recipes import subscriber_user
|
||||||
|
from core.models import Group, Page, User
|
||||||
|
|
||||||
|
|
||||||
|
class TestClubSearch(TestCase):
|
||||||
|
@classmethod
|
||||||
|
def setUpTestData(cls):
|
||||||
|
cls.url = reverse("api:search_club")
|
||||||
|
cls.user = baker.make(
|
||||||
|
User, user_permissions=[Permission.objects.get(codename="access_lookup")]
|
||||||
|
)
|
||||||
|
# delete existing clubs to avoid side effect
|
||||||
|
groups = list(
|
||||||
|
Group.objects.exclude(club=None, club_board=None).values_list(
|
||||||
|
"id", flat=True
|
||||||
|
)
|
||||||
|
)
|
||||||
|
Page.objects.exclude(club=None).delete()
|
||||||
|
Club.objects.all().delete()
|
||||||
|
Group.objects.filter(id__in=groups).delete()
|
||||||
|
|
||||||
|
cls.clubs = baker.make(
|
||||||
|
Club,
|
||||||
|
_quantity=5,
|
||||||
|
name=iter(["AE", "ae 1", "Troll", "Dev AE", "pdf"]),
|
||||||
|
is_active=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_inactive_club(self):
|
||||||
|
self.client.force_login(self.user)
|
||||||
|
inactive_ids = {self.clubs[0].id, self.clubs[2].id}
|
||||||
|
Club.objects.filter(id__in=inactive_ids).update(is_active=False)
|
||||||
|
response = self.client.get(self.url, {"is_active": False})
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert {d["id"] for d in response.json()["results"]} == inactive_ids
|
||||||
|
|
||||||
|
def test_excluded_id(self):
|
||||||
|
self.client.force_login(self.user)
|
||||||
|
response = self.client.get(self.url, {"exclude_ids": [self.clubs[1].id]})
|
||||||
|
assert response.status_code == 200
|
||||||
|
ids = {d["id"] for d in response.json()["results"]}
|
||||||
|
assert ids == {c.id for c in [self.clubs[0], *self.clubs[2:]]}
|
||||||
|
|
||||||
|
def test_club_search(self):
|
||||||
|
self.client.force_login(self.user)
|
||||||
|
response = self.client.get(self.url, {"search": "AE"})
|
||||||
|
assert response.status_code == 200
|
||||||
|
ids = {d["id"] for d in response.json()["results"]}
|
||||||
|
assert ids == {c.id for c in [self.clubs[0], self.clubs[1], self.clubs[3]]}
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
|
|||||||
107
club/views.py
107
club/views.py
@@ -23,6 +23,7 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
import csv
|
import csv
|
||||||
|
import itertools
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
@@ -30,18 +31,14 @@ from django.contrib.auth.mixins import PermissionRequiredMixin
|
|||||||
from django.contrib.messages.views import SuccessMessageMixin
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
from django.core.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError
|
from django.core.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError
|
||||||
from django.core.paginator import InvalidPage, Paginator
|
from django.core.paginator import InvalidPage, Paginator
|
||||||
from django.db.models import Q, Sum
|
from django.db.models import F, Q, Sum
|
||||||
from django.http import (
|
from django.http import Http404, HttpResponseRedirect, StreamingHttpResponse
|
||||||
Http404,
|
|
||||||
HttpResponseRedirect,
|
|
||||||
StreamingHttpResponse,
|
|
||||||
)
|
|
||||||
from django.shortcuts import get_object_or_404, redirect
|
from django.shortcuts import get_object_or_404, redirect
|
||||||
from django.urls import reverse, reverse_lazy
|
from django.urls import reverse, reverse_lazy
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.safestring import SafeString
|
from django.utils.safestring import SafeString
|
||||||
from django.utils.timezone import now
|
from django.utils.timezone import now
|
||||||
from django.utils.translation import gettext as _t
|
from django.utils.translation import gettext
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from django.views.generic import DetailView, ListView, View
|
from django.views.generic import DetailView, ListView, View
|
||||||
from django.views.generic.edit import CreateView, DeleteView, UpdateView
|
from django.views.generic.edit import CreateView, DeleteView, UpdateView
|
||||||
@@ -55,12 +52,7 @@ from club.forms import (
|
|||||||
MailingForm,
|
MailingForm,
|
||||||
SellingsForm,
|
SellingsForm,
|
||||||
)
|
)
|
||||||
from club.models import (
|
from club.models import Club, Mailing, MailingSubscription, Membership
|
||||||
Club,
|
|
||||||
Mailing,
|
|
||||||
MailingSubscription,
|
|
||||||
Membership,
|
|
||||||
)
|
|
||||||
from com.models import Poster
|
from com.models import Poster
|
||||||
from com.views import (
|
from com.views import (
|
||||||
PosterCreateBaseView,
|
PosterCreateBaseView,
|
||||||
@@ -68,9 +60,7 @@ from com.views import (
|
|||||||
PosterEditBaseView,
|
PosterEditBaseView,
|
||||||
PosterListBaseView,
|
PosterListBaseView,
|
||||||
)
|
)
|
||||||
from core.auth.mixins import (
|
from core.auth.mixins import CanEditMixin, PermissionOrClubBoardRequiredMixin
|
||||||
CanEditMixin,
|
|
||||||
)
|
|
||||||
from core.models import PageRev
|
from core.models import PageRev
|
||||||
from core.views import DetailFormView, PageEditViewBase, UseFragmentsMixin
|
from core.views import DetailFormView, PageEditViewBase, UseFragmentsMixin
|
||||||
from core.views.mixins import FragmentMixin, FragmentRenderer, TabedViewMixin
|
from core.views.mixins import FragmentMixin, FragmentRenderer, TabedViewMixin
|
||||||
@@ -381,7 +371,7 @@ class ClubOldMembersView(ClubTabsMixin, PermissionRequiredMixin, DetailView):
|
|||||||
|
|
||||||
|
|
||||||
class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
||||||
"""Sellings of a club."""
|
"""Sales of a club."""
|
||||||
|
|
||||||
model = Club
|
model = Club
|
||||||
pk_url_kwarg = "club_id"
|
pk_url_kwarg = "club_id"
|
||||||
@@ -407,9 +397,8 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
|||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
kwargs = super().get_context_data(**kwargs)
|
kwargs = super().get_context_data(**kwargs)
|
||||||
qs = Selling.objects.filter(club=self.object)
|
|
||||||
|
|
||||||
kwargs["result"] = qs[:0]
|
kwargs["result"] = Selling.objects.none()
|
||||||
kwargs["paginated_result"] = kwargs["result"]
|
kwargs["paginated_result"] = kwargs["result"]
|
||||||
kwargs["total"] = 0
|
kwargs["total"] = 0
|
||||||
kwargs["total_quantity"] = 0
|
kwargs["total_quantity"] = 0
|
||||||
@@ -417,6 +406,7 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
|||||||
|
|
||||||
form = self.get_form()
|
form = self.get_form()
|
||||||
if form.is_valid():
|
if form.is_valid():
|
||||||
|
qs = Selling.objects.filter(club=self.object)
|
||||||
if not len([v for v in form.cleaned_data.values() if v is not None]):
|
if not len([v for v in form.cleaned_data.values() if v is not None]):
|
||||||
qs = Selling.objects.none()
|
qs = Selling.objects.none()
|
||||||
if form.cleaned_data["begin_date"]:
|
if form.cleaned_data["begin_date"]:
|
||||||
@@ -436,18 +426,18 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView):
|
|||||||
if len(selected_products) > 0:
|
if len(selected_products) > 0:
|
||||||
qs = qs.filter(product__in=selected_products)
|
qs = qs.filter(product__in=selected_products)
|
||||||
|
|
||||||
|
kwargs["total"] = qs.annotate(
|
||||||
|
price=F("quantity") * F("unit_price")
|
||||||
|
).aggregate(total=Sum("price", default=0))["total"]
|
||||||
kwargs["result"] = qs.select_related(
|
kwargs["result"] = qs.select_related(
|
||||||
"counter", "counter__club", "customer", "customer__user", "seller"
|
"counter", "counter__club", "customer", "customer__user", "seller"
|
||||||
).order_by("-id")
|
).order_by("-id")
|
||||||
kwargs["total"] = sum([s.quantity * s.unit_price for s in kwargs["result"]])
|
kwargs["total_quantity"] = qs.aggregate(total=Sum("quantity", default=0))[
|
||||||
total_quantity = qs.all().aggregate(Sum("quantity"))
|
"total"
|
||||||
if total_quantity["quantity__sum"]:
|
]
|
||||||
kwargs["total_quantity"] = total_quantity["quantity__sum"]
|
kwargs["benefit"] = qs.exclude(product=None).aggregate(
|
||||||
benefit = (
|
res=Sum("product__purchase_price", default=0)
|
||||||
qs.exclude(product=None).all().aggregate(Sum("product__purchase_price"))
|
)["res"]
|
||||||
)
|
|
||||||
if benefit["product__purchase_price__sum"]:
|
|
||||||
kwargs["benefit"] = benefit["product__purchase_price__sum"]
|
|
||||||
|
|
||||||
kwargs["paginator"] = Paginator(kwargs["result"], self.paginate_by)
|
kwargs["paginator"] = Paginator(kwargs["result"], self.paginate_by)
|
||||||
try:
|
try:
|
||||||
@@ -498,40 +488,40 @@ class ClubSellingCSVView(ClubSellingView):
|
|||||||
kwargs = self.get_context_data(**kwargs)
|
kwargs = self.get_context_data(**kwargs)
|
||||||
|
|
||||||
# Use the StreamWriter class instead of request for streaming
|
# Use the StreamWriter class instead of request for streaming
|
||||||
pseudo_buffer = self.StreamWriter()
|
writer = csv.writer(self.StreamWriter())
|
||||||
writer = csv.writer(
|
|
||||||
pseudo_buffer, delimiter=";", lineterminator="\n", quoting=csv.QUOTE_ALL
|
|
||||||
)
|
|
||||||
|
|
||||||
writer.writerow([_t("Quantity"), kwargs["total_quantity"]])
|
first_rows = [
|
||||||
writer.writerow([_t("Total"), kwargs["total"]])
|
[gettext("Quantity"), kwargs["total_quantity"]],
|
||||||
writer.writerow([_t("Benefit"), kwargs["benefit"]])
|
[gettext("Total"), kwargs["total"]],
|
||||||
writer.writerow(
|
[gettext("Benefit"), kwargs["benefit"]],
|
||||||
[
|
[
|
||||||
_t("Date"),
|
gettext("Date"),
|
||||||
_t("Counter"),
|
gettext("Counter"),
|
||||||
_t("Barman"),
|
gettext("Barman"),
|
||||||
_t("Customer"),
|
gettext("Customer"),
|
||||||
_t("Label"),
|
gettext("Label"),
|
||||||
_t("Quantity"),
|
gettext("Quantity"),
|
||||||
_t("Total"),
|
gettext("Total"),
|
||||||
_t("Payment method"),
|
gettext("Payment method"),
|
||||||
_t("Selling price"),
|
gettext("Selling price"),
|
||||||
_t("Purchase price"),
|
gettext("Purchase price"),
|
||||||
_t("Benefit"),
|
gettext("Benefit"),
|
||||||
|
],
|
||||||
]
|
]
|
||||||
)
|
|
||||||
|
|
||||||
# Stream response
|
# Stream response
|
||||||
response = StreamingHttpResponse(
|
response = StreamingHttpResponse(
|
||||||
|
itertools.chain(
|
||||||
|
(writer.writerow(r) for r in first_rows),
|
||||||
(
|
(
|
||||||
writer.writerow(self.write_selling(selling))
|
writer.writerow(self.write_selling(selling))
|
||||||
for selling in kwargs["result"]
|
for selling in kwargs["result"]
|
||||||
),
|
),
|
||||||
|
),
|
||||||
content_type="text/csv",
|
content_type="text/csv",
|
||||||
)
|
)
|
||||||
name = _("Sellings") + "_" + self.object.name + ".csv"
|
name = f"{gettext('Sellings')}_{self.object.name}.csv"
|
||||||
response["Content-Disposition"] = "filename=" + name
|
response["Content-Disposition"] = f"attachment; filename={name}"
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
@@ -769,11 +759,13 @@ class MailingAutoGenerationView(View):
|
|||||||
return redirect("club:mailing", club_id=club.id)
|
return redirect("club:mailing", club_id=club.id)
|
||||||
|
|
||||||
|
|
||||||
class PosterListView(ClubTabsMixin, PosterListBaseView):
|
class PosterListView(
|
||||||
|
PermissionOrClubBoardRequiredMixin, ClubTabsMixin, PosterListBaseView
|
||||||
|
):
|
||||||
"""List communication posters."""
|
"""List communication posters."""
|
||||||
|
|
||||||
current_tab = "posters"
|
current_tab = "posters"
|
||||||
extra_context = {"app": "club"}
|
permission_required = "com.view_poster"
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
return super().get_queryset().filter(club=self.club.id)
|
return super().get_queryset().filter(club=self.club.id)
|
||||||
@@ -781,6 +773,17 @@ class PosterListView(ClubTabsMixin, PosterListBaseView):
|
|||||||
def get_object(self):
|
def get_object(self):
|
||||||
return self.club
|
return self.club
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
return super().get_context_data(**kwargs) | {
|
||||||
|
"create_url": reverse_lazy(
|
||||||
|
"club:poster_create", kwargs={"club_id": self.club.id}
|
||||||
|
),
|
||||||
|
"get_edit_url": lambda poster: reverse(
|
||||||
|
"club:poster_edit",
|
||||||
|
kwargs={"club_id": self.club.id, "poster_id": poster.id},
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class PosterCreateView(ClubTabsMixin, PosterCreateBaseView):
|
class PosterCreateView(ClubTabsMixin, PosterCreateBaseView):
|
||||||
"""Create communication poster."""
|
"""Create communication poster."""
|
||||||
|
|||||||
@@ -144,7 +144,7 @@ class News(models.Model):
|
|||||||
),
|
),
|
||||||
groups__id=settings.SITH_GROUP_COM_ADMIN_ID,
|
groups__id=settings.SITH_GROUP_COM_ADMIN_ID,
|
||||||
)
|
)
|
||||||
notif_url = reverse("com:news_admin_list")
|
notif_url = reverse("com:news_admin_list", fragment="moderation")
|
||||||
new_notifs = [
|
new_notifs = [
|
||||||
Notification(user=user, url=notif_url, type="NEWS_MODERATION")
|
Notification(user=user, url=notif_url, type="NEWS_MODERATION")
|
||||||
for user in admins_without_notif
|
for user in admins_without_notif
|
||||||
@@ -402,9 +402,7 @@ class Poster(models.Model):
|
|||||||
groups__id__in=[settings.SITH_GROUP_COM_ADMIN_ID]
|
groups__id__in=[settings.SITH_GROUP_COM_ADMIN_ID]
|
||||||
):
|
):
|
||||||
Notification.objects.create(
|
Notification.objects.create(
|
||||||
user=user,
|
user=user, url=reverse("com:poster_list"), type="POSTER_MODERATION"
|
||||||
url=reverse("com:poster_moderate_list"),
|
|
||||||
type="POSTER_MODERATION",
|
|
||||||
)
|
)
|
||||||
return super().save(*args, **kwargs)
|
return super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
|||||||
@@ -83,7 +83,8 @@
|
|||||||
#links_content {
|
#links_content {
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
box-shadow: $shadow-color 1px 1px 1px;
|
box-shadow: $shadow-color 1px 1px 1px;
|
||||||
height: 20em;
|
min-height: 20em;
|
||||||
|
padding-bottom: 1em;
|
||||||
|
|
||||||
h4 {
|
h4 {
|
||||||
margin-left: 5px;
|
margin-left: 5px;
|
||||||
|
|||||||
@@ -20,34 +20,8 @@
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
display: flex;
|
display: flex;
|
||||||
bottom: 5px;
|
bottom: 5px;
|
||||||
|
|
||||||
&.left {
|
|
||||||
left: 0;
|
left: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.right {
|
|
||||||
right: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.link {
|
|
||||||
padding: 5px;
|
|
||||||
padding-left: 20px;
|
|
||||||
padding-right: 20px;
|
|
||||||
margin-left: 5px;
|
|
||||||
border-radius: 20px;
|
|
||||||
background-color: hsl(40, 100%, 50%);
|
|
||||||
color: black;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
color: black;
|
|
||||||
background-color: hsl(40, 58%, 50%);
|
|
||||||
}
|
|
||||||
|
|
||||||
&.delete {
|
|
||||||
background-color: hsl(0, 100%, 40%);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#posters,
|
#posters,
|
||||||
@@ -143,43 +117,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.edit,
|
.actions {
|
||||||
.moderate,
|
display: flex;
|
||||||
.slideshow {
|
flex-direction: column;
|
||||||
padding: 5px;
|
align-items: stretch;
|
||||||
border-radius: 20px;
|
form {
|
||||||
background-color: hsl(40, 100%, 50%);
|
margin: unset;
|
||||||
color: black;
|
padding: unset;
|
||||||
|
button {
|
||||||
&:hover {
|
width: 100%;
|
||||||
color: black;
|
|
||||||
background-color: hsl(40, 58%, 50%);
|
|
||||||
}
|
|
||||||
|
|
||||||
&:nth-child(2n) {
|
|
||||||
margin-top: 5px;
|
|
||||||
margin-bottom: 5px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.tooltip {
|
|
||||||
visibility: hidden;
|
|
||||||
width: 120px;
|
|
||||||
background-color: hsl(210, 20%, 98%);
|
|
||||||
color: hsl(0, 0%, 0%);
|
|
||||||
text-align: center;
|
|
||||||
padding: 5px 0;
|
|
||||||
border-radius: 6px;
|
|
||||||
position: absolute;
|
|
||||||
z-index: 10;
|
|
||||||
|
|
||||||
ul {
|
|
||||||
margin-left: 0;
|
|
||||||
display: inline-block;
|
|
||||||
|
|
||||||
li {
|
|
||||||
display: list-item;
|
|
||||||
list-style-type: none;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -76,18 +76,20 @@
|
|||||||
It will stay hidden for other users until it has been published.
|
It will stay hidden for other users until it has been published.
|
||||||
{% endtrans %}
|
{% endtrans %}
|
||||||
</p>
|
</p>
|
||||||
{% if user.has_perm("com.moderate_news") %}
|
{%- if user.has_perm("com.moderate_news") -%}
|
||||||
{# This is an additional query for each non-moderated news,
|
{# This is an additional query for each non-moderated news,
|
||||||
but it will be executed only for admin users, and only one time
|
but it will be executed only for admin users, and only one time
|
||||||
(if they do their job and moderated news as soon as they see them),
|
(if they do their job and moderate news as soon as they see them),
|
||||||
so it's still reasonable #}
|
so it's still reasonable #}
|
||||||
<div
|
<div
|
||||||
{% if news is integer or news is string %}
|
{% if news is integer or news is string -%}
|
||||||
x-data="{ nbEvents: 0 }"
|
x-data="{ nbEvents: 0 }"
|
||||||
x-init="nbEvents = await nbToPublish()"
|
x-init="nbEvents = await nbToPublish()"
|
||||||
{% else %}
|
{%- elif news.is_published -%}
|
||||||
|
x-data="{ nbEvents: 0 }"
|
||||||
|
{%- else -%}
|
||||||
x-data="{ nbEvents: {{ news.dates.count() }} }"
|
x-data="{ nbEvents: {{ news.dates.count() }} }"
|
||||||
{% endif %}
|
{%- endif -%}
|
||||||
>
|
>
|
||||||
<template x-if="nbEvents > 1">
|
<template x-if="nbEvents > 1">
|
||||||
<div>
|
<div>
|
||||||
|
|||||||
@@ -131,7 +131,7 @@
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
<h5>{% trans %}Events to moderate{% endtrans %}</h5>
|
<h5 id="moderation">{% trans %}Events to moderate{% endtrans %}</h5>
|
||||||
<table>
|
<table>
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
@@ -165,6 +165,3 @@
|
|||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -205,6 +205,10 @@
|
|||||||
<i class="fa-solid fa-graduation-cap fa-xl"></i>
|
<i class="fa-solid fa-graduation-cap fa-xl"></i>
|
||||||
<a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a>
|
<a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a>
|
||||||
</li>
|
</li>
|
||||||
|
<li>
|
||||||
|
<i class="fa-solid fa-calendar-days fa-xl"></i>
|
||||||
|
<a href="{{ url("timetable:generator") }}">{% trans %}Timetable{% endtrans %}</a>
|
||||||
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<i class="fa-solid fa-magnifying-glass fa-xl"></i>
|
<i class="fa-solid fa-magnifying-glass fa-xl"></i>
|
||||||
<a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a>
|
<a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a>
|
||||||
|
|||||||
@@ -13,22 +13,15 @@
|
|||||||
|
|
||||||
<div id="title">
|
<div id="title">
|
||||||
<h3>{% trans %}Posters{% endtrans %}</h3>
|
<h3>{% trans %}Posters{% endtrans %}</h3>
|
||||||
<div id="links" class="right">
|
<div id="links">
|
||||||
{% if app == "com" %}
|
<a id="create" class="btn btn-blue" href="{{ create_url }}">
|
||||||
<a id="create" class="link" href="{{ url(app + ":poster_create") }}">{% trans %}Create{% endtrans %}</a>
|
<i class="fa fa-plus"></i>
|
||||||
<a id="moderation" class="link" href="{{ url("com:poster_moderate_list") }}">{% trans %}Moderation{% endtrans %}</a>
|
{% trans %}Create{% endtrans %}
|
||||||
{% elif app == "club" %}
|
</a>
|
||||||
<a id="create" class="link" href="{{ url(app + ":poster_create", club.id) }}">{% trans %}Create{% endtrans %}</a>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="posters">
|
<div id="posters">
|
||||||
|
|
||||||
{% if poster_list.count() == 0 %}
|
|
||||||
<div id="no-posters">{% trans %}No posters{% endtrans %}</div>
|
|
||||||
{% else %}
|
|
||||||
|
|
||||||
{% for poster in poster_list %}
|
{% for poster in poster_list %}
|
||||||
<div class="poster{% if not poster.is_moderated %} not_moderated{% endif %}">
|
<div class="poster{% if not poster.is_moderated %} not_moderated{% endif %}">
|
||||||
<div class="name">{{ poster.name }}</div>
|
<div class="name">{{ poster.name }}</div>
|
||||||
@@ -36,30 +29,37 @@
|
|||||||
class="image"
|
class="image"
|
||||||
hover="{% trans %}Click to expand{% endtrans %}"
|
hover="{% trans %}Click to expand{% endtrans %}"
|
||||||
@click="active = $el.firstElementChild"
|
@click="active = $el.firstElementChild"
|
||||||
|
tooltip="{%- for screen in poster.screens.all() -%}
|
||||||
|
{{ screen }}
|
||||||
|
{% endfor %}"
|
||||||
>
|
>
|
||||||
<img src="{{ poster.file.url }}"></img>
|
<img src="{{ poster.file.url }}" alt="{{ poster.name }}">
|
||||||
</div>
|
</div>
|
||||||
<div class="dates">
|
<div class="dates">
|
||||||
<div class="begin">{{ poster.date_begin | localtime | date("d/M/Y H:m") }}</div>
|
<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>
|
<div class="end">{{ poster.date_end | localtime | date("d/M/Y H:m") }}</div>
|
||||||
</div>
|
</div>
|
||||||
{% if app == "com" %}
|
<div class="actions">
|
||||||
<a class="edit" href="{{ url(app + ":poster_edit", poster.id) }}">{% trans %}Edit{% endtrans %}</a>
|
{% if poster.is_editable %}
|
||||||
{% elif app == "club" %}
|
<a class="btn btn-blue" href="{{ get_edit_url(poster) }}">
|
||||||
<a class="edit" href="{{ url(app + ":poster_edit", club.id, poster.id) }}">{% trans %}Edit{% endtrans %}</a>
|
<i class="fa fa-pen-to-square"></i>
|
||||||
|
{% trans %}Edit{% endtrans %}
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
{% if not poster.is_moderated and user.has_perm("com.moderate_poster") %}
|
||||||
|
<form action="{{ url("com:poster_moderate", object_id=poster.id) }}" method="post">
|
||||||
|
{% csrf_token %}
|
||||||
|
<button type="submit" class="btn btn-green">
|
||||||
|
<i class="fa fa-check"></i>
|
||||||
|
{% trans %}Moderate{% endtrans %}
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="tooltip">
|
|
||||||
<ul>
|
|
||||||
{% for screen in poster.screens.all() %}
|
|
||||||
<li>{{ screen }}</li>
|
|
||||||
{% endfor %}
|
|
||||||
</ul>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div id="no-posters">{% trans %}No posters{% endtrans %}</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
@@ -68,7 +68,9 @@
|
|||||||
@click="active = null"
|
@click="active = null"
|
||||||
:class="{active: active !== null}"
|
:class="{active: active !== null}"
|
||||||
>
|
>
|
||||||
<div id="placeholder"><img :src="active?.src"></div>
|
<div id="placeholder">
|
||||||
|
<img :src="active?.src" :alt="active?.name">
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,43 +0,0 @@
|
|||||||
{% extends "core/base.jinja" %}
|
|
||||||
|
|
||||||
{% block script %}
|
|
||||||
{{ super() }}
|
|
||||||
<script src="{{ static('com/js/poster_list.js') }}"></script>
|
|
||||||
{% endblock %}
|
|
||||||
|
|
||||||
{% block additional_css %}
|
|
||||||
<link rel="stylesheet" href="{{ static('com/css/posters.scss') }}">
|
|
||||||
{% endblock %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div id="poster_list">
|
|
||||||
|
|
||||||
<div id="title">
|
|
||||||
<div id="links" class="left">
|
|
||||||
<a id="list" class="link" href="{{ url("com:poster_list") }}">{% trans %}List{% endtrans %}</a>
|
|
||||||
</div>
|
|
||||||
<h3>{% trans %}Posters - moderation{% endtrans %}</h3>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="posters">
|
|
||||||
|
|
||||||
{% if object_list.count == 0 %}
|
|
||||||
<div id="no-posters">{% trans %}No objects{% endtrans %}</div>
|
|
||||||
{% else %}
|
|
||||||
|
|
||||||
{% for poster in object_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>
|
|
||||||
<a class="moderate" href="{{ url("com:poster_moderate", object_id=poster.id) }}">Moderate</a>
|
|
||||||
</div>
|
|
||||||
{% endfor %}
|
|
||||||
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="view"><div id="placeholder"></div></div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
||||||
@@ -17,7 +17,9 @@ from unittest.mock import patch
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
from django.contrib.auth.models import Permission
|
||||||
from django.contrib.sites.models import Site
|
from django.contrib.sites.models import Site
|
||||||
|
from django.core.files.uploadedfile import SimpleUploadedFile
|
||||||
from django.test import Client, TestCase
|
from django.test import Client, TestCase
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.utils import html
|
from django.utils import html
|
||||||
@@ -27,9 +29,10 @@ from model_bakery import baker
|
|||||||
from pytest_django.asserts import assertNumQueries, assertRedirects
|
from pytest_django.asserts import assertNumQueries, assertRedirects
|
||||||
|
|
||||||
from club.models import Club, Membership
|
from club.models import Club, Membership
|
||||||
from com.models import News, NewsDate, Sith, Weekmail, WeekmailArticle
|
from com.models import News, NewsDate, Poster, Sith, Weekmail, WeekmailArticle
|
||||||
from core.baker_recipes import subscriber_user
|
from core.baker_recipes import subscriber_user
|
||||||
from core.models import AnonymousUser, Group, User
|
from core.models import AnonymousUser, Group, User
|
||||||
|
from core.utils import RED_PIXEL_PNG
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture()
|
@pytest.fixture()
|
||||||
@@ -314,7 +317,6 @@ def test_feed(client: Client):
|
|||||||
[
|
[
|
||||||
reverse("com:poster_list"),
|
reverse("com:poster_list"),
|
||||||
reverse("com:poster_create"),
|
reverse("com:poster_create"),
|
||||||
reverse("com:poster_moderate_list"),
|
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
def test_poster_management_views_crash_test(client: Client, url: str):
|
def test_poster_management_views_crash_test(client: Client, url: str):
|
||||||
@@ -325,3 +327,37 @@ def test_poster_management_views_crash_test(client: Client, url: str):
|
|||||||
client.force_login(user)
|
client.force_login(user)
|
||||||
res = client.get(url)
|
res = client.get(url)
|
||||||
assert res.status_code == 200
|
assert res.status_code == 200
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"referer",
|
||||||
|
[
|
||||||
|
None,
|
||||||
|
reverse("com:poster_list"),
|
||||||
|
reverse("club:poster_list", kwargs={"club_id": settings.SITH_MAIN_CLUB_ID}),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_moderate_poster(client: Client, referer: str | None):
|
||||||
|
poster = baker.make(
|
||||||
|
Poster,
|
||||||
|
is_moderated=False,
|
||||||
|
file=SimpleUploadedFile("test.png", content=RED_PIXEL_PNG),
|
||||||
|
club_id=settings.SITH_MAIN_CLUB_ID,
|
||||||
|
)
|
||||||
|
user = baker.make(
|
||||||
|
User,
|
||||||
|
user_permissions=Permission.objects.filter(
|
||||||
|
codename__in=["view_poster", "moderate_poster"]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
client.force_login(user)
|
||||||
|
headers = {"REFERER": f"https://{settings.SITH_URL}{referer}"} if referer else {}
|
||||||
|
response = client.post(
|
||||||
|
reverse("com:poster_moderate", kwargs={"object_id": poster.id}), headers=headers
|
||||||
|
)
|
||||||
|
result_url = referer or reverse("com:poster_list")
|
||||||
|
assertRedirects(response, result_url)
|
||||||
|
poster.refresh_from_db()
|
||||||
|
assert poster.is_moderated
|
||||||
|
assert poster.moderator == user
|
||||||
|
|||||||
@@ -33,7 +33,6 @@ from com.views import (
|
|||||||
PosterDeleteView,
|
PosterDeleteView,
|
||||||
PosterEditView,
|
PosterEditView,
|
||||||
PosterListView,
|
PosterListView,
|
||||||
PosterModerateListView,
|
|
||||||
PosterModerateView,
|
PosterModerateView,
|
||||||
ScreenCreateView,
|
ScreenCreateView,
|
||||||
ScreenDeleteView,
|
ScreenDeleteView,
|
||||||
@@ -102,11 +101,6 @@ urlpatterns = [
|
|||||||
PosterDeleteView.as_view(),
|
PosterDeleteView.as_view(),
|
||||||
name="poster_delete",
|
name="poster_delete",
|
||||||
),
|
),
|
||||||
path(
|
|
||||||
"poster/moderate/",
|
|
||||||
PosterModerateListView.as_view(),
|
|
||||||
name="poster_moderate_list",
|
|
||||||
),
|
|
||||||
path(
|
path(
|
||||||
"poster/<int:object_id>/moderate/",
|
"poster/<int:object_id>/moderate/",
|
||||||
PosterModerateView.as_view(),
|
PosterModerateView.as_view(),
|
||||||
|
|||||||
67
com/views.py
67
com/views.py
@@ -25,6 +25,7 @@ import itertools
|
|||||||
from datetime import date, timedelta
|
from datetime import date, timedelta
|
||||||
from smtplib import SMTPRecipientsRefused
|
from smtplib import SMTPRecipientsRefused
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
from dateutil.relativedelta import relativedelta
|
from dateutil.relativedelta import relativedelta
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
@@ -34,7 +35,7 @@ from django.contrib.auth.mixins import (
|
|||||||
)
|
)
|
||||||
from django.contrib.syndication.views import Feed
|
from django.contrib.syndication.views import Feed
|
||||||
from django.core.exceptions import PermissionDenied, ValidationError
|
from django.core.exceptions import PermissionDenied, ValidationError
|
||||||
from django.db.models import Max
|
from django.db.models import Exists, Max, OuterRef, Value
|
||||||
from django.forms.models import modelform_factory
|
from django.forms.models import modelform_factory
|
||||||
from django.http import HttpResponseRedirect
|
from django.http import HttpResponseRedirect
|
||||||
from django.shortcuts import get_object_or_404, redirect
|
from django.shortcuts import get_object_or_404, redirect
|
||||||
@@ -45,7 +46,7 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from django.views.generic import DetailView, ListView, TemplateView, View
|
from django.views.generic import DetailView, ListView, TemplateView, View
|
||||||
from django.views.generic.edit import CreateView, DeleteView, UpdateView
|
from django.views.generic.edit import CreateView, DeleteView, UpdateView
|
||||||
|
|
||||||
from club.models import Club, Mailing
|
from club.models import Club, Mailing, Membership
|
||||||
from com.forms import NewsDateForm, NewsForm, PosterForm
|
from com.forms import NewsDateForm, NewsForm, PosterForm
|
||||||
from com.ics_calendar import IcsCalendar
|
from com.ics_calendar import IcsCalendar
|
||||||
from com.models import News, NewsDate, Poster, Screen, Sith, Weekmail, WeekmailArticle
|
from com.models import News, NewsDate, Poster, Screen, Sith, Weekmail, WeekmailArticle
|
||||||
@@ -561,16 +562,26 @@ class MailingModerateView(View):
|
|||||||
raise PermissionDenied
|
raise PermissionDenied
|
||||||
|
|
||||||
|
|
||||||
class PosterListBaseView(PermissionOrClubBoardRequiredMixin, ListView):
|
class PosterListBaseView(ListView):
|
||||||
"""List communication posters."""
|
"""List communication posters."""
|
||||||
|
|
||||||
model = Poster
|
model = Poster
|
||||||
template_name = "com/poster_list.jinja"
|
template_name = "com/poster_list.jinja"
|
||||||
permission_required = "com.view_poster"
|
permission_required = "com.view_poster"
|
||||||
ordering = ["-date_begin"]
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_queryset(self):
|
||||||
return super().get_context_data(**kwargs) | {"club": self.club}
|
qs = Poster.objects.prefetch_related("screens")
|
||||||
|
if self.request.user.has_perm("com.edit_poster"):
|
||||||
|
qs = qs.annotate(is_editable=Value(value=True))
|
||||||
|
else:
|
||||||
|
qs = qs.annotate(
|
||||||
|
is_editable=Exists(
|
||||||
|
Membership.objects.ongoing()
|
||||||
|
.board()
|
||||||
|
.filter(user=self.request.user, club=OuterRef("club_id"))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return qs.order_by("-date_begin")
|
||||||
|
|
||||||
|
|
||||||
class PosterCreateBaseView(PermissionOrClubBoardRequiredMixin, CreateView):
|
class PosterCreateBaseView(PermissionOrClubBoardRequiredMixin, CreateView):
|
||||||
@@ -633,21 +644,17 @@ class PosterDeleteBaseView(
|
|||||||
permission_required = "com.delete_poster"
|
permission_required = "com.delete_poster"
|
||||||
|
|
||||||
|
|
||||||
class PosterListView(ComTabsMixin, PosterListBaseView):
|
class PosterListView(PermissionRequiredMixin, ComTabsMixin, PosterListBaseView):
|
||||||
"""List communication posters."""
|
"""List communication posters."""
|
||||||
|
|
||||||
current_tab = "posters"
|
current_tab = "posters"
|
||||||
|
extra_context = {
|
||||||
def get_queryset(self):
|
"create_url": reverse_lazy("com:poster_create"),
|
||||||
qs = super().get_queryset()
|
"get_edit_url": lambda poster: reverse(
|
||||||
if self.request.user.has_perm("com.view_poster"):
|
"com:poster_edit", kwargs={"poster_id": poster.id}
|
||||||
return qs
|
),
|
||||||
return qs.filter(club=self.club.id)
|
}
|
||||||
|
permission_required = "com.view_poster"
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
kwargs = super().get_context_data(**kwargs)
|
|
||||||
kwargs["app"] = "com"
|
|
||||||
return kwargs
|
|
||||||
|
|
||||||
|
|
||||||
class PosterCreateView(ComTabsMixin, PosterCreateBaseView):
|
class PosterCreateView(ComTabsMixin, PosterCreateBaseView):
|
||||||
@@ -672,17 +679,6 @@ class PosterDeleteView(PosterDeleteBaseView):
|
|||||||
success_url = reverse_lazy("com:poster_list")
|
success_url = reverse_lazy("com:poster_list")
|
||||||
|
|
||||||
|
|
||||||
class PosterModerateListView(PermissionRequiredMixin, ComTabsMixin, ListView):
|
|
||||||
"""Moderate list communication poster."""
|
|
||||||
|
|
||||||
current_tab = "posters"
|
|
||||||
model = Poster
|
|
||||||
template_name = "com/poster_moderate.jinja"
|
|
||||||
queryset = Poster.objects.filter(is_moderated=False).all()
|
|
||||||
permission_required = "com.moderate_poster"
|
|
||||||
extra_context = {"app": "com"}
|
|
||||||
|
|
||||||
|
|
||||||
class PosterModerateView(PermissionRequiredMixin, ComTabsMixin, View):
|
class PosterModerateView(PermissionRequiredMixin, ComTabsMixin, View):
|
||||||
"""Moderate communication poster."""
|
"""Moderate communication poster."""
|
||||||
|
|
||||||
@@ -690,12 +686,21 @@ class PosterModerateView(PermissionRequiredMixin, ComTabsMixin, View):
|
|||||||
permission_required = "com.moderate_poster"
|
permission_required = "com.moderate_poster"
|
||||||
extra_context = {"app": "com"}
|
extra_context = {"app": "com"}
|
||||||
|
|
||||||
def get(self, request, *args, **kwargs):
|
def post(self, request, *args, **kwargs):
|
||||||
obj = get_object_or_404(Poster, pk=kwargs["object_id"])
|
obj = get_object_or_404(Poster, pk=kwargs["object_id"])
|
||||||
obj.is_moderated = True
|
obj.is_moderated = True
|
||||||
obj.moderator = request.user
|
obj.moderator = request.user
|
||||||
obj.save()
|
obj.save()
|
||||||
return redirect("com:poster_moderate_list")
|
# The moderation request may be originated from a club context (/club/poster)
|
||||||
|
# or a global context (/com/poster),
|
||||||
|
# so the redirection URL will be the URL of the page that called this view,
|
||||||
|
# as long as the latter belongs to the sith.
|
||||||
|
referer = self.request.META.get("HTTP_REFERER")
|
||||||
|
if referer:
|
||||||
|
parsed = urlparse(referer)
|
||||||
|
if parsed.netloc == settings.SITH_URL:
|
||||||
|
return redirect(parsed.path)
|
||||||
|
return redirect(reverse("com:poster_list"))
|
||||||
|
|
||||||
|
|
||||||
class ScreenListView(PermissionRequiredMixin, ComTabsMixin, ListView):
|
class ScreenListView(PermissionRequiredMixin, ComTabsMixin, ListView):
|
||||||
|
|||||||
@@ -651,9 +651,6 @@ class User(AbstractUser):
|
|||||||
|
|
||||||
|
|
||||||
class AnonymousUser(AuthAnonymousUser):
|
class AnonymousUser(AuthAnonymousUser):
|
||||||
def __init__(self):
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def was_subscribed(self):
|
def was_subscribed(self):
|
||||||
return False
|
return False
|
||||||
@@ -662,10 +659,6 @@ class AnonymousUser(AuthAnonymousUser):
|
|||||||
def is_subscribed(self):
|
def is_subscribed(self):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@property
|
|
||||||
def subscribed(self):
|
|
||||||
return False
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_root(self):
|
def is_root(self):
|
||||||
return False
|
return False
|
||||||
@@ -1164,8 +1157,6 @@ class QuickUploadImage(models.Model):
|
|||||||
identifier = str(uuid4())
|
identifier = str(uuid4())
|
||||||
name = Path(image.name).stem[: cls.IMAGE_NAME_SIZE - 1]
|
name = Path(image.name).stem[: cls.IMAGE_NAME_SIZE - 1]
|
||||||
file = File(convert_image(image), name=f"{identifier}.webp")
|
file = File(convert_image(image), name=f"{identifier}.webp")
|
||||||
width, height = Image.open(file).size
|
|
||||||
|
|
||||||
return cls.objects.create(
|
return cls.objects.create(
|
||||||
uuid=identifier,
|
uuid=identifier,
|
||||||
name=name,
|
name=name,
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import { alpinePlugin } from "#core:utils/notifications";
|
import { alpinePlugin as notificationPlugin } from "#core:utils/notifications";
|
||||||
import sort from "@alpinejs/sort";
|
import sort from "@alpinejs/sort";
|
||||||
import Alpine from "alpinejs";
|
import Alpine from "alpinejs";
|
||||||
|
|
||||||
Alpine.plugin(sort);
|
Alpine.plugin(sort);
|
||||||
Alpine.magic("notifications", alpinePlugin);
|
Alpine.magic("notifications", notificationPlugin);
|
||||||
window.Alpine = Alpine;
|
window.Alpine = Alpine;
|
||||||
|
|
||||||
window.addEventListener("DOMContentLoaded", () => {
|
window.addEventListener("DOMContentLoaded", () => {
|
||||||
|
|||||||
@@ -154,11 +154,9 @@ form {
|
|||||||
margin-bottom: 1rem;
|
margin-bottom: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.row {
|
.row > label {
|
||||||
label {
|
|
||||||
margin: unset;
|
margin: unset;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// ------------- LABEL
|
// ------------- LABEL
|
||||||
label, legend {
|
label, legend {
|
||||||
|
|||||||
@@ -503,6 +503,10 @@ th {
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
padding: 5px 10px;
|
padding: 5px 10px;
|
||||||
|
|
||||||
|
>input[type="checkbox"] {
|
||||||
|
padding: unset;
|
||||||
|
}
|
||||||
|
|
||||||
>ul {
|
>ul {
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -77,22 +77,22 @@
|
|||||||
<div class="notification" x-data="{display: false}" :class="{white: display}">
|
<div class="notification" x-data="{display: false}" :class="{white: display}">
|
||||||
<a href="#" @click.prevent="display = !display">
|
<a href="#" @click.prevent="display = !display">
|
||||||
<i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i>
|
<i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i>
|
||||||
{% set notification_count = user.notifications.filter(viewed=False).count() %}
|
{% set notifications = user.notifications.filter(viewed=False).order_by("-date")|list %}
|
||||||
|
|
||||||
{% if notification_count > 0 %}
|
{%- if notifications|length > 0 -%}
|
||||||
<span>
|
<span>
|
||||||
{% if notification_count < 100 %}
|
{% if notifications|length < 100 %}
|
||||||
{{ notification_count }}
|
{{ notifications|length }}
|
||||||
{% else %}
|
{%- else -%}
|
||||||
|
99+
|
||||||
{% endif %}
|
{%- endif -%}
|
||||||
</span>
|
</span>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</a>
|
</a>
|
||||||
<div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false">
|
<div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false">
|
||||||
<ul>
|
<ul>
|
||||||
{% if user.notifications.filter(viewed=False).count() > 0 %}
|
{%- if notifications|length > 0 -%}
|
||||||
{% for n in user.notifications.filter(viewed=False).order_by('-date') %}
|
{%- for n in notifications -%}
|
||||||
<li>
|
<li>
|
||||||
<a href="{{ url("core:notification", notif_id=n.id) }}">
|
<a href="{{ url("core:notification", notif_id=n.id) }}">
|
||||||
<div class="datetime">
|
<div class="datetime">
|
||||||
@@ -108,10 +108,10 @@
|
|||||||
</div>
|
</div>
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
{% endfor %}
|
{%- endfor -%}
|
||||||
{% else %}
|
{%- else -%}
|
||||||
<li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li>
|
<li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li>
|
||||||
{% endif %}
|
{%- endif -%}
|
||||||
</ul>
|
</ul>
|
||||||
<div class="options">
|
<div class="options">
|
||||||
<a href="{{ url('core:notification_list') }}">
|
<a href="{{ url('core:notification_list') }}">
|
||||||
|
|||||||
@@ -13,10 +13,10 @@
|
|||||||
}"
|
}"
|
||||||
@quick-notification-add="(e) => messages.push(e?.detail)"
|
@quick-notification-add="(e) => messages.push(e?.detail)"
|
||||||
@quick-notification-delete="messages = []">
|
@quick-notification-delete="messages = []">
|
||||||
<template x-for="message in messages">
|
<template x-for="(message, index) in messages">
|
||||||
<div x-data="{show: true}" class="alert" :class="`alert-${message.tag}`" x-show="show" x-transition>
|
<div class="alert" :class="`alert-${message.tag}`" x-transition>
|
||||||
<span class="alert-main" x-text="message.text"></span>
|
<span class="alert-main" x-text="message.text"></span>
|
||||||
<span class="clickable" @click="show = false">
|
<span class="clickable" @click="messages = messages.filter((item, i) => i !== index)">
|
||||||
<i class="fa fa-close"></i>
|
<i class="fa fa-close"></i>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -245,3 +245,26 @@
|
|||||||
<button type="button" onclick="checkbox_{{form_id}}(true);">{% trans %}Select All{% endtrans %}</button>
|
<button type="button" onclick="checkbox_{{form_id}}(true);">{% trans %}Select All{% endtrans %}</button>
|
||||||
<button type="button" onclick="checkbox_{{form_id}}(false);">{% trans %}Unselect All{% endtrans %}</button>
|
<button type="button" onclick="checkbox_{{form_id}}(false);">{% trans %}Unselect All{% endtrans %}</button>
|
||||||
{% endmacro %}
|
{% endmacro %}
|
||||||
|
|
||||||
|
{% macro update_notifications(messages, clear) %}
|
||||||
|
{# Update notification area from new messages sent by django backend
|
||||||
|
This is useful when performing fragment swaps to keep messages up to date
|
||||||
|
Without this, the fragment would need to take control of the notification area and
|
||||||
|
this would be an issue when having more than one fragment
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
messages: messages from django.contrib
|
||||||
|
clear : optional boolean that controls if notifications should be cleared first. True is the default
|
||||||
|
#}
|
||||||
|
{% set clear = clear|default(true) %}
|
||||||
|
{% if messages %}
|
||||||
|
<div x-init="() => {
|
||||||
|
{% if clear %}
|
||||||
|
$notifications.clear()
|
||||||
|
{% endif %}
|
||||||
|
{% for message in messages %}
|
||||||
|
$notifications.{{ message.tags }}('{{ message }}')
|
||||||
|
{% endfor %}
|
||||||
|
}"></div>
|
||||||
|
{% endif %}
|
||||||
|
{% endmacro %}
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
{% for js in statics.js %}
|
{% spaceless %}
|
||||||
|
{% for js in statics.js %}
|
||||||
<script-once type="module" src="{{ js }}"></script-once>
|
<script-once type="module" src="{{ js }}"></script-once>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% for css in statics.css %}
|
{% for css in statics.css %}
|
||||||
<link-once rel="stylesheet" type="text/css" href="{{ css }}" defer></link-once>
|
<link-once rel="stylesheet" type="text/css" href="{{ css }}" defer></link-once>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
<{{ component }} name="{{ widget.name }}" {% include "django/forms/widgets/attrs.html" %}>
|
<{{ component }} name="{{ widget.name }}" {% include "django/forms/widgets/attrs.html" %}>
|
||||||
{% for group_name, group_choices, group_index in widget.optgroups %}
|
{% for group_name, group_choices, group_index in widget.optgroups %}
|
||||||
{% if group_name %}
|
{% if group_name %}
|
||||||
<optgroup label="{{ group_name }}">
|
<optgroup label="{{ group_name }}">
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@@ -16,8 +17,9 @@
|
|||||||
{% if group_name %}
|
{% if group_name %}
|
||||||
</optgroup>
|
</optgroup>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% if initial %}
|
{% if initial %}
|
||||||
<slot style="display:none" name="initial">{{ initial }}</slot>
|
<slot style="display:none" name="initial">{{ initial }}</slot>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</{{ component }}>
|
</{{ component }}>
|
||||||
|
{% endspaceless %}
|
||||||
@@ -115,7 +115,7 @@ class SelectUser(TextInput):
|
|||||||
|
|
||||||
def validate_future_timestamp(value: date | datetime):
|
def validate_future_timestamp(value: date | datetime):
|
||||||
if value <= now():
|
if value <= now():
|
||||||
raise ValueError(_("Ensure this timestamp is set in the future"))
|
raise ValidationError(_("Ensure this timestamp is set in the future"))
|
||||||
|
|
||||||
|
|
||||||
class FutureDateTimeField(forms.DateTimeField):
|
class FutureDateTimeField(forms.DateTimeField):
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ from counter.models import (
|
|||||||
Counter,
|
Counter,
|
||||||
Customer,
|
Customer,
|
||||||
Eticket,
|
Eticket,
|
||||||
|
InvoiceCall,
|
||||||
Permanency,
|
Permanency,
|
||||||
Product,
|
Product,
|
||||||
ProductType,
|
ProductType,
|
||||||
@@ -160,3 +161,11 @@ class CashRegisterSummaryAdmin(SearchModelAdmin):
|
|||||||
class EticketAdmin(SearchModelAdmin):
|
class EticketAdmin(SearchModelAdmin):
|
||||||
list_display = ("product", "event_date", "event_title")
|
list_display = ("product", "event_date", "event_title")
|
||||||
search_fields = ("product__name", "event_title")
|
search_fields = ("product__name", "event_title")
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(InvoiceCall)
|
||||||
|
class InvoiceCallAdmin(SearchModelAdmin):
|
||||||
|
list_display = ("club", "month", "is_validated")
|
||||||
|
search_fields = ("club__name",)
|
||||||
|
list_filter = (("club", admin.RelatedOnlyFieldListFilter),)
|
||||||
|
date_hierarchy = "month"
|
||||||
|
|||||||
187
counter/forms.py
187
counter/forms.py
@@ -1,13 +1,26 @@
|
|||||||
|
import json
|
||||||
import math
|
import math
|
||||||
|
import uuid
|
||||||
|
from datetime import date
|
||||||
|
|
||||||
|
from dateutil.relativedelta import relativedelta
|
||||||
from django import forms
|
from django import forms
|
||||||
from django.db.models import Q
|
from django.db.models import Exists, OuterRef, Q
|
||||||
|
from django.forms import BaseModelFormSet
|
||||||
|
from django.utils.timezone import now
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from django_celery_beat.models import ClockedSchedule
|
||||||
from phonenumber_field.widgets import RegionalPhoneNumberWidget
|
from phonenumber_field.widgets import RegionalPhoneNumberWidget
|
||||||
|
|
||||||
|
from club.models import Club
|
||||||
from club.widgets.ajax_select import AutoCompleteSelectClub
|
from club.widgets.ajax_select import AutoCompleteSelectClub
|
||||||
from core.models import User
|
from core.models import User
|
||||||
from core.views.forms import NFCTextInput, SelectDate, SelectDateTime
|
from core.views.forms import (
|
||||||
|
FutureDateTimeField,
|
||||||
|
NFCTextInput,
|
||||||
|
SelectDate,
|
||||||
|
SelectDateTime,
|
||||||
|
)
|
||||||
from core.views.widgets.ajax_select import (
|
from core.views.widgets.ajax_select import (
|
||||||
AutoCompleteSelect,
|
AutoCompleteSelect,
|
||||||
AutoCompleteSelectMultipleGroup,
|
AutoCompleteSelectMultipleGroup,
|
||||||
@@ -19,10 +32,14 @@ from counter.models import (
|
|||||||
Counter,
|
Counter,
|
||||||
Customer,
|
Customer,
|
||||||
Eticket,
|
Eticket,
|
||||||
|
InvoiceCall,
|
||||||
Product,
|
Product,
|
||||||
Refilling,
|
Refilling,
|
||||||
ReturnableProduct,
|
ReturnableProduct,
|
||||||
|
ScheduledProductAction,
|
||||||
|
Selling,
|
||||||
StudentCard,
|
StudentCard,
|
||||||
|
get_product_actions,
|
||||||
)
|
)
|
||||||
from counter.widgets.ajax_select import (
|
from counter.widgets.ajax_select import (
|
||||||
AutoCompleteSelectMultipleCounter,
|
AutoCompleteSelectMultipleCounter,
|
||||||
@@ -158,7 +175,101 @@ class CounterEditForm(forms.ModelForm):
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class ProductEditForm(forms.ModelForm):
|
class ScheduledProductActionForm(forms.ModelForm):
|
||||||
|
"""Form for automatic product archiving.
|
||||||
|
|
||||||
|
The `save` method will update or create tasks using celery-beat.
|
||||||
|
"""
|
||||||
|
|
||||||
|
required_css_class = "required"
|
||||||
|
prefix = "scheduled"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = ScheduledProductAction
|
||||||
|
fields = ["task"]
|
||||||
|
widgets = {"task": forms.RadioSelect(choices=get_product_actions)}
|
||||||
|
labels = {"task": _("Action")}
|
||||||
|
help_texts = {"task": ""}
|
||||||
|
|
||||||
|
trigger_at = FutureDateTimeField(
|
||||||
|
label=_("Date and time of action"), widget=SelectDateTime
|
||||||
|
)
|
||||||
|
counters = forms.ModelMultipleChoiceField(
|
||||||
|
label=_("New counters"),
|
||||||
|
help_text=_("The selected counters will replace the current ones"),
|
||||||
|
required=False,
|
||||||
|
widget=AutoCompleteSelectMultipleCounter,
|
||||||
|
queryset=Counter.objects.all(),
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, *args, product: Product, **kwargs):
|
||||||
|
self.product = product
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
if not self.instance._state.adding:
|
||||||
|
self.fields["trigger_at"].initial = self.instance.clocked.clocked_time
|
||||||
|
self.fields["counters"].initial = json.loads(self.instance.kwargs).get(
|
||||||
|
"counters"
|
||||||
|
)
|
||||||
|
|
||||||
|
def clean(self):
|
||||||
|
if not self.changed_data or "trigger_at" in self.errors:
|
||||||
|
return super().clean()
|
||||||
|
if "trigger_at" in self.changed_data:
|
||||||
|
if not self.instance.clocked_id:
|
||||||
|
self.instance.clocked = ClockedSchedule(
|
||||||
|
clocked_time=self.cleaned_data["trigger_at"]
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self.instance.clocked.clocked_time = self.cleaned_data["trigger_at"]
|
||||||
|
self.instance.clocked.save()
|
||||||
|
task_kwargs = {"product_id": self.product.id}
|
||||||
|
if (
|
||||||
|
self.cleaned_data["task"] == "counter.tasks.change_counters"
|
||||||
|
and "counters" in self.changed_data
|
||||||
|
):
|
||||||
|
task_kwargs["counters"] = [c.id for c in self.cleaned_data["counters"]]
|
||||||
|
self.instance.product = self.product
|
||||||
|
self.instance.kwargs = json.dumps(task_kwargs)
|
||||||
|
self.instance.name = (
|
||||||
|
f"{self.cleaned_data['task']} - {self.product} - {uuid.uuid4()}"
|
||||||
|
)
|
||||||
|
return super().clean()
|
||||||
|
|
||||||
|
|
||||||
|
class BaseScheduledProductActionFormSet(BaseModelFormSet):
|
||||||
|
def __init__(self, *args, product: Product, **kwargs):
|
||||||
|
if product.id:
|
||||||
|
queryset = (
|
||||||
|
product.scheduled_actions.filter(
|
||||||
|
enabled=True, clocked__clocked_time__gt=now()
|
||||||
|
)
|
||||||
|
.order_by("clocked__clocked_time")
|
||||||
|
.select_related("clocked")
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
queryset = ScheduledProductAction.objects.none()
|
||||||
|
form_kwargs = {"product": product}
|
||||||
|
super().__init__(*args, queryset=queryset, form_kwargs=form_kwargs, **kwargs)
|
||||||
|
|
||||||
|
def delete_existing(self, obj: ScheduledProductAction, commit: bool = True): # noqa FBT001
|
||||||
|
clocked = obj.clocked
|
||||||
|
super().delete_existing(obj, commit=commit)
|
||||||
|
if commit:
|
||||||
|
clocked.delete()
|
||||||
|
|
||||||
|
|
||||||
|
ScheduledProductActionFormSet = forms.modelformset_factory(
|
||||||
|
ScheduledProductAction,
|
||||||
|
ScheduledProductActionForm,
|
||||||
|
formset=BaseScheduledProductActionFormSet,
|
||||||
|
absolute_max=None,
|
||||||
|
can_delete=True,
|
||||||
|
can_delete_extra=False,
|
||||||
|
extra=2,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ProductForm(forms.ModelForm):
|
||||||
error_css_class = "error"
|
error_css_class = "error"
|
||||||
required_css_class = "required"
|
required_css_class = "required"
|
||||||
|
|
||||||
@@ -199,22 +310,21 @@ class ProductEditForm(forms.ModelForm):
|
|||||||
queryset=Counter.objects.all(),
|
queryset=Counter.objects.all(),
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, instance=None, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, instance=instance, **kwargs)
|
||||||
if self.instance.id:
|
if self.instance.id:
|
||||||
self.fields["counters"].initial = self.instance.counters.all()
|
self.fields["counters"].initial = self.instance.counters.all()
|
||||||
|
self.action_formset = ScheduledProductActionFormSet(
|
||||||
|
*args, product=self.instance, **kwargs
|
||||||
|
)
|
||||||
|
|
||||||
|
def is_valid(self):
|
||||||
|
return super().is_valid() and self.action_formset.is_valid()
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
ret = super().save(*args, **kwargs)
|
ret = super().save(*args, **kwargs)
|
||||||
if self.fields["counters"].initial:
|
self.instance.counters.set(self.cleaned_data["counters"])
|
||||||
# Remove the product from all counter it was added to
|
self.action_formset.save()
|
||||||
# It will then only be added to selected counters
|
|
||||||
for counter in self.fields["counters"].initial:
|
|
||||||
counter.products.remove(self.instance)
|
|
||||||
counter.save()
|
|
||||||
for counter in self.cleaned_data["counters"]:
|
|
||||||
counter.products.add(self.instance)
|
|
||||||
counter.save()
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
|
|
||||||
@@ -266,7 +376,7 @@ class CloseCustomerAccountForm(forms.Form):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class ProductForm(forms.Form):
|
class BasketProductForm(forms.Form):
|
||||||
quantity = forms.IntegerField(min_value=1, required=True)
|
quantity = forms.IntegerField(min_value=1, required=True)
|
||||||
id = forms.IntegerField(min_value=0, required=True)
|
id = forms.IntegerField(min_value=0, required=True)
|
||||||
|
|
||||||
@@ -371,5 +481,50 @@ class BaseBasketForm(forms.BaseFormSet):
|
|||||||
|
|
||||||
|
|
||||||
BasketForm = forms.formset_factory(
|
BasketForm = forms.formset_factory(
|
||||||
ProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1
|
BasketProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class InvoiceCallForm(forms.Form):
|
||||||
|
def __init__(self, *args, month: date, **kwargs):
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
self.month = month
|
||||||
|
self.clubs = list(
|
||||||
|
Club.objects.filter(
|
||||||
|
Exists(
|
||||||
|
Selling.objects.filter(
|
||||||
|
club=OuterRef("pk"),
|
||||||
|
date__gte=month,
|
||||||
|
date__lte=month + relativedelta(months=1),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
).annotate(
|
||||||
|
validated_invoice=Exists(
|
||||||
|
InvoiceCall.objects.filter(
|
||||||
|
club=OuterRef("pk"), month=month, is_validated=True
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self.fields = {
|
||||||
|
str(club.id): forms.BooleanField(
|
||||||
|
required=False, initial=club.validated_invoice
|
||||||
|
)
|
||||||
|
for club in self.clubs
|
||||||
|
}
|
||||||
|
|
||||||
|
def save(self):
|
||||||
|
invoice_calls = [
|
||||||
|
InvoiceCall(
|
||||||
|
month=self.month,
|
||||||
|
club_id=club.id,
|
||||||
|
is_validated=self.cleaned_data.get(str(club.id), False),
|
||||||
|
)
|
||||||
|
for club in self.clubs
|
||||||
|
]
|
||||||
|
InvoiceCall.objects.bulk_create(
|
||||||
|
invoice_calls,
|
||||||
|
update_conflicts=True,
|
||||||
|
update_fields=["is_validated"],
|
||||||
|
unique_fields=["month", "club"],
|
||||||
|
)
|
||||||
|
|||||||
40
counter/migrations/0032_scheduledproductaction.py
Normal file
40
counter/migrations/0032_scheduledproductaction.py
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
# Generated by Django 5.2.3 on 2025-09-14 11:29
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("counter", "0031_alter_counter_options"),
|
||||||
|
("django_celery_beat", "0019_alter_periodictasks_options"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="ScheduledProductAction",
|
||||||
|
fields=[
|
||||||
|
(
|
||||||
|
"periodictask_ptr",
|
||||||
|
models.OneToOneField(
|
||||||
|
auto_created=True,
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
parent_link=True,
|
||||||
|
primary_key=True,
|
||||||
|
serialize=False,
|
||||||
|
to="django_celery_beat.periodictask",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"product",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="scheduled_actions",
|
||||||
|
to="counter.product",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={"verbose_name": "Product scheduled action"},
|
||||||
|
bases=("django_celery_beat.periodictask",),
|
||||||
|
),
|
||||||
|
]
|
||||||
51
counter/migrations/0033_invoicecall.py
Normal file
51
counter/migrations/0033_invoicecall.py
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
# Generated by Django 5.2.3 on 2025-10-15 21:54
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
import counter.models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("club", "0014_alter_club_options_rename_unix_name_club_slug_name_and_more"),
|
||||||
|
("counter", "0032_scheduledproductaction"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="InvoiceCall",
|
||||||
|
fields=[
|
||||||
|
(
|
||||||
|
"id",
|
||||||
|
models.AutoField(
|
||||||
|
auto_created=True,
|
||||||
|
primary_key=True,
|
||||||
|
serialize=False,
|
||||||
|
verbose_name="ID",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"is_validated",
|
||||||
|
models.BooleanField(default=False, verbose_name="is validated"),
|
||||||
|
),
|
||||||
|
("month", counter.models.MonthField(verbose_name="invoice date")),
|
||||||
|
(
|
||||||
|
"club",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE, to="club.club"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"verbose_name": "Invoice call",
|
||||||
|
"verbose_name_plural": "Invoice calls",
|
||||||
|
"constraints": [
|
||||||
|
models.UniqueConstraint(
|
||||||
|
fields=("club", "month"),
|
||||||
|
name="counter_invoicecall_unique_club_month",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
# Generated by Django 5.2.3 on 2025-11-05 08:47
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [("counter", "0033_invoicecall")]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="selling",
|
||||||
|
name="date",
|
||||||
|
field=models.DateTimeField(db_index=True, verbose_name="date"),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -15,6 +15,7 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import base64
|
import base64
|
||||||
|
import contextlib
|
||||||
import os
|
import os
|
||||||
import random
|
import random
|
||||||
import string
|
import string
|
||||||
@@ -34,6 +35,7 @@ from django.urls import reverse
|
|||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.functional import cached_property
|
from django.utils.functional import cached_property
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from django_celery_beat.models import PeriodicTask
|
||||||
from django_countries.fields import CountryField
|
from django_countries.fields import CountryField
|
||||||
from ordered_model.models import OrderedModel
|
from ordered_model.models import OrderedModel
|
||||||
from phonenumber_field.modelfields import PhoneNumberField
|
from phonenumber_field.modelfields import PhoneNumberField
|
||||||
@@ -84,7 +86,7 @@ class CustomerQuerySet(models.QuerySet):
|
|||||||
.annotate(res=Sum(F("unit_price") * F("quantity"), default=0))
|
.annotate(res=Sum(F("unit_price") * F("quantity"), default=0))
|
||||||
.values("res")
|
.values("res")
|
||||||
)
|
)
|
||||||
return self.update(amount=Coalesce(money_in - money_out, Decimal("0")))
|
return self.update(amount=Coalesce(money_in - money_out, Decimal(0)))
|
||||||
|
|
||||||
|
|
||||||
class Customer(models.Model):
|
class Customer(models.Model):
|
||||||
@@ -445,7 +447,8 @@ class Product(models.Model):
|
|||||||
buying_groups = list(self.buying_groups.all())
|
buying_groups = list(self.buying_groups.all())
|
||||||
if not buying_groups:
|
if not buying_groups:
|
||||||
return True
|
return True
|
||||||
return any(user.is_in_group(pk=group.id) for group in buying_groups)
|
res = any(user.is_in_group(pk=group.id) for group in buying_groups)
|
||||||
|
return res
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def profit(self):
|
def profit(self):
|
||||||
@@ -479,7 +482,7 @@ class CounterQuerySet(models.QuerySet):
|
|||||||
return self.annotate(has_annotated_barman=Exists(subquery))
|
return self.annotate(has_annotated_barman=Exists(subquery))
|
||||||
|
|
||||||
def annotate_is_open(self) -> Self:
|
def annotate_is_open(self) -> Self:
|
||||||
"""Annotate tue queryset with the `is_open` field.
|
"""Annotate the queryset with the `is_open` field.
|
||||||
|
|
||||||
For each counter, if `is_open=True`, then the counter is currently opened.
|
For each counter, if `is_open=True`, then the counter is currently opened.
|
||||||
Else the counter is closed.
|
Else the counter is closed.
|
||||||
@@ -846,7 +849,7 @@ class Selling(models.Model):
|
|||||||
blank=False,
|
blank=False,
|
||||||
on_delete=models.SET_NULL,
|
on_delete=models.SET_NULL,
|
||||||
)
|
)
|
||||||
date = models.DateTimeField(_("date"))
|
date = models.DateTimeField(_("date"), db_index=True)
|
||||||
payment_method = models.CharField(
|
payment_method = models.CharField(
|
||||||
_("payment method"),
|
_("payment method"),
|
||||||
max_length=255,
|
max_length=255,
|
||||||
@@ -1357,3 +1360,85 @@ class ReturnableProductBalance(models.Model):
|
|||||||
f"return balance of {self.customer} "
|
f"return balance of {self.customer} "
|
||||||
f"for {self.returnable.product_id} : {self.balance}"
|
f"for {self.returnable.product_id} : {self.balance}"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_product_actions():
|
||||||
|
return [
|
||||||
|
("counter.tasks.archive_product", _("Archiving")),
|
||||||
|
("counter.tasks.change_counters", _("Counters change")),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ScheduledProductAction(PeriodicTask):
|
||||||
|
"""Extension of celery-beat tasks dedicated to perform actions on Product."""
|
||||||
|
|
||||||
|
product = models.ForeignKey(
|
||||||
|
Product, related_name="scheduled_actions", on_delete=models.CASCADE
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("Product scheduled action")
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
self._meta.get_field("task").choices = get_product_actions()
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
def full_clean(self, *args, **kwargs):
|
||||||
|
self.one_off = True # A product action should occur one time only
|
||||||
|
return super().full_clean(*args, **kwargs)
|
||||||
|
|
||||||
|
def clean_clocked(self):
|
||||||
|
if not self.clocked:
|
||||||
|
raise ValidationError(_("Product actions must declare a clocked schedule."))
|
||||||
|
|
||||||
|
def validate_unique(self, *args, **kwargs):
|
||||||
|
# The checks done in PeriodicTask.validate_unique aren't
|
||||||
|
# adapted in the case of scheduled product action,
|
||||||
|
# so we skip it and execute directly Model.validate_unique
|
||||||
|
return super(PeriodicTask, self).validate_unique(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class MonthField(models.DateField):
|
||||||
|
description = _("Year + month field (day forced to 1)")
|
||||||
|
default_error_messages = {
|
||||||
|
"invalid": _(
|
||||||
|
"“%(value)s” value has an invalid date format. It must be "
|
||||||
|
"in YYYY-MM format."
|
||||||
|
),
|
||||||
|
"invalid_date": _(
|
||||||
|
"“%(value)s” value has the correct format (YYYY-MM) "
|
||||||
|
"but it is an invalid date."
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
def to_python(self, value):
|
||||||
|
if isinstance(value, str):
|
||||||
|
with contextlib.suppress(ValueError):
|
||||||
|
# If the string is given as YYYY-mm, try to parse it.
|
||||||
|
# If it fails, it means that the string may be in the form YYYY-mm-dd
|
||||||
|
# or in an invalid format.
|
||||||
|
# Whatever the case, we let Django deal with it
|
||||||
|
# and raise an error if needed
|
||||||
|
value = datetime.strptime(value, "%Y-%m")
|
||||||
|
value = super().to_python(value)
|
||||||
|
if value is None:
|
||||||
|
return None
|
||||||
|
return value.replace(day=1)
|
||||||
|
|
||||||
|
|
||||||
|
class InvoiceCall(models.Model):
|
||||||
|
is_validated = models.BooleanField(verbose_name=_("is validated"), default=False)
|
||||||
|
club = models.ForeignKey(Club, on_delete=models.CASCADE)
|
||||||
|
month = MonthField(verbose_name=_("invoice date"))
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("Invoice call")
|
||||||
|
verbose_name_plural = _("Invoice calls")
|
||||||
|
constraints = [
|
||||||
|
models.UniqueConstraint(
|
||||||
|
fields=["club", "month"], name="counter_invoicecall_unique_club_month"
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"invoice call of {self.month} made by {self.club}"
|
||||||
|
|||||||
@@ -39,6 +39,7 @@
|
|||||||
flex: auto;
|
flex: auto;
|
||||||
margin: 0.2em;
|
margin: 0.2em;
|
||||||
width: 20%;
|
width: 20%;
|
||||||
|
min-width: 350px;
|
||||||
|
|
||||||
ul {
|
ul {
|
||||||
list-style-type: none;
|
list-style-type: none;
|
||||||
|
|||||||
19
counter/tasks.py
Normal file
19
counter/tasks.py
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# Create your tasks here
|
||||||
|
|
||||||
|
from celery import shared_task
|
||||||
|
|
||||||
|
from counter.models import Counter, Product
|
||||||
|
|
||||||
|
|
||||||
|
@shared_task
|
||||||
|
def archive_product(*, product_id: int, **kwargs):
|
||||||
|
product = Product.objects.get(id=product_id)
|
||||||
|
product.archived = True
|
||||||
|
product.save()
|
||||||
|
|
||||||
|
|
||||||
|
@shared_task
|
||||||
|
def change_counters(*, product_id: int, counters: list[int], **kwargs):
|
||||||
|
product = Product.objects.get(id=product_id)
|
||||||
|
counters = Counter.objects.filter(id__in=counters)
|
||||||
|
product.counters.set(counters)
|
||||||
@@ -67,13 +67,13 @@
|
|||||||
<option value="FIN">{% trans %}Confirm (FIN){% endtrans %}</option>
|
<option value="FIN">{% trans %}Confirm (FIN){% endtrans %}</option>
|
||||||
<option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option>
|
<option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option>
|
||||||
</optgroup>
|
</optgroup>
|
||||||
{% for category in categories.keys() %}
|
{%- for category in categories.keys() -%}
|
||||||
<optgroup label="{{ category }}">
|
<optgroup label="{{ category }}">
|
||||||
{% for product in categories[category] %}
|
{%- for product in categories[category] -%}
|
||||||
<option value="{{ product.id }}">{{ product }}</option>
|
<option value="{{ product.id }}">{{ product }}</option>
|
||||||
{% endfor %}
|
{%- endfor -%}
|
||||||
</optgroup>
|
</optgroup>
|
||||||
{% endfor %}
|
{%- endfor -%}
|
||||||
</counter-product-select>
|
</counter-product-select>
|
||||||
|
|
||||||
<input type="submit" value="{% trans %}Go{% endtrans %}"/>
|
<input type="submit" value="{% trans %}Go{% endtrans %}"/>
|
||||||
|
|||||||
@@ -4,35 +4,49 @@
|
|||||||
{% trans %}Invoices call{% endtrans %}
|
{% trans %}Invoices call{% endtrans %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block notifications %}{# Notifications are moved below #}{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h3>{% trans date=start_date|date("F Y") %}Invoices call for {{ date }}{% endtrans %}</h3>
|
<h3>{% trans date=start_date|date("F Y") %}Invoices call for {{ date }}{% endtrans %}</h3>
|
||||||
<p>{% trans %}Choose another month: {% endtrans %}</p>
|
|
||||||
<form method="get" action="">
|
<form method="get" action="">
|
||||||
<select name="month">
|
<label for="id_form_other_month">{% trans %}Choose another month: {% endtrans %}</label>
|
||||||
|
<select name="month" id="id_form_other_month">
|
||||||
{% for m in months %}
|
{% for m in months %}
|
||||||
<option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option>
|
<option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</select>
|
</select>
|
||||||
<input type="submit" value="{% trans %}Go{% endtrans %}" />
|
<input type="submit" value="{% trans %}Go{% endtrans %}" />
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
<p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p>
|
<p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p>
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
|
{% include "core/base/notifications.jinja" %}
|
||||||
|
|
||||||
|
<form method="post" action="">
|
||||||
|
{% csrf_token %}
|
||||||
<table>
|
<table>
|
||||||
<thead>
|
<thead>
|
||||||
|
<tr>
|
||||||
<td>{% trans %}Club{% endtrans %}</td>
|
<td>{% trans %}Club{% endtrans %}</td>
|
||||||
<td>{% trans %}Sum{% endtrans %}</td>
|
<td>{% trans %}Sum{% endtrans %}</td>
|
||||||
|
<td>{% trans %}Validated{% endtrans %}</td>
|
||||||
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for i in sums %}
|
{% for invoice in invoices %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ i['club__name'] }}</td>
|
<td>{{ invoice.club__name }}</td>
|
||||||
<td>{{ i['selling_sum'] }} €</td>
|
<td>{{ "%.2f"|format(invoice.selling_sum) }} €</td>
|
||||||
|
<td>
|
||||||
|
{{ form[invoice.club_id|string] }}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
<input type="hidden" name="month" value="{{ start_date|date('Y-m') }}">
|
||||||
|
<button type="submit">{% trans %}Save{% endtrans %}</button>
|
||||||
|
</form>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
56
counter/templates/counter/product_form.jinja
Normal file
56
counter/templates/counter/product_form.jinja
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
{% extends "core/base.jinja" %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
{% if object %}
|
||||||
|
<h2>{% trans name=object %}Edit product {{ name }}{% endtrans %}</h2>
|
||||||
|
{% else %}
|
||||||
|
<h2>{% trans %}Product creation{% endtrans %}</h2>
|
||||||
|
{% endif %}
|
||||||
|
<form method="post" enctype="multipart/form-data">
|
||||||
|
{% csrf_token %}
|
||||||
|
{{ form.as_p() }}
|
||||||
|
|
||||||
|
<br />
|
||||||
|
|
||||||
|
<h3>{% trans %}Automatic actions{% endtrans %}</h3>
|
||||||
|
|
||||||
|
<p class="margin-bottom">
|
||||||
|
<em>
|
||||||
|
{%- trans trimmed -%}
|
||||||
|
Automatic actions allows to schedule product changes
|
||||||
|
ahead of time.
|
||||||
|
{%- endtrans -%}
|
||||||
|
</em>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{{ form.action_formset.management_form }}
|
||||||
|
{%- for action_form in form.action_formset.forms -%}
|
||||||
|
<fieldset x-data="{action: '{{ action_form.task.initial }}'}">
|
||||||
|
{{ action_form.non_field_errors() }}
|
||||||
|
<div class="row gap-2x margin-bottom">
|
||||||
|
<div>
|
||||||
|
{{ action_form.task.errors }}
|
||||||
|
{{ action_form.task.label_tag() }}
|
||||||
|
{{ action_form.task|add_attr("x-model=action") }}
|
||||||
|
</div>
|
||||||
|
<div>{{ action_form.trigger_at.as_field_group() }}</div>
|
||||||
|
</div>
|
||||||
|
<div x-show="action==='counter.tasks.change_counters'" class="margin-bottom">
|
||||||
|
{{ action_form.counters.as_field_group() }}
|
||||||
|
</div>
|
||||||
|
{%- if action_form.DELETE -%}
|
||||||
|
<div class="row gap">
|
||||||
|
{{ action_form.DELETE.as_field_group() }}
|
||||||
|
</div>
|
||||||
|
{%- endif -%}
|
||||||
|
{%- for field in action_form.hidden_fields() -%}
|
||||||
|
{{ field }}
|
||||||
|
{%- endfor -%}
|
||||||
|
</fieldset>
|
||||||
|
{%- if not loop.last -%}
|
||||||
|
<hr class="margin-bottom">
|
||||||
|
{%- endif -%}
|
||||||
|
{%- endfor -%}
|
||||||
|
<p><input type="submit" value="{% trans %}Save{% endtrans %}" /></p>
|
||||||
|
</form>
|
||||||
|
{% endblock %}
|
||||||
116
counter/tests/test_auto_actions.py
Normal file
116
counter/tests/test_auto_actions.py
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
import json
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from django.conf import settings
|
||||||
|
from django.test import Client
|
||||||
|
from django.urls import reverse
|
||||||
|
from django.utils.timezone import now
|
||||||
|
from django_celery_beat.models import ClockedSchedule
|
||||||
|
from model_bakery import baker
|
||||||
|
|
||||||
|
from core.models import Group, User
|
||||||
|
from counter.baker_recipes import counter_recipe, product_recipe
|
||||||
|
from counter.forms import ScheduledProductActionForm, ScheduledProductActionFormSet
|
||||||
|
from counter.models import ScheduledProductAction
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_edit_product(client: Client):
|
||||||
|
client.force_login(
|
||||||
|
baker.make(
|
||||||
|
User, groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
product = product_recipe.make()
|
||||||
|
url = reverse("counter:product_edit", kwargs={"product_id": product.id})
|
||||||
|
res = client.get(url)
|
||||||
|
assert res.status_code == 200
|
||||||
|
|
||||||
|
res = client.post(url, data={})
|
||||||
|
# This is actually a failure, but we just want to check that
|
||||||
|
# we don't have a 403 or a 500.
|
||||||
|
# The actual behaviour will be tested directly on the form.
|
||||||
|
assert res.status_code == 200
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
class TestProductActionForm:
|
||||||
|
def test_single_form_archive(self):
|
||||||
|
product = product_recipe.make()
|
||||||
|
trigger_at = now() + timedelta(minutes=10)
|
||||||
|
form = ScheduledProductActionForm(
|
||||||
|
product=product,
|
||||||
|
data={
|
||||||
|
"scheduled-task": "counter.tasks.archive_product",
|
||||||
|
"scheduled-trigger_at": trigger_at,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
assert form.is_valid()
|
||||||
|
instance = form.save()
|
||||||
|
assert instance.clocked.clocked_time == trigger_at
|
||||||
|
assert instance.enabled is True
|
||||||
|
assert instance.one_off is True
|
||||||
|
assert instance.task == "counter.tasks.archive_product"
|
||||||
|
assert instance.kwargs == json.dumps({"product_id": product.id})
|
||||||
|
|
||||||
|
def test_single_form_change_counters(self):
|
||||||
|
product = product_recipe.make()
|
||||||
|
counter = counter_recipe.make()
|
||||||
|
trigger_at = now() + timedelta(minutes=10)
|
||||||
|
form = ScheduledProductActionForm(
|
||||||
|
product=product,
|
||||||
|
data={
|
||||||
|
"scheduled-task": "counter.tasks.change_counters",
|
||||||
|
"scheduled-trigger_at": trigger_at,
|
||||||
|
"scheduled-counters": [counter.id],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
assert form.is_valid()
|
||||||
|
instance = form.save()
|
||||||
|
instance.refresh_from_db()
|
||||||
|
assert instance.clocked.clocked_time == trigger_at
|
||||||
|
assert instance.enabled is True
|
||||||
|
assert instance.one_off is True
|
||||||
|
assert instance.task == "counter.tasks.change_counters"
|
||||||
|
assert instance.kwargs == json.dumps(
|
||||||
|
{"product_id": product.id, "counters": [counter.id]}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
product = product_recipe.make()
|
||||||
|
clocked = baker.make(ClockedSchedule, clocked_time=now() + timedelta(minutes=2))
|
||||||
|
task = baker.make(
|
||||||
|
ScheduledProductAction,
|
||||||
|
product=product,
|
||||||
|
one_off=True,
|
||||||
|
clocked=clocked,
|
||||||
|
task="counter.tasks.archive_product",
|
||||||
|
)
|
||||||
|
formset = ScheduledProductActionFormSet(product=product)
|
||||||
|
formset.delete_existing(task)
|
||||||
|
assert not ScheduledProductAction.objects.filter(id=task.id).exists()
|
||||||
|
assert not ClockedSchedule.objects.filter(id=clocked.id).exists()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
class TestProductActionFormSet:
|
||||||
|
def test_ok(self):
|
||||||
|
product = product_recipe.make()
|
||||||
|
counter = counter_recipe.make()
|
||||||
|
trigger_at = now() + timedelta(minutes=10)
|
||||||
|
formset = ScheduledProductActionFormSet(
|
||||||
|
product=product,
|
||||||
|
data={
|
||||||
|
"form-TOTAL_FORMS": "2",
|
||||||
|
"form-INITIAL_FORMS": "0",
|
||||||
|
"form-0-task": "counter.tasks.archive_product",
|
||||||
|
"form-0-trigger_at": trigger_at,
|
||||||
|
"form-1-task": "counter.tasks.change_counters",
|
||||||
|
"form-1-trigger_at": trigger_at,
|
||||||
|
"form-1-counters": [counter.id],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
assert formset.is_valid()
|
||||||
|
formset.save()
|
||||||
|
assert ScheduledProductAction.objects.filter(product=product).count() == 2
|
||||||
@@ -355,7 +355,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
self.submit_basket(self.barmen, [BasketItem(self.beer.id, 1)])
|
self.submit_basket(self.barmen, [BasketItem(self.beer.id, 1)])
|
||||||
).status_code == 302
|
).status_code == 302
|
||||||
|
|
||||||
assert self.updated_amount(self.barmen) == Decimal("9")
|
assert self.updated_amount(self.barmen) == Decimal(9)
|
||||||
|
|
||||||
def test_click_tray_price(self):
|
def test_click_tray_price(self):
|
||||||
force_refill_user(self.customer, 20)
|
force_refill_user(self.customer, 20)
|
||||||
@@ -364,12 +364,12 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
# Not applying tray price
|
# Not applying tray price
|
||||||
res = self.submit_basket(self.customer, [BasketItem(self.beer_tap.id, 2)])
|
res = self.submit_basket(self.customer, [BasketItem(self.beer_tap.id, 2)])
|
||||||
assert res.status_code == 302
|
assert res.status_code == 302
|
||||||
assert self.updated_amount(self.customer) == Decimal("17")
|
assert self.updated_amount(self.customer) == Decimal(17)
|
||||||
|
|
||||||
# Applying tray price
|
# Applying tray price
|
||||||
res = self.submit_basket(self.customer, [BasketItem(self.beer_tap.id, 7)])
|
res = self.submit_basket(self.customer, [BasketItem(self.beer_tap.id, 7)])
|
||||||
assert res.status_code == 302
|
assert res.status_code == 302
|
||||||
assert self.updated_amount(self.customer) == Decimal("8")
|
assert self.updated_amount(self.customer) == Decimal(8)
|
||||||
|
|
||||||
def test_click_alcool_unauthorized(self):
|
def test_click_alcool_unauthorized(self):
|
||||||
self.login_in_bar()
|
self.login_in_bar()
|
||||||
@@ -381,13 +381,13 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
res = self.submit_basket(user, [BasketItem(self.snack.id, 2)])
|
res = self.submit_basket(user, [BasketItem(self.snack.id, 2)])
|
||||||
assert res.status_code == 302
|
assert res.status_code == 302
|
||||||
|
|
||||||
assert self.updated_amount(user) == Decimal("7")
|
assert self.updated_amount(user) == Decimal(7)
|
||||||
|
|
||||||
# Buy product without age limit
|
# Buy product without age limit
|
||||||
res = self.submit_basket(user, [BasketItem(self.beer.id, 2)])
|
res = self.submit_basket(user, [BasketItem(self.beer.id, 2)])
|
||||||
assert res.status_code == 200
|
assert res.status_code == 200
|
||||||
|
|
||||||
assert self.updated_amount(user) == Decimal("7")
|
assert self.updated_amount(user) == Decimal(7)
|
||||||
|
|
||||||
def test_click_unauthorized_customer(self):
|
def test_click_unauthorized_customer(self):
|
||||||
self.login_in_bar()
|
self.login_in_bar()
|
||||||
@@ -401,7 +401,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
assert resp.status_code == 302
|
assert resp.status_code == 302
|
||||||
assert resp.url == resolve_url(self.counter)
|
assert resp.url == resolve_url(self.counter)
|
||||||
|
|
||||||
assert self.updated_amount(user) == Decimal("10")
|
assert self.updated_amount(user) == Decimal(10)
|
||||||
|
|
||||||
def test_click_user_without_customer(self):
|
def test_click_user_without_customer(self):
|
||||||
self.login_in_bar()
|
self.login_in_bar()
|
||||||
@@ -418,7 +418,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
)
|
)
|
||||||
assert res.status_code == 302
|
assert res.status_code == 302
|
||||||
|
|
||||||
assert self.updated_amount(self.customer_old_can_buy) == Decimal("7")
|
assert self.updated_amount(self.customer_old_can_buy) == Decimal(7)
|
||||||
|
|
||||||
def test_click_wrong_counter(self):
|
def test_click_wrong_counter(self):
|
||||||
self.login_in_bar()
|
self.login_in_bar()
|
||||||
@@ -443,7 +443,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
)
|
)
|
||||||
assertRedirects(res, self.counter.get_absolute_url())
|
assertRedirects(res, self.counter.get_absolute_url())
|
||||||
|
|
||||||
assert self.updated_amount(self.customer) == Decimal("10")
|
assert self.updated_amount(self.customer) == Decimal(10)
|
||||||
|
|
||||||
def test_click_not_connected(self):
|
def test_click_not_connected(self):
|
||||||
force_refill_user(self.customer, 10)
|
force_refill_user(self.customer, 10)
|
||||||
@@ -455,7 +455,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
)
|
)
|
||||||
assert res.status_code == 403
|
assert res.status_code == 403
|
||||||
|
|
||||||
assert self.updated_amount(self.customer) == Decimal("10")
|
assert self.updated_amount(self.customer) == Decimal(10)
|
||||||
|
|
||||||
def test_click_product_not_in_counter(self):
|
def test_click_product_not_in_counter(self):
|
||||||
force_refill_user(self.customer, 10)
|
force_refill_user(self.customer, 10)
|
||||||
@@ -463,7 +463,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
|
|
||||||
res = self.submit_basket(self.customer, [BasketItem(self.stamps.id, 2)])
|
res = self.submit_basket(self.customer, [BasketItem(self.stamps.id, 2)])
|
||||||
assert res.status_code == 200
|
assert res.status_code == 200
|
||||||
assert self.updated_amount(self.customer) == Decimal("10")
|
assert self.updated_amount(self.customer) == Decimal(10)
|
||||||
|
|
||||||
def test_basket_empty(self):
|
def test_basket_empty(self):
|
||||||
force_refill_user(self.customer, 10)
|
force_refill_user(self.customer, 10)
|
||||||
@@ -477,7 +477,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
self.submit_basket(self.customer, basket),
|
self.submit_basket(self.customer, basket),
|
||||||
self.counter.get_absolute_url(),
|
self.counter.get_absolute_url(),
|
||||||
)
|
)
|
||||||
assert self.updated_amount(self.customer) == Decimal("10")
|
assert self.updated_amount(self.customer) == Decimal(10)
|
||||||
|
|
||||||
def test_click_product_invalid(self):
|
def test_click_product_invalid(self):
|
||||||
force_refill_user(self.customer, 10)
|
force_refill_user(self.customer, 10)
|
||||||
@@ -490,7 +490,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
BasketItem(self.beer.id, None),
|
BasketItem(self.beer.id, None),
|
||||||
]:
|
]:
|
||||||
assert self.submit_basket(self.customer, [item]).status_code == 200
|
assert self.submit_basket(self.customer, [item]).status_code == 200
|
||||||
assert self.updated_amount(self.customer) == Decimal("10")
|
assert self.updated_amount(self.customer) == Decimal(10)
|
||||||
|
|
||||||
def test_click_not_enough_money(self):
|
def test_click_not_enough_money(self):
|
||||||
force_refill_user(self.customer, 10)
|
force_refill_user(self.customer, 10)
|
||||||
@@ -501,7 +501,7 @@ class TestCounterClick(TestFullClickBase):
|
|||||||
)
|
)
|
||||||
assert res.status_code == 200
|
assert res.status_code == 200
|
||||||
|
|
||||||
assert self.updated_amount(self.customer) == Decimal("10")
|
assert self.updated_amount(self.customer) == Decimal(10)
|
||||||
|
|
||||||
def test_annotate_has_barman_queryset(self):
|
def test_annotate_has_barman_queryset(self):
|
||||||
"""Test if the custom queryset method `annotate_has_barman` works as intended."""
|
"""Test if the custom queryset method `annotate_has_barman` works as intended."""
|
||||||
|
|||||||
76
counter/tests/test_invoices.py
Normal file
76
counter/tests/test_invoices.py
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
from datetime import date, datetime
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from dateutil.relativedelta import relativedelta
|
||||||
|
from django.contrib.auth.models import Permission
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
from django.test import Client
|
||||||
|
from django.urls import reverse
|
||||||
|
from django.utils.timezone import localdate
|
||||||
|
from model_bakery import baker
|
||||||
|
from pytest_django.asserts import assertRedirects
|
||||||
|
|
||||||
|
from club.models import Club
|
||||||
|
from core.models import User
|
||||||
|
from counter.baker_recipes import sale_recipe
|
||||||
|
from counter.forms import InvoiceCallForm
|
||||||
|
from counter.models import Customer, InvoiceCall, Selling
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"month", [date(2025, 10, 20), "2025-10", datetime(2025, 10, 15, 12, 30)]
|
||||||
|
)
|
||||||
|
def test_invoice_date_with_date(month: date | datetime | str):
|
||||||
|
club = baker.make(Club)
|
||||||
|
invoice = InvoiceCall.objects.create(club=club, month=month)
|
||||||
|
invoice.refresh_from_db()
|
||||||
|
assert not invoice.is_validated
|
||||||
|
assert invoice.month == date(2025, 10, 1)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_invoice_call_invalid_month_string():
|
||||||
|
club = baker.make(Club)
|
||||||
|
|
||||||
|
with pytest.raises(ValidationError):
|
||||||
|
InvoiceCall.objects.create(club=club, month="2025-13")
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
@pytest.mark.parametrize("query", [None, {"month": "2025-08"}])
|
||||||
|
def test_invoice_call_view(client: Client, query: dict | None):
|
||||||
|
user = baker.make(
|
||||||
|
User,
|
||||||
|
user_permissions=[
|
||||||
|
*Permission.objects.filter(
|
||||||
|
codename__in=["view_invoicecall", "change_invoicecall"]
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
client.force_login(user)
|
||||||
|
url = reverse("counter:invoices_call", query=query)
|
||||||
|
assert client.get(url).status_code == 200
|
||||||
|
assertRedirects(client.post(url), url)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_invoice_call_form():
|
||||||
|
Selling.objects.all().delete()
|
||||||
|
month = localdate() - relativedelta(months=1)
|
||||||
|
clubs = baker.make(Club, _quantity=2)
|
||||||
|
recipe = sale_recipe.extend(date=month, customer=baker.make(Customer, amount=10000))
|
||||||
|
recipe.make(club=clubs[0], quantity=2, unit_price=200)
|
||||||
|
recipe.make(club=clubs[0], quantity=3, unit_price=5)
|
||||||
|
recipe.make(club=clubs[1], quantity=20, unit_price=10)
|
||||||
|
form = InvoiceCallForm(
|
||||||
|
month=month, data={str(clubs[0].id): True, str(clubs[1].id): False}
|
||||||
|
)
|
||||||
|
assert form.is_valid()
|
||||||
|
form.save()
|
||||||
|
assert InvoiceCall.objects.filter(
|
||||||
|
club=clubs[0], month=month, is_validated=True
|
||||||
|
).exists()
|
||||||
|
assert InvoiceCall.objects.filter(
|
||||||
|
club=clubs[1], month=month, is_validated=False
|
||||||
|
).exists()
|
||||||
@@ -6,14 +6,16 @@ import pytest
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.cache import cache
|
from django.core.cache import cache
|
||||||
from django.core.files.uploadedfile import SimpleUploadedFile
|
from django.core.files.uploadedfile import SimpleUploadedFile
|
||||||
from django.test import Client
|
from django.test import Client, TestCase
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from model_bakery import baker
|
from model_bakery import baker
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
from pytest_django.asserts import assertNumQueries
|
from pytest_django.asserts import assertNumQueries, assertRedirects
|
||||||
|
|
||||||
|
from club.models import Club
|
||||||
from core.baker_recipes import board_user, subscriber_user
|
from core.baker_recipes import board_user, subscriber_user
|
||||||
from core.models import Group, User
|
from core.models import Group, User
|
||||||
|
from counter.forms import ProductForm
|
||||||
from counter.models import Product, ProductType
|
from counter.models import Product, ProductType
|
||||||
|
|
||||||
|
|
||||||
@@ -84,3 +86,49 @@ def test_fetch_product_nb_queries(client: Client):
|
|||||||
# - 1 for the actual request
|
# - 1 for the actual request
|
||||||
# - 1 to prefetch the related buying_groups
|
# - 1 to prefetch the related buying_groups
|
||||||
client.get(reverse("api:search_products_detailed"))
|
client.get(reverse("api:search_products_detailed"))
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateProduct(TestCase):
|
||||||
|
@classmethod
|
||||||
|
def setUpTestData(cls):
|
||||||
|
cls.product_type = baker.make(ProductType)
|
||||||
|
cls.club = baker.make(Club)
|
||||||
|
cls.data = {
|
||||||
|
"name": "foo",
|
||||||
|
"description": "bar",
|
||||||
|
"product_type": cls.product_type.id,
|
||||||
|
"club": cls.club.id,
|
||||||
|
"code": "FOO",
|
||||||
|
"purchase_price": 1.0,
|
||||||
|
"selling_price": 1.0,
|
||||||
|
"special_selling_price": 1.0,
|
||||||
|
"limit_age": 0,
|
||||||
|
"form-TOTAL_FORMS": 0,
|
||||||
|
"form-INITIAL_FORMS": 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
def test_form(self):
|
||||||
|
form = ProductForm(data=self.data)
|
||||||
|
assert form.is_valid()
|
||||||
|
instance = form.save()
|
||||||
|
assert instance.club == self.club
|
||||||
|
assert instance.product_type == self.product_type
|
||||||
|
assert instance.name == "foo"
|
||||||
|
assert instance.selling_price == 1.0
|
||||||
|
|
||||||
|
def test_view(self):
|
||||||
|
self.client.force_login(
|
||||||
|
baker.make(
|
||||||
|
User,
|
||||||
|
groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)],
|
||||||
|
)
|
||||||
|
)
|
||||||
|
url = reverse("counter:new_product")
|
||||||
|
response = self.client.get(url)
|
||||||
|
assert response.status_code == 200
|
||||||
|
response = self.client.post(url, data=self.data)
|
||||||
|
assertRedirects(response, reverse("counter:product_list"))
|
||||||
|
product = Product.objects.last()
|
||||||
|
assert product.name == "foo"
|
||||||
|
assert product.club == self.club
|
||||||
|
assert product.product_type == self.product_type
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ from core.utils import get_semester_code, get_start_of_semester
|
|||||||
from counter.forms import (
|
from counter.forms import (
|
||||||
CloseCustomerAccountForm,
|
CloseCustomerAccountForm,
|
||||||
CounterEditForm,
|
CounterEditForm,
|
||||||
ProductEditForm,
|
ProductForm,
|
||||||
ReturnableProductForm,
|
ReturnableProductForm,
|
||||||
)
|
)
|
||||||
from counter.models import (
|
from counter.models import (
|
||||||
@@ -146,8 +146,8 @@ class ProductCreateView(CounterAdminTabsMixin, CounterAdminMixin, CreateView):
|
|||||||
"""A create view for the admins."""
|
"""A create view for the admins."""
|
||||||
|
|
||||||
model = Product
|
model = Product
|
||||||
form_class = ProductEditForm
|
form_class = ProductForm
|
||||||
template_name = "core/create.jinja"
|
template_name = "counter/product_form.jinja"
|
||||||
current_tab = "products"
|
current_tab = "products"
|
||||||
|
|
||||||
|
|
||||||
@@ -155,9 +155,9 @@ class ProductEditView(CounterAdminTabsMixin, CounterAdminMixin, UpdateView):
|
|||||||
"""An edit view for the admins."""
|
"""An edit view for the admins."""
|
||||||
|
|
||||||
model = Product
|
model = Product
|
||||||
form_class = ProductEditForm
|
form_class = ProductForm
|
||||||
pk_url_kwarg = "product_id"
|
pk_url_kwarg = "product_id"
|
||||||
template_name = "core/edit.jinja"
|
template_name = "counter/product_form.jinja"
|
||||||
current_tab = "products"
|
current_tab = "products"
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -12,77 +12,81 @@
|
|||||||
# OR WITHIN THE LOCAL FILE "LICENSE"
|
# OR WITHIN THE LOCAL FILE "LICENSE"
|
||||||
#
|
#
|
||||||
#
|
#
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime
|
||||||
from datetime import timezone as tz
|
from urllib.parse import urlencode
|
||||||
|
|
||||||
from django.db.models import F
|
from dateutil.relativedelta import relativedelta
|
||||||
from django.utils import timezone
|
from django.contrib.auth.mixins import PermissionRequiredMixin
|
||||||
from django.views.generic import TemplateView
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.db.models import F, Sum
|
||||||
|
from django.utils.timezone import localdate, make_aware
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from django.views.generic import FormView
|
||||||
|
|
||||||
from counter.fields import CurrencyField
|
from counter.forms import InvoiceCallForm
|
||||||
from counter.models import Refilling, Selling
|
from counter.models import Refilling, Selling
|
||||||
from counter.views.mixins import CounterAdminMixin, CounterAdminTabsMixin
|
from counter.views.mixins import CounterAdminTabsMixin
|
||||||
|
|
||||||
|
|
||||||
class InvoiceCallView(CounterAdminTabsMixin, CounterAdminMixin, TemplateView):
|
class InvoiceCallView(
|
||||||
|
CounterAdminTabsMixin, PermissionRequiredMixin, SuccessMessageMixin, FormView
|
||||||
|
):
|
||||||
template_name = "counter/invoices_call.jinja"
|
template_name = "counter/invoices_call.jinja"
|
||||||
current_tab = "invoices_call"
|
current_tab = "invoices_call"
|
||||||
|
permission_required = ["counter.view_invoicecall", "counter.change_invoicecall"]
|
||||||
|
form_class = InvoiceCallForm
|
||||||
|
success_message = _("Invoice calls status has been updated.")
|
||||||
|
|
||||||
|
def get_month(self):
|
||||||
|
kwargs = self.request.GET or self.request.POST
|
||||||
|
if "month" in kwargs:
|
||||||
|
return make_aware(datetime.strptime(kwargs["month"], "%Y-%m"))
|
||||||
|
return localdate().replace(day=1) - relativedelta(months=1)
|
||||||
|
|
||||||
|
def get_form_kwargs(self):
|
||||||
|
return super().get_form_kwargs() | {"month": self.get_month()}
|
||||||
|
|
||||||
|
def form_valid(self, form):
|
||||||
|
form.save()
|
||||||
|
return super().form_valid(form)
|
||||||
|
|
||||||
|
def get_success_url(self):
|
||||||
|
# redirect to the month from which the request is originated
|
||||||
|
url = self.request.path
|
||||||
|
kwargs = self.request.GET or self.request.POST
|
||||||
|
if "month" in kwargs:
|
||||||
|
query = urlencode({"month": kwargs["month"]})
|
||||||
|
url += f"?{query}"
|
||||||
|
return url
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
"""Add sums to the context."""
|
"""Add sums to the context."""
|
||||||
kwargs = super().get_context_data(**kwargs)
|
kwargs = super().get_context_data(**kwargs)
|
||||||
kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC")
|
kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC")
|
||||||
if "month" in self.request.GET:
|
start_date = self.get_month()
|
||||||
start_date = datetime.strptime(self.request.GET["month"], "%Y-%m")
|
end_date = start_date + relativedelta(months=1)
|
||||||
else:
|
|
||||||
start_date = datetime(
|
|
||||||
year=timezone.now().year,
|
|
||||||
month=(timezone.now().month + 10) % 12 + 1,
|
|
||||||
day=1,
|
|
||||||
)
|
|
||||||
start_date = start_date.replace(tzinfo=tz.utc)
|
|
||||||
end_date = (start_date + timedelta(days=32)).replace(
|
|
||||||
day=1, hour=0, minute=0, microsecond=0
|
|
||||||
)
|
|
||||||
from django.db.models import Case, Sum, When
|
|
||||||
|
|
||||||
kwargs["sum_cb"] = sum(
|
kwargs["sum_cb"] = Refilling.objects.filter(
|
||||||
[
|
payment_method="CARD",
|
||||||
r.amount
|
is_validated=True,
|
||||||
for r in Refilling.objects.filter(
|
date__gte=start_date,
|
||||||
|
date__lte=end_date,
|
||||||
|
).aggregate(res=Sum("amount", default=0))["res"]
|
||||||
|
kwargs["sum_cb"] += (
|
||||||
|
Selling.objects.filter(
|
||||||
payment_method="CARD",
|
payment_method="CARD",
|
||||||
is_validated=True,
|
is_validated=True,
|
||||||
date__gte=start_date,
|
date__gte=start_date,
|
||||||
date__lte=end_date,
|
date__lte=end_date,
|
||||||
)
|
)
|
||||||
]
|
.annotate(amount=F("unit_price") * F("quantity"))
|
||||||
)
|
.aggregate(res=Sum("amount", default=0))["res"]
|
||||||
kwargs["sum_cb"] += sum(
|
|
||||||
[
|
|
||||||
s.quantity * s.unit_price
|
|
||||||
for s in Selling.objects.filter(
|
|
||||||
payment_method="CARD",
|
|
||||||
is_validated=True,
|
|
||||||
date__gte=start_date,
|
|
||||||
date__lte=end_date,
|
|
||||||
)
|
|
||||||
]
|
|
||||||
)
|
)
|
||||||
kwargs["start_date"] = start_date
|
kwargs["start_date"] = start_date
|
||||||
kwargs["sums"] = (
|
kwargs["invoices"] = (
|
||||||
Selling.objects.values("club__name")
|
Selling.objects.filter(date__gte=start_date, date__lt=end_date)
|
||||||
.annotate(
|
.values("club_id", "club__name")
|
||||||
selling_sum=Sum(
|
.annotate(selling_sum=Sum(F("unit_price") * F("quantity")))
|
||||||
Case(
|
|
||||||
When(
|
|
||||||
date__gte=start_date,
|
|
||||||
date__lt=end_date,
|
|
||||||
then=F("unit_price") * F("quantity"),
|
|
||||||
),
|
|
||||||
output_field=CurrencyField(),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
.exclude(selling_sum=None)
|
.exclude(selling_sum=None)
|
||||||
.order_by("-selling_sum")
|
.order_by("-selling_sum")
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -242,7 +242,7 @@ class Invoice(models.Model):
|
|||||||
def validate(self):
|
def validate(self):
|
||||||
if self.validated:
|
if self.validated:
|
||||||
raise DataError(_("Invoice already validated"))
|
raise DataError(_("Invoice already validated"))
|
||||||
customer, created = Customer.get_or_create(user=self.user)
|
customer, _created = Customer.get_or_create(user=self.user)
|
||||||
eboutic = Counter.objects.filter(type="EBOUTIC").first()
|
eboutic = Counter.objects.filter(type="EBOUTIC").first()
|
||||||
for i in self.items.all():
|
for i in self.items.all():
|
||||||
if i.type_id == settings.SITH_COUNTER_PRODUCTTYPE_REFILLING:
|
if i.type_id == settings.SITH_COUNTER_PRODUCTTYPE_REFILLING:
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
{% from 'core/macros.jinja' import update_notifications %}
|
||||||
|
|
||||||
<div id=billing-infos-fragment>
|
<div id=billing-infos-fragment>
|
||||||
<div
|
<div
|
||||||
class="collapse"
|
class="collapse"
|
||||||
@@ -29,7 +31,6 @@
|
|||||||
>
|
>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
{% include "core/base/notifications.jinja" %}
|
{{ update_notifications(messages) }}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{% extends "core/base.jinja" %}
|
{% extends "core/base.jinja" %}
|
||||||
|
|
||||||
{% block notifications %}
|
{% block notifications %}
|
||||||
{# Notifications are moved inside the billing info fragment #}
|
{# Notifications are moved under the billing form #}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block title %}
|
{% block title %}
|
||||||
@@ -60,6 +60,7 @@
|
|||||||
<div @htmx:after-request="fill">
|
<div @htmx:after-request="fill">
|
||||||
{{ billing_infos_form }}
|
{{ billing_infos_form }}
|
||||||
</div>
|
</div>
|
||||||
|
{% include "core/base/notifications.jinja" %}
|
||||||
<form
|
<form
|
||||||
method="post"
|
method="post"
|
||||||
action="{{ settings.SITH_EBOUTIC_ET_URL }}"
|
action="{{ settings.SITH_EBOUTIC_ET_URL }}"
|
||||||
|
|||||||
@@ -108,7 +108,7 @@ class TestPaymentSith(TestPaymentBase):
|
|||||||
)
|
)
|
||||||
assert Basket.objects.filter(id=self.basket.id).first() is None
|
assert Basket.objects.filter(id=self.basket.id).first() is None
|
||||||
self.customer.customer.refresh_from_db()
|
self.customer.customer.refresh_from_db()
|
||||||
assert self.customer.customer.amount == Decimal("1")
|
assert self.customer.customer.amount == Decimal(1)
|
||||||
|
|
||||||
sellings = Selling.objects.filter(customer=self.customer.customer).order_by(
|
sellings = Selling.objects.filter(customer=self.customer.customer).order_by(
|
||||||
"quantity"
|
"quantity"
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ from django_countries.fields import Country
|
|||||||
|
|
||||||
from core.auth.mixins import CanViewMixin
|
from core.auth.mixins import CanViewMixin
|
||||||
from core.views.mixins import FragmentMixin, UseFragmentsMixin
|
from core.views.mixins import FragmentMixin, UseFragmentsMixin
|
||||||
from counter.forms import BaseBasketForm, BillingInfoForm, ProductForm
|
from counter.forms import BaseBasketForm, BasketProductForm, BillingInfoForm
|
||||||
from counter.models import (
|
from counter.models import (
|
||||||
BillingInfo,
|
BillingInfo,
|
||||||
Customer,
|
Customer,
|
||||||
@@ -78,7 +78,7 @@ class BaseEbouticBasketForm(BaseBasketForm):
|
|||||||
|
|
||||||
|
|
||||||
EbouticBasketForm = forms.formset_factory(
|
EbouticBasketForm = forms.formset_factory(
|
||||||
ProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1
|
BasketProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -45,8 +45,9 @@ class Command(BaseCommand):
|
|||||||
"verbosity level should be between 0 and 2 included", stacklevel=2
|
"verbosity level should be between 0 and 2 included", stacklevel=2
|
||||||
)
|
)
|
||||||
|
|
||||||
if options["verbosity"] == 2:
|
if options["verbosity"] >= 2:
|
||||||
logger.setLevel(logging.DEBUG)
|
logger.setLevel(logging.DEBUG)
|
||||||
|
logging.getLogger("django.db.backends").setLevel(logging.DEBUG)
|
||||||
elif options["verbosity"] == 1:
|
elif options["verbosity"] == 1:
|
||||||
logger.setLevel(logging.INFO)
|
logger.setLevel(logging.INFO)
|
||||||
else:
|
else:
|
||||||
@@ -59,6 +60,3 @@ class Command(BaseCommand):
|
|||||||
Galaxy.objects.filter(state__isnull=True).delete()
|
Galaxy.objects.filter(state__isnull=True).delete()
|
||||||
|
|
||||||
logger.info("Ruled the galaxy in {} queries.".format(len(connection.queries)))
|
logger.info("Ruled the galaxy in {} queries.".format(len(connection.queries)))
|
||||||
if options["verbosity"] > 2:
|
|
||||||
for q in connection.queries:
|
|
||||||
logger.debug(q)
|
|
||||||
|
|||||||
@@ -31,13 +31,14 @@ from collections import defaultdict
|
|||||||
from typing import NamedTuple, TypedDict
|
from typing import NamedTuple, TypedDict
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.db.models import Count, F, Q, QuerySet
|
from django.db.models import Count, Exists, F, OuterRef, Q, QuerySet
|
||||||
from django.utils.timezone import localdate
|
from django.utils.timezone import localdate, now
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from club.models import Membership
|
from club.models import Membership
|
||||||
from core.models import User
|
from core.models import User
|
||||||
from sas.models import PeoplePictureRelation, Picture
|
from sas.models import PeoplePictureRelation, Picture
|
||||||
|
from subscription.models import Subscription
|
||||||
|
|
||||||
|
|
||||||
class GalaxyStar(models.Model):
|
class GalaxyStar(models.Model):
|
||||||
@@ -198,8 +199,16 @@ class Galaxy(models.Model):
|
|||||||
cls, picture_count_threshold: int = DEFAULT_PICTURE_COUNT_THRESHOLD
|
cls, picture_count_threshold: int = DEFAULT_PICTURE_COUNT_THRESHOLD
|
||||||
) -> QuerySet[User]:
|
) -> QuerySet[User]:
|
||||||
return (
|
return (
|
||||||
User.objects.exclude(subscriptions=None)
|
User.objects.filter(is_subscriber_viewable=True)
|
||||||
.annotate(pictures_count=Count("pictures"))
|
.exclude(subscriptions=None)
|
||||||
|
.annotate(
|
||||||
|
pictures_count=Count("pictures"),
|
||||||
|
is_active_in_galaxy=Exists(
|
||||||
|
Subscription.objects.filter(
|
||||||
|
member=OuterRef("id"), subscription_end__gt=now()
|
||||||
|
)
|
||||||
|
),
|
||||||
|
)
|
||||||
.filter(pictures_count__gt=picture_count_threshold)
|
.filter(pictures_count__gt=picture_count_threshold)
|
||||||
.distinct()
|
.distinct()
|
||||||
)
|
)
|
||||||
@@ -290,9 +299,9 @@ class Galaxy(models.Model):
|
|||||||
31/12/2022 (also two years, but with an offset of one year), then their
|
31/12/2022 (also two years, but with an offset of one year), then their
|
||||||
club score is 365.
|
club score is 365.
|
||||||
"""
|
"""
|
||||||
memberships = user.memberships.only("start_date", "end_date", "club_id")
|
memberships = user.memberships.values("start_date", "end_date", "club_id")
|
||||||
result = defaultdict(int)
|
result = defaultdict(int)
|
||||||
now = localdate()
|
today = localdate()
|
||||||
for membership in memberships:
|
for membership in memberships:
|
||||||
# This is a N+1 query, but 92% of galaxy users have less than 10 memberships.
|
# This is a N+1 query, but 92% of galaxy users have less than 10 memberships.
|
||||||
# Only 5 users have more than 30 memberships.
|
# Only 5 users have more than 30 memberships.
|
||||||
@@ -300,23 +309,23 @@ class Galaxy(models.Model):
|
|||||||
Membership.objects.exclude(user=user)
|
Membership.objects.exclude(user=user)
|
||||||
.filter(
|
.filter(
|
||||||
Q( # start2 <= start1 <= end2
|
Q( # start2 <= start1 <= end2
|
||||||
start_date__lte=membership.start_date,
|
start_date__lte=membership["start_date"],
|
||||||
end_date__gte=membership.start_date,
|
end_date__gte=membership["start_date"],
|
||||||
)
|
)
|
||||||
| Q( # start2 <= start1 <= now
|
| Q( # start2 <= start1 <= today
|
||||||
start_date__lte=membership.start_date, end_date=None
|
start_date__lte=membership["start_date"], end_date=None
|
||||||
)
|
)
|
||||||
| Q( # start1 <= start2 <= end2
|
| Q( # start1 <= start2 <= end2
|
||||||
start_date__gte=membership.start_date,
|
start_date__gte=membership["start_date"],
|
||||||
start_date__lte=membership.end_date or now,
|
start_date__lte=membership["end_date"] or today,
|
||||||
),
|
),
|
||||||
club_id=membership.club_id,
|
club_id=membership["club_id"],
|
||||||
)
|
)
|
||||||
.only("start_date", "end_date", "user_id")
|
.only("start_date", "end_date", "user_id")
|
||||||
)
|
)
|
||||||
for other in common_memberships:
|
for other in common_memberships:
|
||||||
start = max(membership.start_date, other.start_date)
|
start = max(membership["start_date"], other.start_date)
|
||||||
end = min(membership.end_date or now, other.end_date or now)
|
end = min(membership["end_date"] or today, other.end_date or today)
|
||||||
result[other.user_id] += (end - start).days * cls.CLUBS_POINTS
|
result[other.user_id] += (end - start).days * cls.CLUBS_POINTS
|
||||||
return result
|
return result
|
||||||
|
|
||||||
@@ -382,18 +391,22 @@ class Galaxy(models.Model):
|
|||||||
# this is memory expensive but prevents a lot of db hits, therefore
|
# this is memory expensive but prevents a lot of db hits, therefore
|
||||||
# is far more time efficient
|
# is far more time efficient
|
||||||
|
|
||||||
rulable_users = list(self.get_rulable_users(picture_count_threshold))
|
rulable_users_qs = self.get_rulable_users(picture_count_threshold)
|
||||||
rulable_users_count = len(rulable_users)
|
active_users_count = rulable_users_qs.filter(is_active_in_galaxy=True).count()
|
||||||
|
rulable_users = list(rulable_users_qs)
|
||||||
user1_count = 0
|
user1_count = 0
|
||||||
self.logger.info(
|
self.logger.info(
|
||||||
f"{rulable_users_count} citizen have been listed. Starting to rule."
|
f" {len(rulable_users)} citizens (with {active_users_count} active ones) "
|
||||||
|
f"have been listed. Starting to rule."
|
||||||
)
|
)
|
||||||
|
|
||||||
self.logger.info("Creating stars for all citizen")
|
self.logger.info("Creating stars for all citizen")
|
||||||
individual_scores = self.compute_individual_scores()
|
individual_scores = self.compute_individual_scores()
|
||||||
GalaxyStar.objects.bulk_create(
|
GalaxyStar.objects.bulk_create(
|
||||||
[
|
[
|
||||||
GalaxyStar(owner=user, galaxy=self, mass=individual_scores[user.id])
|
GalaxyStar(
|
||||||
|
owner_id=user.id, galaxy=self, mass=individual_scores[user.id]
|
||||||
|
)
|
||||||
for user in rulable_users
|
for user in rulable_users
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
@@ -405,9 +418,9 @@ class Galaxy(models.Model):
|
|||||||
t_global_start = time.time()
|
t_global_start = time.time()
|
||||||
while len(rulable_users) > 0:
|
while len(rulable_users) > 0:
|
||||||
user1 = rulable_users.pop()
|
user1 = rulable_users.pop()
|
||||||
|
if not user1.is_active_in_galaxy:
|
||||||
|
continue
|
||||||
user1_count += 1
|
user1_count += 1
|
||||||
rulable_users_count2 = len(rulable_users)
|
|
||||||
|
|
||||||
star1 = stars[user1.id]
|
star1 = stars[user1.id]
|
||||||
|
|
||||||
lanes = []
|
lanes = []
|
||||||
@@ -448,17 +461,20 @@ class Galaxy(models.Model):
|
|||||||
self.logger.info("")
|
self.logger.info("")
|
||||||
self.logger.info(f" Ruling of {self} ".center(60, "#"))
|
self.logger.info(f" Ruling of {self} ".center(60, "#"))
|
||||||
self.logger.info(
|
self.logger.info(
|
||||||
f"Progression: {user1_count}/{rulable_users_count} "
|
f"Progression: {user1_count}/{active_users_count} "
|
||||||
f"citizen -- {rulable_users_count - user1_count} remaining"
|
f"citizen -- {active_users_count - user1_count} remaining"
|
||||||
)
|
)
|
||||||
self.logger.info(f"Speed: {global_avg_speed:.2f} citizen per second")
|
self.logger.info(f"Speed: {global_avg_speed:.2f} citizen per second")
|
||||||
eta = rulable_users_count2 // global_avg_speed
|
eta = len(rulable_users) // global_avg_speed
|
||||||
self.logger.info(
|
self.logger.info(
|
||||||
f"ETA: {int(eta // 60 % 60)} minutes {int(eta % 60)} seconds"
|
f"ETA: {int(eta // 60 % 60)} minutes {int(eta % 60)} seconds"
|
||||||
)
|
)
|
||||||
self.logger.info("#" * 60)
|
self.logger.info("#" * 60)
|
||||||
t_global_start = time.time()
|
t_global_start = time.time()
|
||||||
|
|
||||||
|
count, _ = self.stars.filter(Q(lanes1=None) & Q(lanes2=None)).delete()
|
||||||
|
self.logger.info(f"{count} orphan stars have been trimmed.")
|
||||||
|
|
||||||
# Here, we get the IDs of the old galaxies that we'll need to delete. In normal operation, only one galaxy
|
# Here, we get the IDs of the old galaxies that we'll need to delete. In normal operation, only one galaxy
|
||||||
# should be returned, and we can't delete it yet, as it's the one still displayed by the Sith.
|
# should be returned, and we can't delete it yet, as it's the one still displayed by the Sith.
|
||||||
old_galaxies_pks = list(
|
old_galaxies_pks = list(
|
||||||
|
|||||||
@@ -122,7 +122,7 @@ class TestGalaxyModel(TestCase):
|
|||||||
self.com,
|
self.com,
|
||||||
]
|
]
|
||||||
|
|
||||||
with self.assertNumQueries(44):
|
with self.assertNumQueries(38):
|
||||||
while len(users) > 0:
|
while len(users) > 0:
|
||||||
user1 = users.pop(0)
|
user1 = users.pop(0)
|
||||||
family_scores = Galaxy.compute_user_family_score(user1)
|
family_scores = Galaxy.compute_user_family_score(user1)
|
||||||
@@ -150,7 +150,7 @@ class TestGalaxyModel(TestCase):
|
|||||||
that the number of queries to rule the galaxy is stable.
|
that the number of queries to rule the galaxy is stable.
|
||||||
"""
|
"""
|
||||||
galaxy = Galaxy.objects.create()
|
galaxy = Galaxy.objects.create()
|
||||||
with self.assertNumQueries(39):
|
with self.assertNumQueries(36):
|
||||||
galaxy.rule(0) # We want everybody here
|
galaxy.rule(0) # We want everybody here
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2025-09-26 17:36+0200\n"
|
"POT-Creation-Date: 2025-11-07 14:50+0100\n"
|
||||||
"PO-Revision-Date: 2016-07-18\n"
|
"PO-Revision-Date: 2016-07-18\n"
|
||||||
"Last-Translator: Maréchal <thomas.girod@utbm.fr\n"
|
"Last-Translator: Maréchal <thomas.girod@utbm.fr\n"
|
||||||
"Language-Team: AE info <ae.info@utbm.fr>\n"
|
"Language-Team: AE info <ae.info@utbm.fr>\n"
|
||||||
@@ -117,7 +117,7 @@ msgstr "S'abonner"
|
|||||||
msgid "Remove"
|
msgid "Remove"
|
||||||
msgstr "Retirer"
|
msgstr "Retirer"
|
||||||
|
|
||||||
#: club/forms.py pedagogy/templates/pedagogy/moderation.jinja
|
#: club/forms.py counter/forms.py pedagogy/templates/pedagogy/moderation.jinja
|
||||||
msgid "Action"
|
msgid "Action"
|
||||||
msgstr "Action"
|
msgstr "Action"
|
||||||
|
|
||||||
@@ -146,7 +146,7 @@ msgstr "Date de début"
|
|||||||
msgid "End date"
|
msgid "End date"
|
||||||
msgstr "Date de fin"
|
msgstr "Date de fin"
|
||||||
|
|
||||||
#: club/forms.py club/templates/club/club_sellings.jinja
|
#: club/forms.py club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: counter/templates/counter/cash_summary_list.jinja counter/views/mixins.py
|
#: counter/templates/counter/cash_summary_list.jinja counter/views/mixins.py
|
||||||
msgid "Counter"
|
msgid "Counter"
|
||||||
@@ -409,7 +409,7 @@ msgstr "Total : "
|
|||||||
msgid "Benefit: "
|
msgid "Benefit: "
|
||||||
msgstr "Bénéfice : "
|
msgstr "Bénéfice : "
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: counter/templates/counter/cash_summary_list.jinja
|
#: counter/templates/counter/cash_summary_list.jinja
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
@@ -419,34 +419,34 @@ msgstr "Bénéfice : "
|
|||||||
msgid "Date"
|
msgid "Date"
|
||||||
msgstr "Date"
|
msgstr "Date"
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
msgid "Barman"
|
msgid "Barman"
|
||||||
msgstr "Barman"
|
msgstr "Barman"
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: counter/templates/counter/counter_click.jinja
|
#: counter/templates/counter/counter_click.jinja
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
#: counter/templates/counter/refilling_list.jinja
|
#: counter/templates/counter/refilling_list.jinja
|
||||||
msgid "Customer"
|
msgid "Customer"
|
||||||
msgstr "Client"
|
msgstr "Client"
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
#: rootplace/templates/rootplace/logs.jinja
|
#: rootplace/templates/rootplace/logs.jinja
|
||||||
msgid "Label"
|
msgid "Label"
|
||||||
msgstr "Étiquette"
|
msgstr "Étiquette"
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: core/templates/core/user_stats.jinja
|
#: core/templates/core/user_stats.jinja
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
msgid "Quantity"
|
msgid "Quantity"
|
||||||
msgstr "Quantité"
|
msgstr "Quantité"
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account.jinja
|
#: core/templates/core/user_account.jinja
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: counter/templates/counter/cash_summary_list.jinja
|
#: counter/templates/counter/cash_summary_list.jinja
|
||||||
@@ -456,7 +456,7 @@ msgstr "Quantité"
|
|||||||
msgid "Total"
|
msgid "Total"
|
||||||
msgstr "Total"
|
msgstr "Total"
|
||||||
|
|
||||||
#: club/templates/club/club_sellings.jinja
|
#: club/templates/club/club_sellings.jinja club/views.py
|
||||||
#: core/templates/core/user_account_detail.jinja
|
#: core/templates/core/user_account_detail.jinja
|
||||||
#: core/templates/core/user_detail.jinja
|
#: core/templates/core/user_detail.jinja
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
@@ -556,6 +556,8 @@ msgstr ""
|
|||||||
#: core/templates/core/user_godfathers_tree.jinja
|
#: core/templates/core/user_godfathers_tree.jinja
|
||||||
#: core/templates/core/user_preferences.jinja
|
#: core/templates/core/user_preferences.jinja
|
||||||
#: counter/templates/counter/cash_register_summary.jinja
|
#: counter/templates/counter/cash_register_summary.jinja
|
||||||
|
#: counter/templates/counter/invoices_call.jinja
|
||||||
|
#: counter/templates/counter/product_form.jinja
|
||||||
#: forum/templates/forum/reply.jinja
|
#: forum/templates/forum/reply.jinja
|
||||||
#: subscription/templates/subscription/fragments/creation_form.jinja
|
#: subscription/templates/subscription/fragments/creation_form.jinja
|
||||||
#: trombi/templates/trombi/comment.jinja
|
#: trombi/templates/trombi/comment.jinja
|
||||||
@@ -688,14 +690,26 @@ msgstr "Vente"
|
|||||||
msgid "Mailing list"
|
msgid "Mailing list"
|
||||||
msgstr "Listes de diffusion"
|
msgstr "Listes de diffusion"
|
||||||
|
|
||||||
|
#: club/views.py
|
||||||
|
msgid "You are now a member of this club."
|
||||||
|
msgstr "Vous êtes maintenant membre de ce club."
|
||||||
|
|
||||||
#: club/views.py
|
#: club/views.py
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%(user)s has been added to club."
|
msgid "%(user)s has been added to club."
|
||||||
msgstr "%(user)s a été ajouté au club."
|
msgstr "%(user)s a été ajouté au club."
|
||||||
|
|
||||||
#: club/views.py
|
#: club/views.py
|
||||||
msgid "You are now a member of this club."
|
msgid "Benefit"
|
||||||
msgstr "Vous êtes maintenant membre de ce club."
|
msgstr "Bénéfice"
|
||||||
|
|
||||||
|
#: club/views.py
|
||||||
|
msgid "Selling price"
|
||||||
|
msgstr "Prix de vente"
|
||||||
|
|
||||||
|
#: club/views.py
|
||||||
|
msgid "Purchase price"
|
||||||
|
msgstr "Prix d'achat"
|
||||||
|
|
||||||
#: com/forms.py
|
#: com/forms.py
|
||||||
msgid "Format: 16:9 | Resolution: 1920x1080"
|
msgid "Format: 16:9 | Resolution: 1920x1080"
|
||||||
@@ -891,7 +905,8 @@ msgstr "Administration des mailing listes"
|
|||||||
msgid "Actions"
|
msgid "Actions"
|
||||||
msgstr "Actions"
|
msgstr "Actions"
|
||||||
|
|
||||||
#: com/templates/com/mailing_admin.jinja core/templates/core/file_detail.jinja
|
#: com/templates/com/mailing_admin.jinja com/templates/com/poster_list.jinja
|
||||||
|
#: core/templates/core/file_detail.jinja
|
||||||
#: core/templates/core/file_moderation.jinja sas/templates/sas/moderation.jinja
|
#: core/templates/core/file_moderation.jinja sas/templates/sas/moderation.jinja
|
||||||
#: sas/templates/sas/picture.jinja
|
#: sas/templates/sas/picture.jinja
|
||||||
msgid "Moderate"
|
msgid "Moderate"
|
||||||
@@ -1028,7 +1043,7 @@ msgstr "Événements aujourd'hui et dans les prochains jours"
|
|||||||
|
|
||||||
#: com/templates/com/news_list.jinja
|
#: com/templates/com/news_list.jinja
|
||||||
msgid "Administrate news"
|
msgid "Administrate news"
|
||||||
msgstr "Administrer les news"
|
msgstr "Administrer les nouvelles"
|
||||||
|
|
||||||
#: com/templates/com/news_list.jinja
|
#: com/templates/com/news_list.jinja
|
||||||
msgid "Nothing to come..."
|
msgid "Nothing to come..."
|
||||||
@@ -1061,6 +1076,10 @@ msgstr "Nos services"
|
|||||||
msgid "UV Guide"
|
msgid "UV Guide"
|
||||||
msgstr "Guide des UVs"
|
msgstr "Guide des UVs"
|
||||||
|
|
||||||
|
#: com/templates/com/news_list.jinja
|
||||||
|
msgid "Timetable"
|
||||||
|
msgstr "Emploi du temps"
|
||||||
|
|
||||||
#: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja
|
#: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja
|
||||||
msgid "Matmatronch"
|
msgid "Matmatronch"
|
||||||
msgstr "Matmatronch"
|
msgstr "Matmatronch"
|
||||||
@@ -1103,8 +1122,7 @@ msgstr "Vous n'avez pas accès à ce contenu"
|
|||||||
msgid "Poster"
|
msgid "Poster"
|
||||||
msgstr "Affiche"
|
msgstr "Affiche"
|
||||||
|
|
||||||
#: com/templates/com/poster_edit.jinja com/templates/com/poster_moderate.jinja
|
#: com/templates/com/poster_edit.jinja com/templates/com/screen_edit.jinja
|
||||||
#: com/templates/com/screen_edit.jinja
|
|
||||||
msgid "List"
|
msgid "List"
|
||||||
msgstr "Liste"
|
msgstr "Liste"
|
||||||
|
|
||||||
@@ -1117,25 +1135,13 @@ msgstr "Affiche - modifier"
|
|||||||
msgid "Create"
|
msgid "Create"
|
||||||
msgstr "Créer"
|
msgstr "Créer"
|
||||||
|
|
||||||
#: com/templates/com/poster_list.jinja
|
|
||||||
msgid "Moderation"
|
|
||||||
msgstr "Modération"
|
|
||||||
|
|
||||||
#: com/templates/com/poster_list.jinja
|
|
||||||
msgid "No posters"
|
|
||||||
msgstr "Aucune affiche"
|
|
||||||
|
|
||||||
#: com/templates/com/poster_list.jinja com/templates/com/screen_slideshow.jinja
|
#: com/templates/com/poster_list.jinja com/templates/com/screen_slideshow.jinja
|
||||||
msgid "Click to expand"
|
msgid "Click to expand"
|
||||||
msgstr "Cliquez pour agrandir"
|
msgstr "Cliquez pour agrandir"
|
||||||
|
|
||||||
#: com/templates/com/poster_moderate.jinja
|
#: com/templates/com/poster_list.jinja
|
||||||
msgid "Posters - moderation"
|
msgid "No posters"
|
||||||
msgstr "Affiches - modération"
|
msgstr "Aucune affiche"
|
||||||
|
|
||||||
#: com/templates/com/poster_moderate.jinja
|
|
||||||
msgid "No objects"
|
|
||||||
msgstr "Aucun éléments"
|
|
||||||
|
|
||||||
#: com/templates/com/screen_edit.jinja
|
#: com/templates/com/screen_edit.jinja
|
||||||
msgid "Screen"
|
msgid "Screen"
|
||||||
@@ -2951,6 +2957,18 @@ msgstr "Cet UID est invalide"
|
|||||||
msgid "User not found"
|
msgid "User not found"
|
||||||
msgstr "Utilisateur non trouvé"
|
msgstr "Utilisateur non trouvé"
|
||||||
|
|
||||||
|
#: counter/forms.py
|
||||||
|
msgid "Date and time of action"
|
||||||
|
msgstr "Date et heure de l'action"
|
||||||
|
|
||||||
|
#: counter/forms.py
|
||||||
|
msgid "New counters"
|
||||||
|
msgstr "Nouveaux comptoirs"
|
||||||
|
|
||||||
|
#: counter/forms.py
|
||||||
|
msgid "The selected counters will replace the current ones"
|
||||||
|
msgstr "Les comptoirs sélectionnés remplaceront les comptoirs actuels"
|
||||||
|
|
||||||
#: counter/forms.py
|
#: counter/forms.py
|
||||||
msgid ""
|
msgid ""
|
||||||
"Describe the product. If it's an event's click, give some insights about it, "
|
"Describe the product. If it's an event's click, give some insights about it, "
|
||||||
@@ -3285,6 +3303,52 @@ msgid "The returnable product cannot be the same as the returned one"
|
|||||||
msgstr ""
|
msgstr ""
|
||||||
"Le produit consigné ne peut pas être le même que le produit de déconsigne"
|
"Le produit consigné ne peut pas être le même que le produit de déconsigne"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Archiving"
|
||||||
|
msgstr "Archivage"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Counters change"
|
||||||
|
msgstr "Changement des comptoirs"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Product scheduled action"
|
||||||
|
msgstr "Actions sur produit planifiées"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Product actions must declare a clocked schedule."
|
||||||
|
msgstr "Les actions sur les produits doivent avoir un horaire planifié."
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Year + month field (day forced to 1)"
|
||||||
|
msgstr "Champ Année + mois (jour forcé à 1)"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
#, python-format
|
||||||
|
msgid ""
|
||||||
|
"“%(value)s” value has an invalid date format. It must be in YYYY-MM format."
|
||||||
|
msgstr ""
|
||||||
|
"La valeur « %(value)s » a un format de date invalide. Ce doit être au format "
|
||||||
|
"YYYY-MM."
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
#, python-format
|
||||||
|
msgid ""
|
||||||
|
"“%(value)s” value has the correct format (YYYY-MM) but it is an invalid date."
|
||||||
|
msgstr "La valeur « %(value)s » a le bon format, mais est une date invalide."
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "invoice date"
|
||||||
|
msgstr "date de la facture"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Invoice call"
|
||||||
|
msgstr "Appel à facture"
|
||||||
|
|
||||||
|
#: counter/models.py
|
||||||
|
msgid "Invoice calls"
|
||||||
|
msgstr "Appels à facture"
|
||||||
|
|
||||||
#: counter/templates/counter/activity.jinja
|
#: counter/templates/counter/activity.jinja
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%(counter_name)s activity"
|
msgid "%(counter_name)s activity"
|
||||||
@@ -3515,6 +3579,10 @@ msgstr "Payements en Carte Bancaire"
|
|||||||
msgid "Sum"
|
msgid "Sum"
|
||||||
msgstr "Somme"
|
msgstr "Somme"
|
||||||
|
|
||||||
|
#: counter/templates/counter/invoices_call.jinja
|
||||||
|
msgid "Validated"
|
||||||
|
msgstr "Validé"
|
||||||
|
|
||||||
#: counter/templates/counter/last_ops.jinja
|
#: counter/templates/counter/last_ops.jinja
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%(counter_name)s last operations"
|
msgid "%(counter_name)s last operations"
|
||||||
@@ -3603,6 +3671,25 @@ msgstr ""
|
|||||||
"votre cotisation. Si vous ne renouvelez pas votre cotisation, il n'y aura "
|
"votre cotisation. Si vous ne renouvelez pas votre cotisation, il n'y aura "
|
||||||
"aucune conséquence autre que le retrait de l'argent de votre compte."
|
"aucune conséquence autre que le retrait de l'argent de votre compte."
|
||||||
|
|
||||||
|
#: counter/templates/counter/product_form.jinja
|
||||||
|
#, python-format
|
||||||
|
msgid "Edit product %(name)s"
|
||||||
|
msgstr "Édition du produit %(name)s"
|
||||||
|
|
||||||
|
#: counter/templates/counter/product_form.jinja
|
||||||
|
msgid "Product creation"
|
||||||
|
msgstr "Création de produit"
|
||||||
|
|
||||||
|
#: counter/templates/counter/product_form.jinja
|
||||||
|
msgid "Automatic actions"
|
||||||
|
msgstr "Actions automatiques"
|
||||||
|
|
||||||
|
#: counter/templates/counter/product_form.jinja
|
||||||
|
msgid "Automatic actions allows to schedule product changes ahead of time."
|
||||||
|
msgstr ""
|
||||||
|
"Les actions automatiques vous permettent de planifier des modifications du "
|
||||||
|
"produit à l'avance."
|
||||||
|
|
||||||
#: counter/templates/counter/product_list.jinja
|
#: counter/templates/counter/product_list.jinja
|
||||||
msgid "Product list"
|
msgid "Product list"
|
||||||
msgstr "Liste des produits"
|
msgstr "Liste des produits"
|
||||||
@@ -3785,6 +3872,10 @@ msgstr "L'utilisateur n'est pas barman."
|
|||||||
msgid "Bad location, someone is already logged in somewhere else"
|
msgid "Bad location, someone is already logged in somewhere else"
|
||||||
msgstr "Mauvais comptoir, quelqu'un est déjà connecté ailleurs"
|
msgstr "Mauvais comptoir, quelqu'un est déjà connecté ailleurs"
|
||||||
|
|
||||||
|
#: counter/views/invoice.py
|
||||||
|
msgid "Invoice calls status has been updated."
|
||||||
|
msgstr "Le statut des appels à facture a été mis à jour."
|
||||||
|
|
||||||
#: counter/views/mixins.py
|
#: counter/views/mixins.py
|
||||||
msgid "Cash summary"
|
msgid "Cash summary"
|
||||||
msgstr "Relevé de caisse"
|
msgstr "Relevé de caisse"
|
||||||
@@ -4974,47 +5065,47 @@ msgstr "Suppression de rechargement"
|
|||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "One semester"
|
msgid "One semester"
|
||||||
msgstr "Un semestre, 20 €"
|
msgstr "Un semestre"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Two semesters"
|
msgid "Two semesters"
|
||||||
msgstr "Deux semestres, 35 €"
|
msgstr "Deux semestres"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Common core cursus"
|
msgid "Common core cursus"
|
||||||
msgstr "Cursus tronc commun, 60 €"
|
msgstr "Cursus tronc commun"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Branch cursus"
|
msgid "Branch cursus"
|
||||||
msgstr "Cursus branche, 60 €"
|
msgstr "Cursus branche"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Alternating cursus"
|
msgid "Alternating cursus"
|
||||||
msgstr "Cursus alternant, 30 €"
|
msgstr "Cursus alternant"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Honorary member"
|
msgid "Honorary member"
|
||||||
msgstr "Membre honoraire, 0 €"
|
msgstr "Membre honoraire"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Assidu member"
|
msgid "Assidu member"
|
||||||
msgstr "Membre d'Assidu, 0 €"
|
msgstr "Membre d'Assidu"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Amicale/DOCEO member"
|
msgid "Amicale/DOCEO member"
|
||||||
msgstr "Membre de l'Amicale/DOCEO, 0 €"
|
msgstr "Membre de l'Amicale/DOCEO"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "UT network member"
|
msgid "UT network member"
|
||||||
msgstr "Cotisant du réseau UT, 0 €"
|
msgstr "Cotisant du réseau UT"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "CROUS member"
|
msgid "CROUS member"
|
||||||
msgstr "Membres du CROUS, 0 €"
|
msgstr "Membres du CROUS"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Sbarro/ESTA member"
|
msgid "Sbarro/ESTA member"
|
||||||
msgstr "Membre de Sbarro ou de l'ESTA, 20 €"
|
msgstr "Membre de Sbarro ou de l'ESTA"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "One semester Welcome Week"
|
msgid "One semester Welcome Week"
|
||||||
@@ -5041,28 +5132,28 @@ msgid "One day"
|
|||||||
msgstr "Un jour"
|
msgstr "Un jour"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "GA staff member"
|
msgid "GA staff member (2 weeks)"
|
||||||
msgstr "Membre staff GA (2 semaines), 1 €"
|
msgstr "Membre staff GA (2 semaines)"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "One semester (-20%)"
|
msgid "One semester (-20%)"
|
||||||
msgstr "Un semestre (-20%), 12 €"
|
msgstr "Un semestre (-20%)"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Two semesters (-20%)"
|
msgid "Two semesters (-20%)"
|
||||||
msgstr "Deux semestres (-20%), 22 €"
|
msgstr "Deux semestres (-20%)"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Common core cursus (-20%)"
|
msgid "Common core cursus (-20%)"
|
||||||
msgstr "Cursus tronc commun (-20%), 36 €"
|
msgstr "Cursus tronc commun (-20%)"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Branch cursus (-20%)"
|
msgid "Branch cursus (-20%)"
|
||||||
msgstr "Cursus branche (-20%), 36 €"
|
msgstr "Cursus branche (-20%)"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "Alternating cursus (-20%)"
|
msgid "Alternating cursus (-20%)"
|
||||||
msgstr "Cursus alternant (-20%), 24 €"
|
msgstr "Cursus alternant (-20%)"
|
||||||
|
|
||||||
#: sith/settings.py
|
#: sith/settings.py
|
||||||
msgid "One year for free(CA offer)"
|
msgid "One year for free(CA offer)"
|
||||||
@@ -5236,6 +5327,18 @@ msgstr "Membre existant"
|
|||||||
msgid "the groups that can create subscriptions"
|
msgid "the groups that can create subscriptions"
|
||||||
msgstr "les groupes pouvant créer des cotisations"
|
msgstr "les groupes pouvant créer des cotisations"
|
||||||
|
|
||||||
|
#: timetable/templates/timetable/generator.jinja
|
||||||
|
msgid "Timetable generator"
|
||||||
|
msgstr "Générateur d'emploi du temps"
|
||||||
|
|
||||||
|
#: timetable/templates/timetable/generator.jinja
|
||||||
|
msgid "Generate"
|
||||||
|
msgstr "Générer"
|
||||||
|
|
||||||
|
#: timetable/templates/timetable/generator.jinja
|
||||||
|
msgid "Save to PNG"
|
||||||
|
msgstr "Sauver en PNG"
|
||||||
|
|
||||||
#: trombi/models.py
|
#: trombi/models.py
|
||||||
msgid "subscription deadline"
|
msgid "subscription deadline"
|
||||||
msgstr "fin des inscriptions"
|
msgstr "fin des inscriptions"
|
||||||
|
|||||||
1104
package-lock.json
generated
1104
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
59
package.json
59
package.json
@@ -24,47 +24,48 @@
|
|||||||
"#com:*": "./com/static/bundled/*"
|
"#com:*": "./com/static/bundled/*"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.25.2",
|
"@babel/core": "^7.28.5",
|
||||||
"@babel/preset-env": "^7.25.4",
|
"@babel/preset-env": "^7.28.5",
|
||||||
"@biomejs/biome": "1.9.4",
|
"@biomejs/biome": "^1.9.4",
|
||||||
"@hey-api/openapi-ts": "^0.73.0",
|
"@hey-api/openapi-ts": "^0.73.0",
|
||||||
"@rollup/plugin-inject": "^5.0.5",
|
"@rollup/plugin-inject": "^5.0.5",
|
||||||
"@types/alpinejs": "^3.13.10",
|
"@types/alpinejs": "^3.13.11",
|
||||||
"@types/cytoscape-cxtmenu": "^3.4.4",
|
"@types/cytoscape-cxtmenu": "^3.4.5",
|
||||||
"@types/cytoscape-klay": "^3.1.4",
|
"@types/cytoscape-klay": "^3.1.5",
|
||||||
"@types/js-cookie": "^3.0.6",
|
"@types/js-cookie": "^3.0.6",
|
||||||
"typescript": "^5.8.3",
|
"typescript": "^5.9.3",
|
||||||
"vite": "^6.3.6",
|
"vite": "^6.4.1",
|
||||||
"vite-bundle-visualizer": "^1.2.1",
|
"vite-bundle-visualizer": "^1.2.1",
|
||||||
"vite-plugin-static-copy": "^3.1.2"
|
"vite-plugin-static-copy": "^3.1.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@alpinejs/sort": "^3.14.7",
|
"@alpinejs/sort": "^3.15.1",
|
||||||
"@arendjr/text-clipper": "npm:@jsr/arendjr__text-clipper@^3.0.0",
|
"@arendjr/text-clipper": "npm:@jsr/arendjr__text-clipper@^3.0.0",
|
||||||
"@floating-ui/dom": "^1.6.13",
|
"@floating-ui/dom": "^1.7.4",
|
||||||
"@fortawesome/fontawesome-free": "^6.6.0",
|
"@fortawesome/fontawesome-free": "^6.7.2",
|
||||||
"@fullcalendar/core": "^6.1.15",
|
"@fullcalendar/core": "^6.1.19",
|
||||||
"@fullcalendar/daygrid": "^6.1.15",
|
"@fullcalendar/daygrid": "^6.1.19",
|
||||||
"@fullcalendar/icalendar": "^6.1.15",
|
"@fullcalendar/icalendar": "^6.1.19",
|
||||||
"@fullcalendar/list": "^6.1.15",
|
"@fullcalendar/list": "^6.1.19",
|
||||||
"@sentry/browser": "^9.29.0",
|
"@sentry/browser": "^9.46.0",
|
||||||
"@zip.js/zip.js": "^2.7.52",
|
"@zip.js/zip.js": "^2.8.9",
|
||||||
"3d-force-graph": "^1.73.4",
|
"3d-force-graph": "^1.79.0",
|
||||||
"alpinejs": "^3.14.7",
|
"alpinejs": "^3.15.1",
|
||||||
"chart.js": "^4.4.4",
|
"chart.js": "^4.5.1",
|
||||||
"country-flag-emoji-polyfill": "^0.1.8",
|
"country-flag-emoji-polyfill": "^0.1.8",
|
||||||
"cytoscape": "^3.30.2",
|
"cytoscape": "^3.33.1",
|
||||||
"cytoscape-cxtmenu": "^3.5.0",
|
"cytoscape-cxtmenu": "^3.5.0",
|
||||||
"cytoscape-klay": "^3.1.4",
|
"cytoscape-klay": "^3.1.4",
|
||||||
"d3-force-3d": "^3.0.5",
|
"d3-force-3d": "^3.0.6",
|
||||||
"easymde": "^2.19.0",
|
"easymde": "^2.20.0",
|
||||||
"glob": "^11.0.0",
|
"glob": "^11.0.3",
|
||||||
"htmx.org": "^2.0.3",
|
"html2canvas": "^1.4.1",
|
||||||
|
"htmx.org": "^2.0.8",
|
||||||
"js-cookie": "^3.0.5",
|
"js-cookie": "^3.0.5",
|
||||||
"lit-html": "^3.3.0",
|
"lit-html": "^3.3.1",
|
||||||
"native-file-system-adapter": "^3.0.1",
|
"native-file-system-adapter": "^3.0.1",
|
||||||
"three": "^0.177.0",
|
"three": "^0.177.0",
|
||||||
"three-spritetext": "^1.9.0",
|
"three-spritetext": "^1.10.0",
|
||||||
"tom-select": "^2.3.1"
|
"tom-select": "^2.4.3"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,36 +19,36 @@ authors = [
|
|||||||
license = { text = "GPL-3.0-only" }
|
license = { text = "GPL-3.0-only" }
|
||||||
requires-python = "<4.0,>=3.12"
|
requires-python = "<4.0,>=3.12"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"django>=5.2.1,<6.0.0",
|
"django>=5.2.8,<6.0.0",
|
||||||
"django-ninja<2.0.0,>=1.4.0",
|
"django-ninja>=1.4.5,<2.0.0",
|
||||||
"django-ninja-extra<1.0.0,>=0.22.9",
|
"django-ninja-extra>=0.30.2,<1.0.0",
|
||||||
"Pillow<12.0.0,>=11.1.0",
|
"Pillow>=12.0.0,<13.0.0",
|
||||||
"mistune<4.0.0,>=3.1.3",
|
"mistune>=3.1.4,<4.0.0",
|
||||||
"django-jinja<3.0.0,>=2.11.0",
|
"django-jinja<3.0.0,>=2.11.0",
|
||||||
"cryptography>=45.0.3,<46.0.0",
|
"cryptography>=46.0.3,<47.0.0",
|
||||||
"django-phonenumber-field<9.0.0,>=8.1.0",
|
"django-phonenumber-field>=8.3.0,<9.0.0",
|
||||||
"phonenumbers>=9.0.2,<10.0.0",
|
"phonenumbers>=9.0.18,<10.0.0",
|
||||||
"reportlab<5.0.0,>=4.3.1",
|
"reportlab>=4.4.4,<5.0.0",
|
||||||
"django-haystack<4.0.0,>=3.3.0",
|
"django-haystack<4.0.0,>=3.3.0",
|
||||||
"xapian-haystack<4.0.0,>=3.1.0",
|
"xapian-haystack<4.0.0,>=3.1.0",
|
||||||
"libsass<1.0.0,>=0.23.0",
|
"libsass<1.0.0,>=0.23.0",
|
||||||
"django-ordered-model<4.0.0,>=3.7.4",
|
"django-ordered-model<4.0.0,>=3.7.4",
|
||||||
"django-simple-captcha<1.0.0,>=0.6.2",
|
"django-simple-captcha<1.0.0,>=0.6.2",
|
||||||
"python-dateutil<3.0.0.0,>=2.9.0.post0",
|
"python-dateutil<3.0.0.0,>=2.9.0.post0",
|
||||||
"sentry-sdk<3.0.0,>=2.25.1",
|
"sentry-sdk>=2.43.0,<3.0.0",
|
||||||
"jinja2<4.0.0,>=3.1.6",
|
"jinja2<4.0.0,>=3.1.6",
|
||||||
"django-countries<8.0.0,>=7.6.1",
|
"django-countries>=8.0.0,<9.0.0",
|
||||||
"dict2xml<2.0.0,>=1.7.6",
|
"dict2xml>=1.7.7,<2.0.0",
|
||||||
"Sphinx<6,>=5",
|
"Sphinx<6,>=5",
|
||||||
"tomli<3.0.0,>=2.2.1",
|
"tomli>=2.3.0,<3.0.0",
|
||||||
"django-honeypot>=1.3.0,<2",
|
"django-honeypot>=1.3.0,<2",
|
||||||
"pydantic-extra-types<3.0.0,>=2.10.3",
|
"pydantic-extra-types>=2.10.6,<3.0.0",
|
||||||
"ical>=11,<12",
|
"ical>=11.1.0,<12",
|
||||||
"redis[hiredis]<7,>=5.3.0",
|
"redis[hiredis]>=5.3.0,<8",
|
||||||
"environs[django]<15.0.0,>=14.1.1",
|
"environs[django]>=14.5.0,<15.0.0",
|
||||||
"requests>=2.32.3",
|
"requests>=2.32.5,<3.0.0",
|
||||||
"honcho>=2.0.0",
|
"honcho>=2.0.0",
|
||||||
"psutil>=7.0.0",
|
"psutil>=7.1.3,<8.0.0",
|
||||||
"celery[redis]>=5.5.2",
|
"celery[redis]>=5.5.2",
|
||||||
"django-celery-results>=2.5.1",
|
"django-celery-results>=2.5.1",
|
||||||
"django-celery-beat>=2.7.0",
|
"django-celery-beat>=2.7.0",
|
||||||
@@ -60,32 +60,32 @@ documentation = "https://sith-ae.readthedocs.io/"
|
|||||||
|
|
||||||
[dependency-groups]
|
[dependency-groups]
|
||||||
prod = [
|
prod = [
|
||||||
"psycopg[c]>=3.2.9,<4.0.0",
|
"psycopg[c]>=3.2.12,<4.0.0",
|
||||||
]
|
]
|
||||||
dev = [
|
dev = [
|
||||||
"django-debug-toolbar>=6,<7",
|
"django-debug-toolbar>=6.1.0,<7",
|
||||||
"ipython<10.0.0,>=9.0.2",
|
"ipython>=9.7.0,<10.0.0",
|
||||||
"pre-commit<5.0.0,>=4.1.0",
|
"pre-commit>=4.3.0,<5.0.0",
|
||||||
"ruff>=0.11.13,<1.0.0",
|
"ruff>=0.14.4,<1.0.0",
|
||||||
"djhtml<4.0.0,>=3.0.7",
|
"djhtml>=3.0.10,<4.0.0",
|
||||||
"faker<38.0.0,>=37.0.0",
|
"faker>=37.12.0,<38.0.0",
|
||||||
"rjsmin<2.0.0,>=1.2.4",
|
"rjsmin>=1.2.5,<2.0.0",
|
||||||
]
|
]
|
||||||
tests = [
|
tests = [
|
||||||
"freezegun<2.0.0,>=1.5.1",
|
"freezegun>=1.5.5,<2.0.0",
|
||||||
"pytest<9.0.0,>=8.3.5",
|
"pytest>=8.4.2,<9.0.0",
|
||||||
"pytest-cov<7.0.0,>=6.0.0",
|
"pytest-cov>=7.0.0,<8.0.0",
|
||||||
"pytest-django<5.0.0,>=4.10.0",
|
"pytest-django<5.0.0,>=4.10.0",
|
||||||
"model-bakery<2.0.0,>=1.20.4",
|
"model-bakery<2.0.0,>=1.20.4",
|
||||||
"beautifulsoup4>=4.13.3,<5",
|
"beautifulsoup4>=4.14.2,<5",
|
||||||
"lxml>=6,<7",
|
"lxml>=6.0.2,<7",
|
||||||
]
|
]
|
||||||
docs = [
|
docs = [
|
||||||
"mkdocs<2.0.0,>=1.6.1",
|
"mkdocs<2.0.0,>=1.6.1",
|
||||||
"mkdocs-material<10.0.0,>=9.6.7",
|
"mkdocs-material>=9.6.23,<10.0.0",
|
||||||
"mkdocstrings<1.0.0,>=0.28.3",
|
"mkdocstrings>=0.30.1,<1.0.0",
|
||||||
"mkdocstrings-python<2.0.0,>=1.16.3",
|
"mkdocstrings-python>=1.18.2,<2.0.0",
|
||||||
"mkdocs-include-markdown-plugin<8.0.0,>=7.1.5",
|
"mkdocs-include-markdown-plugin>=7.2.0,<8.0.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[tool.uv]
|
[tool.uv]
|
||||||
|
|||||||
@@ -5,12 +5,13 @@ import type { PictureSchema } from "#openapi";
|
|||||||
document.addEventListener("alpine:init", () => {
|
document.addEventListener("alpine:init", () => {
|
||||||
Alpine.data("pictures_download", () => ({
|
Alpine.data("pictures_download", () => ({
|
||||||
isDownloading: false,
|
isDownloading: false,
|
||||||
|
downloadPictures: [] as PictureSchema[],
|
||||||
|
|
||||||
async downloadZip() {
|
async downloadZip() {
|
||||||
this.isDownloading = true;
|
this.isDownloading = true;
|
||||||
const bar = this.$refs.progress;
|
const bar = this.$refs.progress;
|
||||||
bar.value = 0;
|
bar.value = 0;
|
||||||
bar.max = this.pictures.length;
|
bar.max = this.downloadPictures.length;
|
||||||
|
|
||||||
const incrementProgressBar = (_total: number): undefined => {
|
const incrementProgressBar = (_total: number): undefined => {
|
||||||
bar.value++;
|
bar.value++;
|
||||||
@@ -29,7 +30,7 @@ document.addEventListener("alpine:init", () => {
|
|||||||
const zipWriter = new ZipWriter(await fileHandle.createWritable());
|
const zipWriter = new ZipWriter(await fileHandle.createWritable());
|
||||||
|
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
this.pictures.map((p: PictureSchema) => {
|
this.downloadPictures.map(async (p: PictureSchema) => {
|
||||||
const imgName = `${p.album}/IMG_${p.date.replace(/[:\-]/g, "_")}${p.name.slice(p.name.lastIndexOf("."))}`;
|
const imgName = `${p.album}/IMG_${p.date.replace(/[:\-]/g, "_")}${p.name.slice(p.name.lastIndexOf("."))}`;
|
||||||
return zipWriter.add(imgName, new HttpReader(p.full_size_url), {
|
return zipWriter.add(imgName, new HttpReader(p.full_size_url), {
|
||||||
level: 9,
|
level: 9,
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import {
|
|||||||
|
|
||||||
interface PagePictureConfig {
|
interface PagePictureConfig {
|
||||||
userId: number;
|
userId: number;
|
||||||
|
nbPictures?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Album {
|
interface Album {
|
||||||
@@ -20,11 +21,27 @@ document.addEventListener("alpine:init", () => {
|
|||||||
loading: true,
|
loading: true,
|
||||||
albums: [] as Album[],
|
albums: [] as Album[],
|
||||||
|
|
||||||
async init() {
|
async fetchPictures(): Promise<PictureSchema[]> {
|
||||||
|
const localStorageKey = `user${config.userId}Pictures`;
|
||||||
|
const localStorageInvalidationKey = `user${config.userId}PicturesNumber`;
|
||||||
|
const lastCachedNumber = localStorage.getItem(localStorageInvalidationKey);
|
||||||
|
if (
|
||||||
|
lastCachedNumber !== null &&
|
||||||
|
Number.parseInt(lastCachedNumber) === config.nbPictures
|
||||||
|
) {
|
||||||
|
return JSON.parse(localStorage.getItem(localStorageKey));
|
||||||
|
}
|
||||||
const pictures = await paginated(picturesFetchPictures, {
|
const pictures = await paginated(picturesFetchPictures, {
|
||||||
// biome-ignore lint/style/useNamingConvention: from python api
|
// biome-ignore lint/style/useNamingConvention: from python api
|
||||||
query: { users_identified: [config.userId] },
|
query: { users_identified: [config.userId] },
|
||||||
} as PicturesFetchPicturesData);
|
} as PicturesFetchPicturesData);
|
||||||
|
localStorage.setItem(localStorageInvalidationKey, config.nbPictures.toString());
|
||||||
|
localStorage.setItem(localStorageKey, JSON.stringify(pictures));
|
||||||
|
return pictures;
|
||||||
|
},
|
||||||
|
|
||||||
|
async init() {
|
||||||
|
const pictures = await this.fetchPictures();
|
||||||
const groupedAlbums = Object.groupBy(pictures, (i: PictureSchema) => i.album.id);
|
const groupedAlbums = Object.groupBy(pictures, (i: PictureSchema) => i.album.id);
|
||||||
this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => {
|
this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => {
|
||||||
return {
|
return {
|
||||||
@@ -40,5 +57,9 @@ document.addEventListener("alpine:init", () => {
|
|||||||
}
|
}
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
allPictures(): PictureSchema[] {
|
||||||
|
return this.albums.flatMap((album: Album) => album.pictures);
|
||||||
|
},
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -85,7 +85,7 @@
|
|||||||
<div x-data="pictures({ albumId: {{ album.id }}, maxPageSize: {{ settings.SITH_SAS_IMAGES_PER_PAGE }} })">
|
<div x-data="pictures({ albumId: {{ album.id }}, maxPageSize: {{ settings.SITH_SAS_IMAGES_PER_PAGE }} })">
|
||||||
<h4>{% trans %}Pictures{% endtrans %}</h4>
|
<h4>{% trans %}Pictures{% endtrans %}</h4>
|
||||||
<br>
|
<br>
|
||||||
{{ download_button(_("Download album")) }}
|
{{ download_button(_("Download album"), "pictures") }}
|
||||||
<div class="photos" :aria-busy="loading" @pictures-upload-done.window="fetchPictures">
|
<div class="photos" :aria-busy="loading" @pictures-upload-done.window="fetchPictures">
|
||||||
<template x-for="picture in getPage(page)">
|
<template x-for="picture in getPage(page)">
|
||||||
<a :href="picture.sas_url">
|
<a :href="picture.sas_url">
|
||||||
|
|||||||
@@ -36,21 +36,20 @@
|
|||||||
{% endmacro %}
|
{% endmacro %}
|
||||||
|
|
||||||
{# Helper macro to create a download button for a
|
{# Helper macro to create a download button for a
|
||||||
record of albums with alpine
|
record of albums with alpine.
|
||||||
|
|
||||||
This needs to be used inside an alpine environment.
|
|
||||||
Downloaded pictures will be `pictures` from the
|
|
||||||
parent data store.
|
|
||||||
|
|
||||||
Note:
|
Note:
|
||||||
This requires importing `bundled/sas/pictures-download-index.ts`
|
This requires importing `bundled/sas/pictures-download-index.ts`
|
||||||
|
|
||||||
Parameters:
|
Parameters:
|
||||||
name (str): name displayed on the button
|
name (str): name displayed on the button
|
||||||
|
pictures (str): an alpine variable or function
|
||||||
|
which holds the images this button should download.
|
||||||
|
It must be different from "downloadPictures", or it won't work.
|
||||||
#}
|
#}
|
||||||
{% macro download_button(name) %}
|
{% macro download_button(name, pictures) %}
|
||||||
<div x-data="pictures_download">
|
<div x-data="pictures_download()" x-modelable="downloadPictures" x-model="{{ pictures }}">
|
||||||
<div x-show="albums.length > 0" x-cloak>
|
<div x-show="downloadPictures.length > 0" x-cloak>
|
||||||
<button
|
<button
|
||||||
:disabled="isDownloading"
|
:disabled="isDownloading"
|
||||||
class="btn btn-blue {% if name == "" %}btn-no-text{% endif %}"
|
class="btn btn-blue {% if name == "" %}btn-no-text{% endif %}"
|
||||||
|
|||||||
@@ -15,18 +15,18 @@
|
|||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<main x-data="user_pictures({ userId: {{ object.id }} })">
|
<main x-data="user_pictures({ userId: {{ object.id }}, nbPictures: {{ object.nb_pictures }} })">
|
||||||
{% if user.id == object.id %}
|
{% if user.id == object.id %}
|
||||||
{{ download_button(_("Download all my pictures")) }}
|
{{ download_button(_("Download all my pictures"), "allPictures()") }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<template x-for="album in albums" x-cloak>
|
<template x-for="album in albums" x-cloak>
|
||||||
<section>
|
<section>
|
||||||
<br />
|
<br />
|
||||||
<div class="row">
|
<div class="row gap">
|
||||||
<h4 x-text="album.name" :id="`album-${album.id}`"></h4>
|
<h4 x-text="album.name" :id="`album-${album.id}`"></h4>
|
||||||
{% if user.id == object.id %}
|
{% if user.id == object.id %}
|
||||||
{{ download_button("") }}
|
{{ download_button("", "album.pictures") }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
<div class="photos">
|
<div class="photos">
|
||||||
|
|||||||
10
sas/views.py
10
sas/views.py
@@ -16,6 +16,7 @@ from typing import Any
|
|||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.exceptions import PermissionDenied
|
from django.core.exceptions import PermissionDenied
|
||||||
|
from django.db.models import Count, OuterRef, Subquery
|
||||||
from django.http import Http404, HttpResponseRedirect
|
from django.http import Http404, HttpResponseRedirect
|
||||||
from django.shortcuts import get_object_or_404
|
from django.shortcuts import get_object_or_404
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
@@ -36,7 +37,7 @@ from sas.forms import (
|
|||||||
PictureModerationRequestForm,
|
PictureModerationRequestForm,
|
||||||
PictureUploadForm,
|
PictureUploadForm,
|
||||||
)
|
)
|
||||||
from sas.models import Album, Picture
|
from sas.models import Album, PeoplePictureRelation, Picture
|
||||||
|
|
||||||
|
|
||||||
class AlbumCreateFragment(FragmentMixin, CreateView):
|
class AlbumCreateFragment(FragmentMixin, CreateView):
|
||||||
@@ -178,6 +179,13 @@ class UserPicturesView(UserTabsMixin, CanViewMixin, DetailView):
|
|||||||
context_object_name = "profile"
|
context_object_name = "profile"
|
||||||
template_name = "sas/user_pictures.jinja"
|
template_name = "sas/user_pictures.jinja"
|
||||||
current_tab = "pictures"
|
current_tab = "pictures"
|
||||||
|
queryset = User.objects.annotate(
|
||||||
|
nb_pictures=Subquery(
|
||||||
|
PeoplePictureRelation.objects.filter(user=OuterRef("id"))
|
||||||
|
.values("user_id")
|
||||||
|
.values(count=Count("*"))
|
||||||
|
)
|
||||||
|
).all()
|
||||||
|
|
||||||
|
|
||||||
# Admin views
|
# Admin views
|
||||||
|
|||||||
@@ -125,6 +125,7 @@ INSTALLED_APPS = (
|
|||||||
"pedagogy",
|
"pedagogy",
|
||||||
"galaxy",
|
"galaxy",
|
||||||
"antispam",
|
"antispam",
|
||||||
|
"timetable",
|
||||||
"api",
|
"api",
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -541,7 +542,7 @@ SITH_SUBSCRIPTIONS = {
|
|||||||
"duration": 4,
|
"duration": 4,
|
||||||
},
|
},
|
||||||
"cursus-branche": {"name": _("Branch cursus"), "price": 60, "duration": 6},
|
"cursus-branche": {"name": _("Branch cursus"), "price": 60, "duration": 6},
|
||||||
"cursus-alternant": {"name": _("Alternating cursus"), "price": 30, "duration": 6},
|
"cursus-alternant": {"name": _("Alternating cursus"), "price": 35, "duration": 6},
|
||||||
"membre-honoraire": {"name": _("Honorary member"), "price": 0, "duration": 666},
|
"membre-honoraire": {"name": _("Honorary member"), "price": 0, "duration": 666},
|
||||||
"assidu": {"name": _("Assidu member"), "price": 0, "duration": 2},
|
"assidu": {"name": _("Assidu member"), "price": 0, "duration": 2},
|
||||||
"amicale/doceo": {"name": _("Amicale/DOCEO member"), "price": 0, "duration": 2},
|
"amicale/doceo": {"name": _("Amicale/DOCEO member"), "price": 0, "duration": 2},
|
||||||
@@ -553,8 +554,6 @@ SITH_SUBSCRIPTIONS = {
|
|||||||
"price": 0,
|
"price": 0,
|
||||||
"duration": 1,
|
"duration": 1,
|
||||||
},
|
},
|
||||||
"un-mois-essai": {"name": _("One month for free"), "price": 0, "duration": 0.166},
|
|
||||||
"deux-mois-essai": {"name": _("Two months for free"), "price": 0, "duration": 0.33},
|
|
||||||
"benevoles-euroks": {"name": _("Eurok's volunteer"), "price": 5, "duration": 0.1},
|
"benevoles-euroks": {"name": _("Eurok's volunteer"), "price": 5, "duration": 0.1},
|
||||||
"six-semaines-essai": {
|
"six-semaines-essai": {
|
||||||
"name": _("Six weeks for free"),
|
"name": _("Six weeks for free"),
|
||||||
|
|||||||
@@ -53,6 +53,7 @@ urlpatterns = [
|
|||||||
path("i18n/", include("django.conf.urls.i18n")),
|
path("i18n/", include("django.conf.urls.i18n")),
|
||||||
path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"),
|
path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"),
|
||||||
path("captcha/", include("captcha.urls")),
|
path("captcha/", include("captcha.urls")),
|
||||||
|
path("edt/", include(("timetable.urls", "timetable"), namespace="timetable")),
|
||||||
]
|
]
|
||||||
|
|
||||||
if settings.DEBUG:
|
if settings.DEBUG:
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
# Generated by Django 5.2.3 on 2025-10-06 11:24
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
import subscription.models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [("subscription", "0015_alter_subscription_location_and_more")]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="subscription",
|
||||||
|
name="subscription_type",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=subscription.models.get_subscription_types,
|
||||||
|
max_length=255,
|
||||||
|
verbose_name="subscription type",
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
@@ -38,16 +38,19 @@ def validate_payment(value):
|
|||||||
raise ValidationError(_("Bad payment method"))
|
raise ValidationError(_("Bad payment method"))
|
||||||
|
|
||||||
|
|
||||||
|
def get_subscription_types():
|
||||||
|
return (
|
||||||
|
(k, f"{v['name']}, {v['price']}€")
|
||||||
|
for k, v in sorted(settings.SITH_SUBSCRIPTIONS.items())
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class Subscription(models.Model):
|
class Subscription(models.Model):
|
||||||
member = models.ForeignKey(
|
member = models.ForeignKey(
|
||||||
User, related_name="subscriptions", on_delete=models.CASCADE
|
User, related_name="subscriptions", on_delete=models.CASCADE
|
||||||
)
|
)
|
||||||
subscription_type = models.CharField(
|
subscription_type = models.CharField(
|
||||||
_("subscription type"),
|
_("subscription type"), max_length=255, choices=get_subscription_types
|
||||||
max_length=255,
|
|
||||||
choices=(
|
|
||||||
(k, v["name"]) for k, v in sorted(settings.SITH_SUBSCRIPTIONS.items())
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
subscription_start = models.DateField(_("subscription start"))
|
subscription_start = models.DateField(_("subscription start"))
|
||||||
subscription_end = models.DateField(_("subscription end"))
|
subscription_end = models.DateField(_("subscription end"))
|
||||||
@@ -78,7 +81,7 @@ class Subscription(models.Model):
|
|||||||
|
|
||||||
from counter.models import Customer
|
from counter.models import Customer
|
||||||
|
|
||||||
customer, _ = Customer.get_or_create(self.member)
|
Customer.get_or_create(self.member)
|
||||||
# Someone who subscribed once will be considered forever
|
# Someone who subscribed once will be considered forever
|
||||||
# as an old subscriber.
|
# as an old subscriber.
|
||||||
self.member.groups.add(settings.SITH_GROUP_OLD_SUBSCRIBERS_ID)
|
self.member.groups.add(settings.SITH_GROUP_OLD_SUBSCRIBERS_ID)
|
||||||
|
|||||||
@@ -175,45 +175,3 @@ class TestSubscriptionIntegration(TestCase):
|
|||||||
user=user,
|
user=user,
|
||||||
)
|
)
|
||||||
assert d == date(2017, 8, 29)
|
assert d == date(2017, 8, 29)
|
||||||
|
|
||||||
def test_dates_renewal_sliding_during_two_free_monthes(self):
|
|
||||||
user = self.user
|
|
||||||
s = Subscription(
|
|
||||||
member=user,
|
|
||||||
subscription_type="deux-mois-essai",
|
|
||||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
|
||||||
)
|
|
||||||
s.subscription_start = date(2015, 8, 29)
|
|
||||||
s.subscription_end = s.compute_end(
|
|
||||||
duration=settings.SITH_SUBSCRIPTIONS[s.subscription_type]["duration"],
|
|
||||||
start=s.subscription_start,
|
|
||||||
)
|
|
||||||
s.save()
|
|
||||||
assert s.subscription_end == date(2015, 10, 29)
|
|
||||||
with freezegun.freeze_time("2015-09-25"):
|
|
||||||
d = Subscription.compute_end(
|
|
||||||
duration=settings.SITH_SUBSCRIPTIONS["deux-semestres"]["duration"],
|
|
||||||
user=user,
|
|
||||||
)
|
|
||||||
assert d == date(2016, 10, 29)
|
|
||||||
|
|
||||||
def test_dates_renewal_sliding_after_two_free_monthes(self):
|
|
||||||
user = self.user
|
|
||||||
s = Subscription(
|
|
||||||
member=user,
|
|
||||||
subscription_type="deux-mois-essai",
|
|
||||||
payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1],
|
|
||||||
)
|
|
||||||
s.subscription_start = date(2015, 8, 29)
|
|
||||||
s.subscription_end = s.compute_end(
|
|
||||||
duration=settings.SITH_SUBSCRIPTIONS[s.subscription_type]["duration"],
|
|
||||||
start=s.subscription_start,
|
|
||||||
)
|
|
||||||
s.save()
|
|
||||||
assert s.subscription_end == date(2015, 10, 29)
|
|
||||||
with freezegun.freeze_time("2015-11-05"):
|
|
||||||
d = Subscription.compute_end(
|
|
||||||
duration=settings.SITH_SUBSCRIPTIONS["deux-semestres"]["duration"],
|
|
||||||
user=user,
|
|
||||||
)
|
|
||||||
assert d == date(2016, 11, 5)
|
|
||||||
|
|||||||
0
timetable/__init__.py
Normal file
0
timetable/__init__.py
Normal file
1
timetable/admin.py
Normal file
1
timetable/admin.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
# Register your models here.
|
||||||
6
timetable/apps.py
Normal file
6
timetable/apps.py
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class TimetableConfig(AppConfig):
|
||||||
|
default_auto_field = "django.db.models.BigAutoField"
|
||||||
|
name = "timetable"
|
||||||
0
timetable/migrations/__init__.py
Normal file
0
timetable/migrations/__init__.py
Normal file
1
timetable/models.py
Normal file
1
timetable/models.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
# Create your models here.
|
||||||
184
timetable/static/bundled/timetable/generator-index.ts
Normal file
184
timetable/static/bundled/timetable/generator-index.ts
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
import html2canvas from "html2canvas";
|
||||||
|
|
||||||
|
// see https://regex101.com/r/QHSaPM/2
|
||||||
|
const TIMETABLE_ROW_RE: RegExp =
|
||||||
|
/^(?<ueCode>\w.+\w)\s+(?<courseType>[A-Z]{2}\d)\s+((?<weekGroup>[AB])\s+)?(?<weekday>(lundi)|(mardi)|(mercredi)|(jeudi)|(vendredi)|(samedi)|(dimanche))\s+(?<startHour>\d{2}:\d{2})\s+(?<endHour>\d{2}:\d{2})\s+[\dA-B]\s+((?<attendance>[\wé]*)\s+)?(?<room>\w+(?:, \w+)?)$/;
|
||||||
|
|
||||||
|
const DEFAULT_TIMETABLE: string = `DS52\t\tCM1\t\tlundi\t08:00\t10:00\t1\tPrésentiel\tA113
|
||||||
|
DS53\t\tCM1\t\tlundi\t10:15\t12:15\t1\tPrésentiel\tA101
|
||||||
|
DS53\t\tTP1\t\tlundi\t13:00\t16:00\t1\tPrésentiel\tH010
|
||||||
|
SO03\t\tCM1\t\tlundi\t16:15\t17:45\t1\tPrésentiel\tA103
|
||||||
|
SO03\t\tTD1\t\tlundi\t17:45\t19:45\t1\tPrésentiel\tA103
|
||||||
|
DS50\t\tTP1\t\tmardi\t08:00\t10:00\t1\tPrésentiel\tA216
|
||||||
|
DS51\t\tCM1\t\tmardi\t10:15\t12:15\t1\tPrésentiel\tA216
|
||||||
|
DS51\t\tTP1\t\tmardi\t14:00\t18:00\t1\tPrésentiel\tH010
|
||||||
|
DS52\t\tTP2\tA\tjeudi\t08:00\t10:00\tA\tPrésentiel\tA110a, A110b
|
||||||
|
DS52\t\tTD1\t\tjeudi\t10:15\t12:15\t1\tPrésentiel\tA110a, A110b
|
||||||
|
LC02\t\tTP1\t\tjeudi\t15:00\t16:00\t1\tPrésentiel\tA209
|
||||||
|
LC02\t\tTD1\t\tjeudi\t16:15\t18:15\t1\tPrésentiel\tA206`;
|
||||||
|
|
||||||
|
type WeekDay =
|
||||||
|
| "lundi"
|
||||||
|
| "mardi"
|
||||||
|
| "mercredi"
|
||||||
|
| "jeudi"
|
||||||
|
| "vendredi"
|
||||||
|
| "samedi"
|
||||||
|
| "dimanche";
|
||||||
|
|
||||||
|
const WEEKDAYS = [
|
||||||
|
"lundi",
|
||||||
|
"mardi",
|
||||||
|
"mercredi",
|
||||||
|
"jeudi",
|
||||||
|
"vendredi",
|
||||||
|
"samedi",
|
||||||
|
"dimanche",
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
const SLOT_HEIGHT = 20 as const; // Each 15min has a height of 20px in the timetable
|
||||||
|
const SLOT_WIDTH = 250 as const; // Each weekday ha a width of 400px in the timetable
|
||||||
|
const MINUTES_PER_SLOT = 15 as const;
|
||||||
|
|
||||||
|
interface TimetableSlot {
|
||||||
|
courseType: string;
|
||||||
|
room: string;
|
||||||
|
startHour: string;
|
||||||
|
endHour: string;
|
||||||
|
startSlot: number;
|
||||||
|
endSlot: number;
|
||||||
|
ueCode: string;
|
||||||
|
weekGroup?: string;
|
||||||
|
weekday: WeekDay;
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseSlots(s: string): TimetableSlot[] {
|
||||||
|
return s
|
||||||
|
.split("\n")
|
||||||
|
.filter((s: string) => s.length > 0)
|
||||||
|
.map((row: string) => {
|
||||||
|
const parsed = TIMETABLE_ROW_RE.exec(row);
|
||||||
|
if (!parsed) {
|
||||||
|
throw new Error(`Couldn't parse row ${row}`);
|
||||||
|
}
|
||||||
|
const [startHour, startMin] = parsed.groups.startHour
|
||||||
|
.split(":")
|
||||||
|
.map((i) => Number.parseInt(i));
|
||||||
|
const [endHour, endMin] = parsed.groups.endHour
|
||||||
|
.split(":")
|
||||||
|
.map((i) => Number.parseInt(i));
|
||||||
|
return {
|
||||||
|
...parsed.groups,
|
||||||
|
startSlot: Math.floor((startHour * 60 + startMin) / MINUTES_PER_SLOT),
|
||||||
|
endSlot: Math.floor((endHour * 60 + endMin) / MINUTES_PER_SLOT),
|
||||||
|
} as unknown as TimetableSlot;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("alpine:init", () => {
|
||||||
|
Alpine.data("timetableGenerator", () => ({
|
||||||
|
content: DEFAULT_TIMETABLE,
|
||||||
|
error: "",
|
||||||
|
displayedWeekdays: [] as WeekDay[],
|
||||||
|
courses: [] as TimetableSlot[],
|
||||||
|
startSlot: 0,
|
||||||
|
endSlot: 0,
|
||||||
|
table: {
|
||||||
|
height: 0,
|
||||||
|
width: 0,
|
||||||
|
},
|
||||||
|
|
||||||
|
colors: {} as Record<string, string>,
|
||||||
|
colorPalette: [
|
||||||
|
"#27ae60",
|
||||||
|
"#2980b9",
|
||||||
|
"#c0392b",
|
||||||
|
"#7f8c8d",
|
||||||
|
"#f1c40f",
|
||||||
|
"#1abc9c",
|
||||||
|
"#95a5a6",
|
||||||
|
"#26C6DA",
|
||||||
|
"#c2185b",
|
||||||
|
"#e64a19",
|
||||||
|
"#1b5e20",
|
||||||
|
],
|
||||||
|
|
||||||
|
generate() {
|
||||||
|
try {
|
||||||
|
this.courses = parseSlots(this.content);
|
||||||
|
} catch {
|
||||||
|
this.error = gettext(
|
||||||
|
"Wrong timetable format. Make sure you copied if from your student folder.",
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// color each UE
|
||||||
|
let colorIndex = 0;
|
||||||
|
for (const slot of this.courses) {
|
||||||
|
if (!this.colors[slot.ueCode]) {
|
||||||
|
this.colors[slot.ueCode] =
|
||||||
|
this.colorPalette[colorIndex % this.colorPalette.length];
|
||||||
|
colorIndex++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.displayedWeekdays = WEEKDAYS.filter((day) =>
|
||||||
|
this.courses.some((slot: TimetableSlot) => slot.weekday === day),
|
||||||
|
);
|
||||||
|
this.startSlot = this.courses.reduce(
|
||||||
|
(acc: number, curr: TimetableSlot) => Math.min(acc, curr.startSlot),
|
||||||
|
25 * 4,
|
||||||
|
);
|
||||||
|
this.endSlot = this.courses.reduce(
|
||||||
|
(acc: number, curr: TimetableSlot) => Math.max(acc, curr.endSlot),
|
||||||
|
1,
|
||||||
|
);
|
||||||
|
this.table.height = SLOT_HEIGHT * (this.endSlot - this.startSlot);
|
||||||
|
this.table.width = SLOT_WIDTH * this.displayedWeekdays.length;
|
||||||
|
},
|
||||||
|
|
||||||
|
getStyle(slot: TimetableSlot) {
|
||||||
|
const hasWeekGroup = slot.weekGroup !== undefined;
|
||||||
|
const width = hasWeekGroup ? SLOT_WIDTH / 2 : SLOT_WIDTH;
|
||||||
|
const leftOffset = slot.weekGroup === "B" ? SLOT_WIDTH / 2 : 0;
|
||||||
|
return {
|
||||||
|
height: `${(slot.endSlot - slot.startSlot) * SLOT_HEIGHT}px`,
|
||||||
|
width: `${width}px`,
|
||||||
|
top: `${(slot.startSlot - this.startSlot) * SLOT_HEIGHT}px`,
|
||||||
|
left: `${this.displayedWeekdays.indexOf(slot.weekday) * SLOT_WIDTH + leftOffset}px`,
|
||||||
|
backgroundColor: this.colors[slot.ueCode],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
getHours(): [string, object][] {
|
||||||
|
let hour: number = Number.parseInt(
|
||||||
|
this.courses
|
||||||
|
.map((c: TimetableSlot) => c.startHour)
|
||||||
|
.reduce((res: string, hour: string) => (hour < res ? hour : res), "24:00")
|
||||||
|
.split(":")[0],
|
||||||
|
);
|
||||||
|
const res: [string, object][] = [];
|
||||||
|
for (let i = 0; i <= this.endSlot - this.startSlot; i += 60 / MINUTES_PER_SLOT) {
|
||||||
|
res.push([`${hour}:00`, { top: `${i * SLOT_HEIGHT}px` }]);
|
||||||
|
hour += 1;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
},
|
||||||
|
|
||||||
|
getWidth() {
|
||||||
|
return this.displayedWeekdays.length * SLOT_WIDTH + 20;
|
||||||
|
},
|
||||||
|
|
||||||
|
async savePng() {
|
||||||
|
const elem = document.getElementById("timetable");
|
||||||
|
const img = (await html2canvas(elem)).toDataURL();
|
||||||
|
const downloadLink = document.createElement("a");
|
||||||
|
downloadLink.href = img;
|
||||||
|
downloadLink.download = "edt.png";
|
||||||
|
document.body.appendChild(downloadLink);
|
||||||
|
downloadLink.click();
|
||||||
|
downloadLink.remove();
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
});
|
||||||
67
timetable/static/timetable/css/generator.scss
Normal file
67
timetable/static/timetable/css/generator.scss
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
@import "core/static/core/colors";
|
||||||
|
|
||||||
|
#timetable {
|
||||||
|
--hour-side-width: 60px;
|
||||||
|
|
||||||
|
display: block;
|
||||||
|
margin: 2em auto;
|
||||||
|
.header {
|
||||||
|
background-color: $white-color;
|
||||||
|
font-weight: bold;
|
||||||
|
box-shadow: none;
|
||||||
|
width: calc(100% - var(--hour-side-width) - 10px);
|
||||||
|
margin-left: var(--hour-side-width);
|
||||||
|
padding-left: 0;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
gap: 0;
|
||||||
|
span {
|
||||||
|
flex: 1;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.content {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
.hours {
|
||||||
|
position: absolute;
|
||||||
|
width: 40px;
|
||||||
|
left: 0;
|
||||||
|
top: -.5em;
|
||||||
|
|
||||||
|
.hour {
|
||||||
|
position: absolute;
|
||||||
|
|
||||||
|
.hour-bar {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
height: 1px;
|
||||||
|
background: lightgray;
|
||||||
|
top: 50%;
|
||||||
|
left: 100%;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.courses {
|
||||||
|
position: absolute;
|
||||||
|
text-align: center;
|
||||||
|
top: 0;
|
||||||
|
left: var(--hour-side-width);
|
||||||
|
|
||||||
|
.slot {
|
||||||
|
background-color: cadetblue;
|
||||||
|
position: absolute;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
.course-type {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
68
timetable/templates/timetable/generator.jinja
Normal file
68
timetable/templates/timetable/generator.jinja
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
{% extends 'core/base.jinja' %}
|
||||||
|
|
||||||
|
{%- block additional_css -%}
|
||||||
|
<link rel="stylesheet" href="{{ static('timetable/css/generator.scss') }}">
|
||||||
|
{%- endblock -%}
|
||||||
|
|
||||||
|
{%- block additional_js -%}
|
||||||
|
<script type="module" src="{{ static('bundled/timetable/generator-index.ts') }}"></script>
|
||||||
|
{%- endblock -%}
|
||||||
|
|
||||||
|
{% block title %}
|
||||||
|
{% trans %}Timetable generator{% endtrans %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div x-data="timetableGenerator">
|
||||||
|
<form @submit.prevent="generate()">
|
||||||
|
<h1>Générateur d'emploi du temps</h1>
|
||||||
|
<div class="alert alert-red" x-show="!!error" x-cloak>
|
||||||
|
<span class="alert-main" x-text="error"></span>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="timetable-input">Colle ton emploi du temps (sans l'entête)</label>
|
||||||
|
<textarea id="timetable-input" cols="30" rows="15" x-model="content"></textarea>
|
||||||
|
</div>
|
||||||
|
<input type="submit" class="btn btn-blue" value="{% trans %}Generate{% endtrans %}">
|
||||||
|
</form>
|
||||||
|
<div
|
||||||
|
id="timetable"
|
||||||
|
x-show="table.height > 0 && table.width > 0"
|
||||||
|
:style="{width: `${table.width+80}px`, height: `${table.height+40}px`}"
|
||||||
|
>
|
||||||
|
<div class="header">
|
||||||
|
<template x-for="weekday in displayedWeekdays">
|
||||||
|
<span x-text="weekday"></span>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
<div class="content">
|
||||||
|
<div class="hours" :height="(endSlot - endSlot%4) - (startSlot - startSlot%4)">
|
||||||
|
<template x-for="[hour, style] in getHours()">
|
||||||
|
<div class="hour" :style="style">
|
||||||
|
<div x-text="hour"></div>
|
||||||
|
<div class="hour-bar" :style="{width: `${getWidth()}px`}"></div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
<div class="courses">
|
||||||
|
<template x-for="course in courses">
|
||||||
|
<div class="slot" :style="getStyle(course)">
|
||||||
|
<span class="course-type" x-text="course.courseType"></span>
|
||||||
|
<span x-text="course.ueCode"></span>
|
||||||
|
<span x-text="`${course.startHour} - ${course.endHour}`"></span>
|
||||||
|
<span x-text="(course.weekGroup ? `\nGroupe ${course.weekGroup}` : '')"></span>
|
||||||
|
<span x-text="course.room"></span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
class="margin-bottom btn btn-blue"
|
||||||
|
@click="savePng"
|
||||||
|
x-show="table.height > 0 && table.width > 0"
|
||||||
|
>
|
||||||
|
{% trans %}Save to PNG{% endtrans %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% endblock content %}
|
||||||
1
timetable/tests.py
Normal file
1
timetable/tests.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
# Create your tests here.
|
||||||
5
timetable/urls.py
Normal file
5
timetable/urls.py
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
from django.urls import path
|
||||||
|
|
||||||
|
from timetable.views import GeneratorView
|
||||||
|
|
||||||
|
urlpatterns = [path("", GeneratorView.as_view(), name="generator")]
|
||||||
8
timetable/views.py
Normal file
8
timetable/views.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# Create your views here.
|
||||||
|
from django.views.generic import TemplateView
|
||||||
|
|
||||||
|
from core.auth.mixins import FormerSubscriberMixin
|
||||||
|
|
||||||
|
|
||||||
|
class GeneratorView(FormerSubscriberMixin, TemplateView):
|
||||||
|
template_name = "timetable/generator.jinja"
|
||||||
Reference in New Issue
Block a user