mirror of
https://github.com/ae-utbm/sith.git
synced 2025-12-17 21:33:21 +00:00
Compare commits
9 Commits
product-fo
...
taiste
| Author | SHA1 | Date | |
|---|---|---|---|
| 3a57439d6e | |||
|
|
fbe5c741d1 | ||
|
749cd067da
|
|||
|
|
1abfbeb76c | ||
|
|
d3edcaff14 | ||
|
|
8c127a96f7 | ||
|
|
55d6e2bbec | ||
|
|
e9fbac8264 | ||
|
|
1911f2e6dd |
@@ -4,15 +4,16 @@ from dateutil.relativedelta import relativedelta
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.sites.models import Site
|
from django.contrib.sites.models import Site
|
||||||
from django.contrib.syndication.views import add_domain
|
from django.contrib.syndication.views import add_domain
|
||||||
from django.db.models import F, QuerySet
|
from django.db.models import Count, OuterRef, QuerySet, Subquery
|
||||||
from django.http import HttpRequest
|
from django.http import HttpRequest
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from ical.calendar import Calendar
|
from ical.calendar import Calendar
|
||||||
from ical.calendar_stream import IcsCalendarStream
|
from ical.calendar_stream import IcsCalendarStream
|
||||||
from ical.event import Event
|
from ical.event import Event
|
||||||
|
from ical.types import Frequency, Recur
|
||||||
|
|
||||||
from com.models import NewsDate
|
from com.models import News, NewsDate
|
||||||
from core.models import User
|
from core.models import User
|
||||||
|
|
||||||
|
|
||||||
@@ -42,9 +43,9 @@ class IcsCalendar:
|
|||||||
with open(cls._INTERNAL_CALENDAR, "wb") as f:
|
with open(cls._INTERNAL_CALENDAR, "wb") as f:
|
||||||
_ = f.write(
|
_ = f.write(
|
||||||
cls.ics_from_queryset(
|
cls.ics_from_queryset(
|
||||||
NewsDate.objects.filter(
|
News.objects.filter(
|
||||||
news__is_published=True,
|
is_published=True,
|
||||||
end_date__gte=timezone.now() - (relativedelta(months=6)),
|
dates__end_date__gte=timezone.now() - relativedelta(months=6),
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@@ -53,24 +54,35 @@ class IcsCalendar:
|
|||||||
@classmethod
|
@classmethod
|
||||||
def get_unpublished(cls, user: User) -> bytes:
|
def get_unpublished(cls, user: User) -> bytes:
|
||||||
return cls.ics_from_queryset(
|
return cls.ics_from_queryset(
|
||||||
NewsDate.objects.viewable_by(user).filter(
|
News.objects.viewable_by(user).filter(
|
||||||
news__is_published=False,
|
is_published=False,
|
||||||
end_date__gte=timezone.now() - (relativedelta(months=6)),
|
dates__end_date__gte=timezone.now() - relativedelta(months=6),
|
||||||
),
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def ics_from_queryset(cls, queryset: QuerySet[NewsDate]) -> bytes:
|
def ics_from_queryset(cls, queryset: QuerySet[News]) -> bytes:
|
||||||
calendar = Calendar()
|
calendar = Calendar()
|
||||||
for news_date in queryset.annotate(news_title=F("news__title")):
|
date_subquery = NewsDate.objects.filter(news=OuterRef("pk")).order_by(
|
||||||
|
"start_date"
|
||||||
|
)
|
||||||
|
queryset = queryset.annotate(
|
||||||
|
start=Subquery(date_subquery.values("start_date")[:1]),
|
||||||
|
end=Subquery(date_subquery.values("end_date")[:1]),
|
||||||
|
nb_dates=Count("dates"),
|
||||||
|
)
|
||||||
|
for news in queryset:
|
||||||
event = Event(
|
event = Event(
|
||||||
summary=news_date.news_title,
|
summary=news.title,
|
||||||
start=news_date.start_date,
|
description=news.summary,
|
||||||
end=news_date.end_date,
|
dtstart=news.start,
|
||||||
|
dtend=news.end,
|
||||||
url=as_absolute_url(
|
url=as_absolute_url(
|
||||||
reverse("com:news_detail", kwargs={"news_id": news_date.news_id})
|
reverse("com:news_detail", kwargs={"news_id": news.id})
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
if news.nb_dates > 1:
|
||||||
|
event.rrule = Recur(freq=Frequency.WEEKLY, count=news.nb_dates)
|
||||||
calendar.events.append(event)
|
calendar.events.append(event)
|
||||||
|
|
||||||
return IcsCalendarStream.calendar_to_ics(calendar).encode("utf-8")
|
return IcsCalendarStream.calendar_to_ics(calendar).encode("utf-8")
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { makeUrl } from "#core:utils/api";
|
import { makeUrl } from "#core:utils/api";
|
||||||
import { inheritHtmlElement, registerComponent } from "#core:utils/web-components";
|
import { inheritHtmlElement, registerComponent } from "#core:utils/web-components";
|
||||||
import { Calendar, type EventClickArg } from "@fullcalendar/core";
|
import { Calendar, type EventClickArg, type EventContentArg } from "@fullcalendar/core";
|
||||||
import type { EventImpl } from "@fullcalendar/core/internal";
|
import type { EventImpl } from "@fullcalendar/core/internal";
|
||||||
import enLocale from "@fullcalendar/core/locales/en-gb";
|
import enLocale from "@fullcalendar/core/locales/en-gb";
|
||||||
import frLocale from "@fullcalendar/core/locales/fr";
|
import frLocale from "@fullcalendar/core/locales/fr";
|
||||||
@@ -25,6 +25,11 @@ export class IcsCalendar extends inheritHtmlElement("div") {
|
|||||||
private canDelete = false;
|
private canDelete = false;
|
||||||
private helpUrl = "";
|
private helpUrl = "";
|
||||||
|
|
||||||
|
// Hack variable to detect recurring events
|
||||||
|
// The underlying ics library doesn't include any info about rrules
|
||||||
|
// That's why we have to detect those events ourselves
|
||||||
|
private recurrenceMap: Map<string, EventImpl> = new Map();
|
||||||
|
|
||||||
attributeChangedCallback(name: string, _oldValue?: string, newValue?: string) {
|
attributeChangedCallback(name: string, _oldValue?: string, newValue?: string) {
|
||||||
if (name === "locale") {
|
if (name === "locale") {
|
||||||
this.locale = newValue;
|
this.locale = newValue;
|
||||||
@@ -95,6 +100,7 @@ export class IcsCalendar extends inheritHtmlElement("div") {
|
|||||||
|
|
||||||
refreshEvents() {
|
refreshEvents() {
|
||||||
this.click(); // Remove focus from popup
|
this.click(); // Remove focus from popup
|
||||||
|
this.recurrenceMap.clear(); // Avoid double detection of the same non recurring event
|
||||||
this.calendar.refetchEvents();
|
this.calendar.refetchEvents();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -153,12 +159,24 @@ export class IcsCalendar extends inheritHtmlElement("div") {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async getEventSources() {
|
async getEventSources() {
|
||||||
|
const tagRecurringEvents = (eventData: EventImpl) => {
|
||||||
|
// This functions tags events with a similar event url
|
||||||
|
// We rely on the fact that the event url is always the same
|
||||||
|
// for recurring events and always different for single events
|
||||||
|
const firstEvent = this.recurrenceMap.get(eventData.url);
|
||||||
|
if (firstEvent !== undefined) {
|
||||||
|
eventData.extendedProps.isRecurring = true;
|
||||||
|
firstEvent.extendedProps.isRecurring = true; // Don't forget the first event
|
||||||
|
}
|
||||||
|
this.recurrenceMap.set(eventData.url, eventData);
|
||||||
|
};
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
url: `${await makeUrl(calendarCalendarInternal)}`,
|
url: `${await makeUrl(calendarCalendarInternal)}`,
|
||||||
format: "ics",
|
format: "ics",
|
||||||
className: "internal",
|
className: "internal",
|
||||||
cache: false,
|
cache: false,
|
||||||
|
eventDataTransform: tagRecurringEvents,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
url: `${await makeUrl(calendarCalendarUnpublished)}`,
|
url: `${await makeUrl(calendarCalendarUnpublished)}`,
|
||||||
@@ -166,6 +184,7 @@ export class IcsCalendar extends inheritHtmlElement("div") {
|
|||||||
color: "red",
|
color: "red",
|
||||||
className: "unpublished",
|
className: "unpublished",
|
||||||
cache: false,
|
cache: false,
|
||||||
|
eventDataTransform: tagRecurringEvents,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -361,6 +380,14 @@ export class IcsCalendar extends inheritHtmlElement("div") {
|
|||||||
event.jsEvent.preventDefault();
|
event.jsEvent.preventDefault();
|
||||||
this.createEventDetailPopup(event);
|
this.createEventDetailPopup(event);
|
||||||
},
|
},
|
||||||
|
eventClassNames: (classNamesEvent: EventContentArg) => {
|
||||||
|
const classes: string[] = [];
|
||||||
|
if (classNamesEvent.event.extendedProps?.isRecurring) {
|
||||||
|
classes.push("recurring");
|
||||||
|
}
|
||||||
|
|
||||||
|
return classes;
|
||||||
|
},
|
||||||
});
|
});
|
||||||
this.calendar.render();
|
this.calendar.render();
|
||||||
|
|
||||||
|
|||||||
@@ -18,6 +18,8 @@
|
|||||||
--event-details-border-radius: 4px;
|
--event-details-border-radius: 4px;
|
||||||
--event-details-box-shadow: 0px 6px 20px 4px rgb(0 0 0 / 16%);
|
--event-details-box-shadow: 0px 6px 20px 4px rgb(0 0 0 / 16%);
|
||||||
--event-details-max-width: 600px;
|
--event-details-max-width: 600px;
|
||||||
|
--event-recurring-internal-color: #6f69cd;
|
||||||
|
--event-recurring-unpublished-color: orange;
|
||||||
}
|
}
|
||||||
|
|
||||||
ics-calendar {
|
ics-calendar {
|
||||||
@@ -146,4 +148,29 @@ ics-calendar {
|
|||||||
.tooltip.calendar-copy-tooltip.text-copied {
|
.tooltip.calendar-copy-tooltip.text-copied {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
transition: opacity 500ms ease-out;
|
transition: opacity 500ms ease-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We have to override the color set by the lib in the html
|
||||||
|
// Hence the !important tag everywhere
|
||||||
|
.internal.recurring {
|
||||||
|
.fc-daygrid-event-dot {
|
||||||
|
border-color: var(--event-recurring-internal-color) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.fc-daygrid-block-event {
|
||||||
|
background-color: var(--event-recurring-internal-color) !important;
|
||||||
|
border-color: var(--event-recurring-internal-color) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.unpublished.recurring {
|
||||||
|
.fc-daygrid-event-dot {
|
||||||
|
border-color: var(--event-recurring-unpublished-color) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.fc-daygrid-block-event {
|
||||||
|
background-color: var(--event-recurring-unpublished-color) !important;
|
||||||
|
border-color: var(--event-recurring-unpublished-color) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -195,18 +195,18 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&.delete {
|
form .link-like {
|
||||||
margin-top: 10px;
|
margin-top: 10px;
|
||||||
display: block;
|
display: block;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: orangered;
|
color: orangered;
|
||||||
|
|
||||||
@media (max-width: 375px) {
|
@media (max-width: 375px) {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -78,12 +78,6 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
{% endmacro %}
|
{% endmacro %}
|
||||||
|
|
||||||
{% macro delete_godfather(user, profile, godfather, is_father) %}
|
|
||||||
{% if user == profile or user.is_root or user.is_board_member %}
|
|
||||||
<a class="delete" href="{{ url("core:user_godfathers_delete", user_id=profile.id, godfather_id=godfather.id, is_father=is_father) }}">{% trans %}Delete{% endtrans %}</a>
|
|
||||||
{% endif %}
|
|
||||||
{% endmacro %}
|
|
||||||
|
|
||||||
{% macro paginate_alpine(page, nb_pages) %}
|
{% macro paginate_alpine(page, nb_pages) %}
|
||||||
{# Add pagination buttons for ajax based content with alpine
|
{# Add pagination buttons for ajax based content with alpine
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
{% block content %}
|
{% block content %}
|
||||||
|
|
||||||
{% if target %}
|
{% if target %}
|
||||||
<p>{% trans user=target.get_display_name() %}Change password for {{ user }}{% endtrans %}</p>
|
<p>{% trans user=form.user.get_display_name() %}Change password for {{ user }}{% endtrans %}</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<form method="post" action="">
|
<form method="post" action="">
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
|
|||||||
@@ -29,7 +29,16 @@
|
|||||||
<a href="{{ url('core:user_godfathers', user_id=u.id) }}" class="mini_profile_link">
|
<a href="{{ url('core:user_godfathers', user_id=u.id) }}" class="mini_profile_link">
|
||||||
{{ u.get_mini_item() | safe }}
|
{{ u.get_mini_item() | safe }}
|
||||||
</a>
|
</a>
|
||||||
{{ delete_godfather(user, profile, u, True) }}
|
{% if user == profile or user.is_root or user.is_board_member %}
|
||||||
|
<form
|
||||||
|
method="post"
|
||||||
|
class="no-margin"
|
||||||
|
action="{{ url("core:user_godfathers_delete", user_id=profile.id, godfather_id=u.id, is_father=True) }}"
|
||||||
|
>
|
||||||
|
{% csrf_token %}
|
||||||
|
<input type="submit" class="link-like" value="{% trans %}Delete{% endtrans %}">
|
||||||
|
</form>
|
||||||
|
{% endif %}
|
||||||
</li>
|
</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
@@ -46,7 +55,16 @@
|
|||||||
<a href="{{ url('core:user_godfathers', user_id=u.id) }}" class="mini_profile_link">
|
<a href="{{ url('core:user_godfathers', user_id=u.id) }}" class="mini_profile_link">
|
||||||
{{ u.get_mini_item()|safe }}
|
{{ u.get_mini_item()|safe }}
|
||||||
</a>
|
</a>
|
||||||
{{ delete_godfather(user, profile, u, False) }}
|
{% if user == profile or user.is_root or user.is_board_member %}
|
||||||
|
<form
|
||||||
|
method="post"
|
||||||
|
class="no-margin"
|
||||||
|
action="{{ url("core:user_godfathers_delete", user_id=profile.id, godfather_id=u.id, is_father=False) }}"
|
||||||
|
>
|
||||||
|
{% csrf_token %}
|
||||||
|
<input type="submit" class="link-like link-red" value="{% trans %}Delete{% endtrans %}">
|
||||||
|
</form>
|
||||||
|
{% endif %}
|
||||||
</li>
|
</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
|
|||||||
@@ -427,6 +427,19 @@ class TestUserQuerySetViewableBy:
|
|||||||
assert not viewable.exists()
|
assert not viewable.exists()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_user_preferences(client: Client):
|
||||||
|
user = subscriber_user.make()
|
||||||
|
client.force_login(user)
|
||||||
|
url = reverse("core:user_prefs", kwargs={"user_id": user.id})
|
||||||
|
response = client.get(url)
|
||||||
|
assert response.status_code == 200
|
||||||
|
response = client.post(url, {"notify_on_click": "true"})
|
||||||
|
assertRedirects(response, url)
|
||||||
|
user.preferences.refresh_from_db()
|
||||||
|
assert user.preferences.notify_on_click is True
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
def test_user_stats(client: Client):
|
def test_user_stats(client: Client):
|
||||||
user = subscriber_user.make()
|
user = subscriber_user.make()
|
||||||
@@ -450,3 +463,68 @@ def test_user_stats(client: Client):
|
|||||||
client.force_login(user)
|
client.force_login(user)
|
||||||
response = client.get(reverse("core:user_stats", kwargs={"user_id": user.id}))
|
response = client.get(reverse("core:user_stats", kwargs={"user_id": user.id}))
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
class TestChangeUserPassword:
|
||||||
|
def test_as_root(self, client: Client, admin_user: User):
|
||||||
|
client.force_login(admin_user)
|
||||||
|
user = subscriber_user.make()
|
||||||
|
url = reverse("core:password_root_change", kwargs={"user_id": user.id})
|
||||||
|
response = client.get(url)
|
||||||
|
assert response.status_code == 200
|
||||||
|
response = client.post(
|
||||||
|
url, {"new_password1": "poutou", "new_password2": "poutou"}
|
||||||
|
)
|
||||||
|
assertRedirects(response, reverse("core:password_change_done"))
|
||||||
|
user.refresh_from_db()
|
||||||
|
assert user.check_password("poutou") is True
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
class TestUserGodfather:
|
||||||
|
@pytest.mark.parametrize("godfather", [True, False])
|
||||||
|
def test_add_family(self, client: Client, godfather):
|
||||||
|
user = subscriber_user.make()
|
||||||
|
other_user = subscriber_user.make()
|
||||||
|
client.force_login(user)
|
||||||
|
url = reverse("core:user_godfathers", kwargs={"user_id": user.id})
|
||||||
|
response = client.get(url)
|
||||||
|
assert response.status_code == 200
|
||||||
|
response = client.post(
|
||||||
|
url,
|
||||||
|
{"type": "godfather" if godfather else "godchild", "user": other_user.id},
|
||||||
|
)
|
||||||
|
assertRedirects(response, url)
|
||||||
|
if godfather:
|
||||||
|
assert user.godfathers.contains(other_user)
|
||||||
|
else:
|
||||||
|
assert user.godchildren.contains(other_user)
|
||||||
|
|
||||||
|
def test_tree(self, client: Client):
|
||||||
|
user = subscriber_user.make()
|
||||||
|
client.force_login(user)
|
||||||
|
response = client.get(
|
||||||
|
reverse("core:user_godfathers_tree", kwargs={"user_id": user.id})
|
||||||
|
)
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
def test_remove_family(self, client: Client):
|
||||||
|
user = subscriber_user.make()
|
||||||
|
other_user = subscriber_user.make()
|
||||||
|
user.godfathers.add(other_user)
|
||||||
|
client.force_login(user)
|
||||||
|
response = client.post(
|
||||||
|
reverse(
|
||||||
|
"core:user_godfathers_delete",
|
||||||
|
kwargs={
|
||||||
|
"user_id": user.id,
|
||||||
|
"godfather_id": other_user.id,
|
||||||
|
"is_father": True,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
)
|
||||||
|
assertRedirects(
|
||||||
|
response, reverse("core:user_godfathers", kwargs={"user_id": user.id})
|
||||||
|
)
|
||||||
|
assert not user.godfathers.contains(other_user)
|
||||||
|
|||||||
@@ -54,6 +54,7 @@ from core.views import (
|
|||||||
PagePropView,
|
PagePropView,
|
||||||
PageRevView,
|
PageRevView,
|
||||||
PageView,
|
PageView,
|
||||||
|
PasswordRootChangeView,
|
||||||
SearchView,
|
SearchView,
|
||||||
SithLoginView,
|
SithLoginView,
|
||||||
SithPasswordChangeDoneView,
|
SithPasswordChangeDoneView,
|
||||||
@@ -80,7 +81,6 @@ from core.views import (
|
|||||||
delete_user_godfather,
|
delete_user_godfather,
|
||||||
logout,
|
logout,
|
||||||
notification,
|
notification,
|
||||||
password_root_change,
|
|
||||||
send_file,
|
send_file,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -100,7 +100,7 @@ urlpatterns = [
|
|||||||
path("password_change/", SithPasswordChangeView.as_view(), name="password_change"),
|
path("password_change/", SithPasswordChangeView.as_view(), name="password_change"),
|
||||||
path(
|
path(
|
||||||
"password_change/<int:user_id>/",
|
"password_change/<int:user_id>/",
|
||||||
password_root_change,
|
PasswordRootChangeView.as_view(),
|
||||||
name="password_root_change",
|
name="password_root_change",
|
||||||
),
|
),
|
||||||
path(
|
path(
|
||||||
|
|||||||
@@ -303,7 +303,6 @@ class UserGodfathersForm(forms.Form):
|
|||||||
)
|
)
|
||||||
user = forms.ModelChoiceField(
|
user = forms.ModelChoiceField(
|
||||||
label=_("Select user"),
|
label=_("Select user"),
|
||||||
help_text=None,
|
|
||||||
required=True,
|
required=True,
|
||||||
widget=AutoCompleteSelectUser,
|
widget=AutoCompleteSelectUser,
|
||||||
queryset=User.objects.all(),
|
queryset=User.objects.all(),
|
||||||
@@ -315,8 +314,6 @@ class UserGodfathersForm(forms.Form):
|
|||||||
|
|
||||||
def clean_user(self):
|
def clean_user(self):
|
||||||
other_user = self.cleaned_data.get("user")
|
other_user = self.cleaned_data.get("user")
|
||||||
if not other_user:
|
|
||||||
raise ValidationError(_("This user does not exist"))
|
|
||||||
if other_user == self.target_user:
|
if other_user == self.target_user:
|
||||||
raise ValidationError(_("You cannot be related to yourself"))
|
raise ValidationError(_("You cannot be related to yourself"))
|
||||||
return other_user
|
return other_user
|
||||||
|
|||||||
@@ -29,8 +29,9 @@ from operator import itemgetter
|
|||||||
from smtplib import SMTPException
|
from smtplib import SMTPException
|
||||||
|
|
||||||
from django.contrib.auth import login, views
|
from django.contrib.auth import login, views
|
||||||
from django.contrib.auth.forms import PasswordChangeForm
|
from django.contrib.auth.decorators import login_required
|
||||||
from django.contrib.auth.mixins import LoginRequiredMixin
|
from django.contrib.auth.forms import PasswordChangeForm, SetPasswordForm
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin, UserPassesTestMixin
|
||||||
from django.core.exceptions import PermissionDenied
|
from django.core.exceptions import PermissionDenied
|
||||||
from django.db.models import DateField, F, QuerySet, Sum
|
from django.db.models import DateField, F, QuerySet, Sum
|
||||||
from django.db.models.functions import Trunc
|
from django.db.models.functions import Trunc
|
||||||
@@ -38,11 +39,11 @@ from django.forms.models import modelform_factory
|
|||||||
from django.http import Http404
|
from django.http import Http404
|
||||||
from django.shortcuts import get_object_or_404, redirect
|
from django.shortcuts import get_object_or_404, redirect
|
||||||
from django.template.loader import render_to_string
|
from django.template.loader import render_to_string
|
||||||
from django.template.response import TemplateResponse
|
|
||||||
from django.urls import reverse, reverse_lazy
|
from django.urls import reverse, reverse_lazy
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
from django.utils.safestring import SafeString
|
from django.utils.safestring import SafeString
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.decorators.http import require_POST
|
||||||
from django.views.generic import (
|
from django.views.generic import (
|
||||||
CreateView,
|
CreateView,
|
||||||
DeleteView,
|
DeleteView,
|
||||||
@@ -98,21 +99,23 @@ def logout(request):
|
|||||||
return views.logout_then_login(request)
|
return views.logout_then_login(request)
|
||||||
|
|
||||||
|
|
||||||
def password_root_change(request, user_id):
|
class PasswordRootChangeView(UserPassesTestMixin, FormView):
|
||||||
"""Allows a root user to change someone's password."""
|
"""Allows a root user to change someone's password."""
|
||||||
if not request.user.is_root:
|
|
||||||
raise PermissionDenied
|
template_name = "core/password_change.jinja"
|
||||||
user = get_object_or_404(User, id=user_id)
|
form_class = SetPasswordForm
|
||||||
if request.method == "POST":
|
success_url = reverse_lazy("core:password_change_done")
|
||||||
form = views.SetPasswordForm(user=user, data=request.POST)
|
|
||||||
if form.is_valid():
|
def test_func(self):
|
||||||
form.save()
|
return self.request.user.is_root
|
||||||
return redirect("core:password_change_done")
|
|
||||||
else:
|
def get_form_kwargs(self):
|
||||||
form = views.SetPasswordForm(user=user)
|
user = get_object_or_404(User, id=self.kwargs["user_id"])
|
||||||
return TemplateResponse(
|
return super().get_form_kwargs() | {"user": user}
|
||||||
request, "core/password_change.jinja", {"form": form, "target": user}
|
|
||||||
)
|
def form_valid(self, form: SetPasswordForm):
|
||||||
|
form.save()
|
||||||
|
return super().form_valid(form)
|
||||||
|
|
||||||
|
|
||||||
@method_decorator(check_honeypot, name="post")
|
@method_decorator(check_honeypot, name="post")
|
||||||
@@ -287,10 +290,12 @@ class UserView(UserTabsMixin, CanViewMixin, DetailView):
|
|||||||
return kwargs
|
return kwargs
|
||||||
|
|
||||||
|
|
||||||
|
@require_POST
|
||||||
|
@login_required
|
||||||
def delete_user_godfather(request, user_id, godfather_id, is_father):
|
def delete_user_godfather(request, user_id, godfather_id, is_father):
|
||||||
user_is_admin = request.user.is_root or request.user.is_board_member
|
user_is_admin = request.user.is_root or request.user.is_board_member
|
||||||
if user_id != request.user.id and not user_is_admin:
|
if user_id != request.user.id and not user_is_admin:
|
||||||
raise PermissionDenied()
|
raise PermissionDenied
|
||||||
user = get_object_or_404(User, id=user_id)
|
user = get_object_or_404(User, id=user_id)
|
||||||
to_remove = get_object_or_404(User, id=godfather_id)
|
to_remove = get_object_or_404(User, id=godfather_id)
|
||||||
if is_father:
|
if is_father:
|
||||||
@@ -417,7 +422,6 @@ class UserUpdateProfileView(UserTabsMixin, CanEditMixin, UpdateView):
|
|||||||
form_class = UserProfileForm
|
form_class = UserProfileForm
|
||||||
current_tab = "edit"
|
current_tab = "edit"
|
||||||
edit_once = ["profile_pict", "date_of_birth", "first_name", "last_name"]
|
edit_once = ["profile_pict", "date_of_birth", "first_name", "last_name"]
|
||||||
board_only = []
|
|
||||||
|
|
||||||
def remove_restricted_fields(self, request):
|
def remove_restricted_fields(self, request):
|
||||||
"""Removes edit_once and board_only fields."""
|
"""Removes edit_once and board_only fields."""
|
||||||
@@ -426,9 +430,6 @@ class UserUpdateProfileView(UserTabsMixin, CanEditMixin, UpdateView):
|
|||||||
request.user.is_board_member or request.user.is_root
|
request.user.is_board_member or request.user.is_root
|
||||||
):
|
):
|
||||||
self.form.fields.pop(i, None)
|
self.form.fields.pop(i, None)
|
||||||
for i in self.board_only:
|
|
||||||
if not (request.user.is_board_member or request.user.is_root):
|
|
||||||
self.form.fields.pop(i, None)
|
|
||||||
|
|
||||||
def get(self, request, *args, **kwargs):
|
def get(self, request, *args, **kwargs):
|
||||||
self.object = self.get_object()
|
self.object = self.get_object()
|
||||||
@@ -480,10 +481,10 @@ class UserPreferencesView(UserTabsMixin, UseFragmentsMixin, CanEditMixin, Update
|
|||||||
current_tab = "prefs"
|
current_tab = "prefs"
|
||||||
|
|
||||||
def get_form_kwargs(self):
|
def get_form_kwargs(self):
|
||||||
kwargs = super().get_form_kwargs()
|
return super().get_form_kwargs() | {"instance": self.object.preferences}
|
||||||
pref = self.object.preferences
|
|
||||||
kwargs.update({"instance": pref})
|
def get_success_url(self):
|
||||||
return kwargs
|
return self.request.path
|
||||||
|
|
||||||
def get_fragment_context_data(self) -> dict[str, SafeString]:
|
def get_fragment_context_data(self) -> dict[str, SafeString]:
|
||||||
# Avoid cyclic import error
|
# Avoid cyclic import error
|
||||||
|
|||||||
Reference in New Issue
Block a user