Compare commits
10 Commits
671072864d
...
detail_res
| Author | SHA1 | Date | |
|---|---|---|---|
| 267994044c | |||
| a4b4a68dd2 | |||
| 07dc097d27 | |||
| 4146563f41 | |||
| c0cdca48fa | |||
| 9d9e6c6549 | |||
| 784822478e | |||
| ab87baaa3a | |||
| 6026af5498 | |||
| 91bdd791f1 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -4,3 +4,4 @@ media/
|
||||
staticfiles/
|
||||
.env
|
||||
migrations/
|
||||
*.pyc
|
||||
|
||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -135,66 +135,69 @@ def demander_conge(request):
|
||||
|
||||
@login_required
|
||||
def liste_demande_conges(request):
|
||||
"""Vue de liste des demandes de congés en attente de validation selon le statut de l'utilisateur actuel"""
|
||||
|
||||
try:
|
||||
employe = Employe.objects.get(user__username = request.user)
|
||||
employe = Employe.objects.get(user=request.user)
|
||||
except Employe.DoesNotExist:
|
||||
return JsonResponse({
|
||||
"success": False,
|
||||
"message": "Votre profil Utilisateur n'est lié à aucun profil Employé. Veuillez contacter l'administrateur."
|
||||
"message": "Profil employé introuvable"
|
||||
})
|
||||
|
||||
try:
|
||||
affectation = Affectation.objects.get(
|
||||
employe=employe,
|
||||
date_fin_daffectation__gte=timezone.now().date()
|
||||
)
|
||||
except Affectation.DoesNotExist:
|
||||
affectation = None
|
||||
affectation = Affectation.objects.filter(
|
||||
employe=employe,
|
||||
date_fin_daffectation__gte=timezone.now().date()
|
||||
).first()
|
||||
|
||||
is_direction = employe.user.groups.filter(name='direction').exists()
|
||||
|
||||
if employe.chef:
|
||||
print("chef")
|
||||
conges_en_attente = Conge.objects.filter(
|
||||
employe__departement = employe.departement,
|
||||
validation_hierarchique = None
|
||||
|
||||
conges = Conge.objects.filter(
|
||||
Q(employe__departement=employe.departement) |
|
||||
Q(employe=employe)
|
||||
).order_by('-date_demande')
|
||||
|
||||
elif affectation and affectation.role == "chef_projet":
|
||||
|
||||
employes_du_projet = Affectation.objects.filter(
|
||||
projet = affectation.projet,
|
||||
date_fin_daffectation__gte = timezone.now().date()
|
||||
).values('employe')
|
||||
projet=affectation.projet,
|
||||
date_fin_daffectation__gte=timezone.now().date()
|
||||
).values_list('employe', flat=True)
|
||||
|
||||
conges_en_attente = Conge.objects.filter(
|
||||
employe__in = employes_du_projet,
|
||||
validation_hierarchique = None
|
||||
conges = Conge.objects.filter(
|
||||
Q(employe__in=employes_du_projet) |
|
||||
Q(employe=employe)
|
||||
).order_by('-date_demande')
|
||||
|
||||
elif 'direction' in employe.user.groups.values_list('name', flat=True):
|
||||
conges_en_attente = Conge.objects.filter(
|
||||
validation_hierarchique = True,
|
||||
validation_direction = None
|
||||
).order_by('-date_demande')
|
||||
|
||||
elif is_direction:
|
||||
|
||||
conges = Conge.objects.filter(
|
||||
Q(validation_hierarchique=True) |
|
||||
Q(employe__user__groups__name='direction')
|
||||
).distinct().order_by('-date_demande')
|
||||
|
||||
else:
|
||||
conges_en_attente = Conge.objects.filter(
|
||||
employe__user__username = request.user
|
||||
|
||||
conges = Conge.objects.filter(
|
||||
employe=employe
|
||||
).order_by('-date_demande')
|
||||
|
||||
return JsonResponse({
|
||||
"success": True,
|
||||
"data":[
|
||||
{
|
||||
**model_to_dict(conge),
|
||||
"prenom_nom": f"{conge.employe.user.first_name} {conge.employe.user.last_name}",
|
||||
"date_demande": conge.date_demande,
|
||||
"nombre_jours": conge.nombre_jours,
|
||||
"type": dict(conge.TYPE_CHOICES).get(conge.type),
|
||||
"solde_conge": fonctions_utilitaire.solde_conge(conge.employe)["quota_annuel"]
|
||||
}
|
||||
for conge in conges_en_attente]},
|
||||
safe=False
|
||||
)
|
||||
"data": [
|
||||
{
|
||||
**model_to_dict(conge),
|
||||
"prenom_nom": f"{conge.employe.user.first_name} {conge.employe.user.last_name}",
|
||||
"date_demande": conge.date_demande,
|
||||
"nombre_jours": conge.nombre_jours,
|
||||
"type": dict(conge.TYPE_CHOICES).get(conge.type),
|
||||
"solde_conge": fonctions_utilitaire.solde_conge(conge.employe)["quota_annuel"]
|
||||
}
|
||||
for conge in conges
|
||||
]
|
||||
})
|
||||
|
||||
@login_required
|
||||
def validation_de_conge(request):
|
||||
|
||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -19,12 +19,13 @@
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
|
||||
{% if not expiration_contrat %}
|
||||
<div class="alert alert-danger fade show alert-dismissible mt-2">
|
||||
{% if not has_contrat %}
|
||||
<div class="alert alert-danger mt-2">
|
||||
<strong>Important :</strong> Les informations sur votre contrat n'ont pas été renseignées, veuillez contacter les ressources humaines.
|
||||
</div>
|
||||
{% elif contrat_nb_jours_restant %}
|
||||
<div class="alert alert-danger fade show alert-dismissible mt-2">
|
||||
|
||||
{% elif expiration_contrat %}
|
||||
<div class="alert alert-warning mt-2">
|
||||
<strong>Important :</strong> Votre contrat de travail expire dans {{ contrat_nb_jours_restant }} jours, veuillez contacter les ressources humaines.
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
@@ -0,0 +1,24 @@
|
||||
<div class="modal fade" id="modalContratsEmploye" tabindex="-1">
|
||||
<div class="modal-dialog modal-lg">
|
||||
<div class="modal-content">
|
||||
|
||||
<div class="modal-header">
|
||||
<h5 class="modal-title">📄 Contrats de l'employé</h5>
|
||||
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
|
||||
</div>
|
||||
|
||||
<div class="modal-body">
|
||||
<div id="contrats-employe-body">
|
||||
<!-- contenu dynamique JS -->
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="modal-footer">
|
||||
<button class="btn btn-secondary" data-bs-dismiss="modal">
|
||||
Fermer
|
||||
</button>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
Binary file not shown.
@@ -129,26 +129,41 @@ def mon_profil(request):
|
||||
try:
|
||||
employe = Employe.objects.get(user__username=request.user)
|
||||
except Employe.DoesNotExist:
|
||||
messages.error(request, "Impossible d'acceder au menu 'Mon profil' car votre profil Utilisateur n'est lié à aucun profil Employé. Veuillez contacter l'Administrateur.")
|
||||
messages.error(
|
||||
request,
|
||||
"Impossible d'acceder au menu 'Mon profil' car votre profil Utilisateur n'est lié à aucun profil Employé. Veuillez contacter l'Administrateur."
|
||||
)
|
||||
return redirect("gestion_conges:conge")
|
||||
|
||||
|
||||
contrats = Contrat.objects.filter(employe=employe, statut='actif').first()
|
||||
|
||||
projets = Affectation.objects.filter(
|
||||
employe = employe,
|
||||
date_fin_daffectation__gte = timezone.now().date()
|
||||
employe=employe,
|
||||
date_fin_daffectation__gte=timezone.now().date()
|
||||
).select_related('projet')
|
||||
|
||||
has_contrat = contrats is not None
|
||||
expiration_contrat = False
|
||||
contrat_nb_jours_restant = None
|
||||
|
||||
if contrats:
|
||||
nb_jours = contrats.nombre_jours_restant
|
||||
contrat_nb_jours_restant = nb_jours
|
||||
expiration_contrat = nb_jours <= fonctions_utilitaire.DUREE_FIN_CONTRAT
|
||||
|
||||
return render(
|
||||
request,
|
||||
'gestion_employe/monprofil.html',
|
||||
request,
|
||||
'gestion_employe/monprofil.html',
|
||||
{
|
||||
'employe': employe,
|
||||
|
||||
'contrats': [{
|
||||
**model_to_dict(contrats),
|
||||
"type_contrat": dict(Contrat.TYPE_CONTRAT).get(contrats.type_contrat),
|
||||
"statut": dict(Contrat.STATUT_CONTRAT).get(contrats.statut),
|
||||
"fichier_contrat": contrats.fichier_contrat.url if contrats.fichier_contrat else "",
|
||||
} if contrats else []],
|
||||
}] if contrats else [],
|
||||
|
||||
'projets': [
|
||||
{
|
||||
**model_to_dict(a.projet),
|
||||
@@ -158,12 +173,13 @@ def mon_profil(request):
|
||||
"pourcentage_temps_affectation": a.pourcentage_temps_affectation
|
||||
} for a in projets
|
||||
],
|
||||
|
||||
"formation_form": FormationForm(),
|
||||
"expiration_contrat": contrats.nombre_jours_restant <= fonctions_utilitaire.DUREE_FIN_CONTRAT if contrats else False,
|
||||
"contrat_nb_jours_restant": contrats.nombre_jours_restant if contrats else None
|
||||
"has_contrat": has_contrat,
|
||||
"expiration_contrat": expiration_contrat,
|
||||
"contrat_nb_jours_restant": contrat_nb_jours_restant
|
||||
}
|
||||
)
|
||||
|
||||
@login_required
|
||||
def modifier_mot_passe(request):
|
||||
"""Vue pour permettre à un utilisateur de modifier son mot de passe et ses informations de profil"""
|
||||
@@ -261,6 +277,7 @@ def creation_contrat(request):
|
||||
except Employe.DoesNotExist:
|
||||
messages.error(request, "Employé non trouvé.")
|
||||
return redirect('employe-index')
|
||||
|
||||
contrat_actif = Contrat.objects.filter(
|
||||
employe=employe,
|
||||
date_fin__gte=date.today()
|
||||
|
||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -1,4 +1,5 @@
|
||||
const btnEnregistrerBailleur = document.getElementById('btnEnregistrerBailleur');
|
||||
let table;
|
||||
|
||||
btnEnregistrerBailleur.addEventListener('click', function() {
|
||||
const form = document.getElementById('formBailleur');
|
||||
@@ -20,4 +21,27 @@ btnEnregistrerBailleur.addEventListener('click', function() {
|
||||
alert('Ce bailleur existe déjà dans la base de données.');
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
|
||||
table = new Tabulator("#table-bailleurs", {
|
||||
ajaxURL: "/gestion-projet/bailleurs/",
|
||||
layout: "fitColumns",
|
||||
pagination: "local",
|
||||
paginationSize: 5,
|
||||
|
||||
columns: [
|
||||
{title: "#", formatter: "rownum", width: 60},
|
||||
{title: "Organisme", field: "nom_organisme"},
|
||||
{title: "Contact", field: "contact"},
|
||||
{title: "Email", field: "email"},
|
||||
{title: "Pays", field: "pays"},
|
||||
],
|
||||
rowDblClick: function(e, row) {
|
||||
const data = row.getData();
|
||||
|
||||
if (confirm(`Voulez-vous vraiment supprimer ${data.nom_organisme} ?`)) {
|
||||
supprimerBailleur(data.id);
|
||||
}
|
||||
}
|
||||
});
|
||||
@@ -1,19 +1,49 @@
|
||||
<div class="modal fade" id="modalBailleur" tabindex="-1" aria-labelledby="modalBailleurLabel" aria-hidden="true">
|
||||
<div class="modal-dialog modal-dialog-centered">
|
||||
<div class="modal-dialog ">
|
||||
<div class="modal-content">
|
||||
<div class="modal-header">
|
||||
<h5 class="modal-title" id="modalBailleurLabel">Ajouter un Bailleur</h5>
|
||||
<h5 class="modal-title" id="modalBailleurLabel">Gestion des Bailleurs</h5>
|
||||
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
|
||||
</div>
|
||||
<div class="modal-body p-4">
|
||||
<form id="formBailleur" method="POST" action="{% url 'gestion_projet:creation-bailleur' %}">
|
||||
{% csrf_token %}
|
||||
{{ form_ajout_bailleur.as_p }}
|
||||
</form>
|
||||
</div>
|
||||
<div class="modal-footer">
|
||||
<button type="submit" id="btnEnregistrerBailleur" class="btn btn-success"><i class="bi bi-save me-1"></i> Enregistrer</button>
|
||||
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Annuler</button>
|
||||
</div>
|
||||
<ul class="nav nav-tabs px-3 pt-2" id="bailleurTab" role="tablist">
|
||||
<li class="nav-item" role="presentation">
|
||||
<button class="nav-link active" id="ajouter-tab" data-bs-toggle="tab"
|
||||
data-bs-target="#ajouter" type="button" role="tab">
|
||||
Ajouter
|
||||
</button>
|
||||
</li>
|
||||
<li class="nav-item" role="presentation">
|
||||
<button class="nav-link" id="liste-tab" data-bs-toggle="tab"
|
||||
data-bs-target="#liste" type="button" role="tab">
|
||||
Liste
|
||||
</button>
|
||||
</li>
|
||||
</ul>
|
||||
<div class="modal-body">
|
||||
<div class="tab-content pt-3">
|
||||
<div class="tab-pane fade show active" id="ajouter" role="tabpanel">
|
||||
<form id="formBailleur" method="POST" action="{% url 'gestion_projet:creation-bailleur' %}">
|
||||
{% csrf_token %}
|
||||
{{ form_ajout_bailleur.as_p }}
|
||||
<button type="submit" class="btn btn-success mt-3" id="btnEnregistrerBailleur">
|
||||
<i class="bi bi-save me-1"></i> Enregistrer
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
<div class="tab-pane fade" id="liste" role="tabpanel">
|
||||
|
||||
<div class="d-flex justify-content-between align-items-center mb-3">
|
||||
<h6 class="mb-0">Liste des bailleurs</h6>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col">
|
||||
<div id="table-bailleurs"
|
||||
data-url="{% url 'gestion_projet:liste-bailleurs' %}">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -19,6 +19,12 @@ urlpatterns = [
|
||||
views.creation_projet,
|
||||
name='creation-projet'
|
||||
),
|
||||
path(
|
||||
'bailleurs/',
|
||||
views.liste_bailleur,
|
||||
name='liste-bailleurs'
|
||||
),
|
||||
|
||||
path(
|
||||
'projet/modifier/<int:projet_id>/',
|
||||
views.modification_projet,
|
||||
@@ -84,6 +90,7 @@ urlpatterns = [
|
||||
views.liste_activites_projet,
|
||||
name='liste-activites-projet'
|
||||
),
|
||||
|
||||
# path(
|
||||
# 'projet/ajout-de-document/',
|
||||
# views.ajouter_document_projet,
|
||||
@@ -119,4 +126,6 @@ urlpatterns = [
|
||||
views.mises_a_jour_projet,
|
||||
name='mises-a-jour-projet'
|
||||
)
|
||||
]
|
||||
|
||||
]
|
||||
|
||||
|
||||
@@ -2,6 +2,7 @@ from datetime import date
|
||||
from decimal import Decimal, InvalidOperation
|
||||
from django.http import JsonResponse
|
||||
from django.shortcuts import redirect, render
|
||||
from django.urls import reverse
|
||||
from django.utils import timezone
|
||||
from django.contrib import messages
|
||||
from django.contrib.auth.decorators import login_required
|
||||
@@ -143,6 +144,22 @@ def creation_bailleur(request):
|
||||
return JsonResponse({'success': True})
|
||||
return JsonResponse({'success': False})
|
||||
|
||||
@login_required
|
||||
def liste_bailleur(request):
|
||||
""" Vue pour retourner la liste de tous les bailleurs """
|
||||
bailleurs = Bailleur.objects.all().order_by('-id')
|
||||
data = []
|
||||
for b in bailleurs:
|
||||
data.append({
|
||||
"id": b.id,
|
||||
"nom_organisme": b.nom_organisme,
|
||||
"contact": b.contact,
|
||||
"email": b.email,
|
||||
"pays": b.pays,
|
||||
})
|
||||
return JsonResponse(data, safe=False)
|
||||
|
||||
|
||||
@login_required
|
||||
def ajouter_financement_projet(request):
|
||||
"""Ajoute un financement à un projet en vérifiant que le total ne dépasse pas 100%"""
|
||||
|
||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -30,4 +30,4 @@ class Reservation(models.Model):
|
||||
statut = models.CharField(choices=STATUT, default='en_attente', max_length=25)
|
||||
|
||||
def __str__(self):
|
||||
return f"{self.salle} - {self.employe.user.first_name} {self.employe.user.last_name}"
|
||||
return f"{self.salle} - {self.employe.user.first_name} {self.employe.user.last_name}"
|
||||
|
||||
@@ -27,8 +27,14 @@ const calendrier = Schedule(document.getElementById('planning-reservation'), {
|
||||
$("motif_reservation").value=data.motif_reservation;
|
||||
$("besoin_zoom").checked=data.besoin_zoom;
|
||||
$("besoin_ordinateur").checked=data.besoin_ordinateur;
|
||||
$("lien_zoom").value=data.lien_zoom;
|
||||
|
||||
$("lien_zoom").value=data.lien_zoom;
|
||||
|
||||
const zoomContainer = $("lien_zoom_container");
|
||||
if (!data.besoin_zoom) {
|
||||
zoomContainer.classList.add("d-none");
|
||||
} else {
|
||||
zoomContainer.classList.remove("d-none");
|
||||
}
|
||||
if (data.besoin_zoom === false){
|
||||
$("lien_zoom_container").className = "d-none";
|
||||
}
|
||||
@@ -170,10 +176,6 @@ tableau_reservation_attente.on("rowClick", (row, rowData) => {
|
||||
$("lien_zoom_container").className = 'd-none';
|
||||
}
|
||||
|
||||
if(data.statut !== "refusee"){
|
||||
$("motif_refus_container").className = 'd-none';
|
||||
}
|
||||
|
||||
$("id_reservation_detail").value = data.id;
|
||||
$("id_reservation_refus").value = data.id;
|
||||
$("id_reservation_zoom").value = data.id;
|
||||
@@ -181,16 +183,17 @@ tableau_reservation_attente.on("rowClick", (row, rowData) => {
|
||||
$("employe").value=data.employe;
|
||||
$("salle").value=data.salle;
|
||||
$("statut-reservation").innerHTML=data.statut;
|
||||
$("date_evenement").value=data.date_debut;
|
||||
$("date_debut").value = data.date_debut;
|
||||
$("date_fin").value = data.date_fin;
|
||||
|
||||
$("heure_debut").value=data.heure_debut;
|
||||
$("heure_fin").value=data.heure_fin;
|
||||
$("motif_reservation").value=data.motif_reservation;
|
||||
$("besoin_zoom").checked=data.besoin_zoom;
|
||||
$("besoin_ordinateur").checked=data.besoin_ordi;
|
||||
$("lien_zoom").value=data.lien_zoom;
|
||||
$("motif_refus").value=data.motif_refus;
|
||||
|
||||
const modal = new bootstrap.Modal($("modalDetailReservation"));
|
||||
bootstrap.Modal.getOrCreateInstance($("modalReservationAttente")).hide();
|
||||
modal.show();
|
||||
})
|
||||
})
|
||||
|
||||
@@ -21,7 +21,7 @@
|
||||
<input class="form-control" id="salle" readonly>
|
||||
</div>
|
||||
<div class="form-group mb-2">
|
||||
<label>Date de l'évènement :</label>
|
||||
<label>Date de debut :</label>
|
||||
<input type='date' class="form-control" id="date_evenement" readonly >
|
||||
</div>
|
||||
<div class="form-group mb-2">
|
||||
@@ -48,24 +48,19 @@
|
||||
<label label="form-check-label">Besoin d'un ordinateur</label>
|
||||
<input type="checkbox" class="form-check-input" id="besoin_ordinateur" readonly >
|
||||
</div>
|
||||
<div class="form-group mb-2" id='motif_refus_container'>
|
||||
<label>Motif de refus de la reservation :</label>
|
||||
<textarea class="form-control" id="motif_refus" readonly></textarea>
|
||||
</div>
|
||||
<div class='d-flex justify-content-around mt-2'>
|
||||
{% if appartient_au_departement_informatique %}
|
||||
<button class="btn btn-primary" id="ajoutZoom">Ajout du lien zoom</button>
|
||||
{% endif %}
|
||||
{% if appartient_direction and reservation.statut == "en_attente" %}
|
||||
<button class="btn btn-danger" id="refuserReservation" data-lienrefus="{% url 'gestion_salle:refuser-reservation' %}">Refuser</button>
|
||||
{% endif %}
|
||||
<button class="btn btn-danger" id="bouton-annuler">Annuler</button>
|
||||
{% if appartient_direction %}
|
||||
<button class="btn btn-success" id="bouton-valider">Valider</button>
|
||||
{% endif %}
|
||||
<span id="current-user-id" data-user-id="{{ request.user.id }}"></span>
|
||||
<button class="btn btn-danger" id="bouton-annuler">Annuler</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
import json
|
||||
from datetime import timedelta
|
||||
from django.contrib import messages
|
||||
from django.contrib.auth.decorators import login_required
|
||||
@@ -9,72 +8,101 @@ from fonction_utilitaire import fonctions_utilitaire
|
||||
from gestion_employe.models import Employe
|
||||
from gestion_salle.forms import ReservationForm
|
||||
from .models import Reservation
|
||||
from datetime import timedelta
|
||||
from django.contrib.auth.models import User
|
||||
|
||||
|
||||
|
||||
@login_required
|
||||
def index(request:HttpRequest):
|
||||
"""Vue de gestion de la reservation de la salle"""
|
||||
def index(request: HttpRequest):
|
||||
|
||||
try:
|
||||
employe = Employe.objects.get(user=request.user)
|
||||
except Employe.DoesNotExist:
|
||||
messages.error(request, "Impossible d'accéder au menu 'Reservation de salle' car votre profil Utilisateur n'est lié à aucun profil Employe. Veuillez contacter l'administrateur.")
|
||||
messages.error(request, "Profil employé introuvable.")
|
||||
return redirect('gestion_conges:conge')
|
||||
|
||||
|
||||
|
||||
|
||||
if request.method == "POST":
|
||||
form = ReservationForm(request.POST)
|
||||
|
||||
if form.is_valid():
|
||||
date_debut = form.cleaned_data.get('date_debut')
|
||||
date_fin = form.cleaned_data.get('date_fin')
|
||||
salle = form.cleaned_data.get('salle')
|
||||
heure_debut = form.cleaned_data.get('heure_debut')
|
||||
heure_fin = form.cleaned_data.get('heure_fin')
|
||||
motif_reservation = form.cleaned_data.get('motif_reservation')
|
||||
besoin_zoom = form.cleaned_data.get('besoin_zoom')
|
||||
besoin_ordi = form.cleaned_data.get('besoin_ordi')
|
||||
date_debut = form.cleaned_data['date_debut']
|
||||
date_fin = form.cleaned_data['date_fin']
|
||||
salle = form.cleaned_data['salle']
|
||||
heure_debut = form.cleaned_data['heure_debut']
|
||||
heure_fin = form.cleaned_data['heure_fin']
|
||||
motif = form.cleaned_data['motif_reservation']
|
||||
besoin_zoom = form.cleaned_data['besoin_zoom']
|
||||
besoin_ordi = form.cleaned_data['besoin_ordi']
|
||||
|
||||
if date_fin < date_debut:
|
||||
messages.error(request, "Date fin invalide.")
|
||||
return redirect('gestion_salle:reservation-salle')
|
||||
|
||||
if heure_fin <= heure_debut:
|
||||
messages.error(request, "Heure invalide.")
|
||||
return redirect('gestion_salle:reservation-salle')
|
||||
|
||||
while date_debut <= date_fin :
|
||||
reservation = Reservation(
|
||||
employe = employe,
|
||||
date_debut = date_debut,
|
||||
date_fin = date_debut,
|
||||
salle = salle,
|
||||
heure_debut = heure_debut,
|
||||
heure_fin = heure_fin,
|
||||
besoin_zoom = besoin_zoom,
|
||||
besoin_ordi = besoin_ordi,
|
||||
motif_reservation=motif_reservation,
|
||||
if not request.user.first_name.strip() or not request.user.last_name.strip():
|
||||
messages.error(
|
||||
request,
|
||||
"Veuillez renseigner votre nom et prénom pour pouvoir faire une réservation."
|
||||
)
|
||||
reservation.save()
|
||||
date_debut = date_debut + timedelta(days=1)
|
||||
return redirect('gestion_salle:reservation-salle')
|
||||
|
||||
messages.success(request, "Réservation(s) créées avec succès.")
|
||||
created = []
|
||||
current_date = date_debut
|
||||
|
||||
while current_date <= date_fin:
|
||||
|
||||
reservation = Reservation.objects.create(
|
||||
employe=employe,
|
||||
date_debut=current_date,
|
||||
date_fin=current_date,
|
||||
salle=salle,
|
||||
heure_debut=heure_debut,
|
||||
heure_fin=heure_fin,
|
||||
besoin_zoom=besoin_zoom,
|
||||
besoin_ordi=besoin_ordi,
|
||||
motif_reservation=motif,
|
||||
statut="en_attente"
|
||||
)
|
||||
|
||||
created.append(reservation)
|
||||
current_date += timedelta(days=1)
|
||||
|
||||
messages.success(request, "Réservation(s) créée(s) avec succès.")
|
||||
if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION:
|
||||
fonctions_utilitaire.envoyer_mail(
|
||||
sujet = "Reservation de salle",
|
||||
message = f"""
|
||||
Une nouvelle demande de réservation de la {dict(Reservation.TYPE_CHOICES).get(salle)} a été effectuée par {employe.user.first_name} {employe.user.last_name} du {form.cleaned_data.get('date_debut').strftime('%d/%m/%Y')} au {form.cleaned_data.get('date_fin').strftime('%d/%m/%Y')} pour motif "{motif_reservation}".
|
||||
Une nouvelle demande de réservation de la {dict(Reservation.TYPE_CHOICES).get(salle)} a été effectuée par {employe.user.first_name} {employe.user.last_name} du {form.cleaned_data.get('date_debut').strftime('%d/%m/%Y')} au {form.cleaned_data.get('date_fin').strftime('%d/%m/%Y')} pour motif "{motif}".
|
||||
Veuillez vous connecter à la plateforme pour plus de détails.""",
|
||||
destinataires = list(fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION)
|
||||
)
|
||||
return redirect('gestion_salle:reservation-salle')
|
||||
|
||||
formulaire_reservation = ReservationForm()
|
||||
departement = Employe.objects.get(user__username=request.user).departement
|
||||
appartient_direction = 'direction' in request.user.groups.values_list('name', flat=True)
|
||||
liste_demande_reservation = [
|
||||
reservation.id for reservation in
|
||||
Reservation.objects.filter(employe=employe, statut='en_attente')
|
||||
]
|
||||
|
||||
departement = employe.departement
|
||||
|
||||
appartient_direction = request.user.groups.filter(name='direction').exists()
|
||||
|
||||
liste_demande_reservation = Reservation.objects.filter(
|
||||
employe=employe,
|
||||
statut='en_attente'
|
||||
).values_list('id', flat=True)
|
||||
|
||||
context = {
|
||||
'formulaire_reservation': formulaire_reservation,
|
||||
'nb_reservation_attente': Reservation.objects.filter(statut='en_attente').count(),
|
||||
'appartient_au_departement_informatique': 'Informatique' == departement.nom if departement else False,
|
||||
'appartient_au_departement_informatique': departement and departement.nom == "Systeme informatique",
|
||||
'appartient_direction': appartient_direction,
|
||||
'liste_demande_reservation': liste_demande_reservation
|
||||
'liste_demande_reservation': list(liste_demande_reservation),
|
||||
}
|
||||
return render(request, "gestion_salle/index.html", context)
|
||||
|
||||
return render(request, "gestion_salle/index.html", context)
|
||||
def liste_reservation(request:HttpRequest):
|
||||
"""Vue d'affichage des creneaux disponibles"""
|
||||
reservations = Reservation.objects.filter(statut = "validee")
|
||||
@@ -96,7 +124,6 @@ def liste_reservation(request:HttpRequest):
|
||||
"end": reservation.heure_fin,
|
||||
"color": color,
|
||||
})
|
||||
|
||||
return JsonResponse(liste_reservation, safe=False)
|
||||
|
||||
@login_required
|
||||
@@ -113,10 +140,12 @@ def liste_reservation_attente(request):
|
||||
return JsonResponse(liste_reservation, safe=False)
|
||||
|
||||
def detail_reservation(request:HttpRequest, reservation_id:int):
|
||||
|
||||
reservation = Reservation.objects.get(id=reservation_id)
|
||||
employe = reservation.employe.user
|
||||
|
||||
reservation_json = {
|
||||
|
||||
'id_reservation': reservation_id,
|
||||
'employe': f"{employe.first_name} {employe.last_name}",
|
||||
'salle': reservation.salle,
|
||||
@@ -129,7 +158,6 @@ def detail_reservation(request:HttpRequest, reservation_id:int):
|
||||
'besoin_ordinateur': reservation.besoin_ordi,
|
||||
'lien_zoom': reservation.lien_zoom or '',
|
||||
}
|
||||
|
||||
return JsonResponse(reservation_json, safe=True)
|
||||
|
||||
@login_required
|
||||
@@ -175,16 +203,17 @@ def annuler_reservation(request:HttpRequest):
|
||||
return redirect('gestion_salle:reservation-salle')
|
||||
|
||||
@login_required
|
||||
def valider_reservation(request:HttpRequest):
|
||||
"""Vue de gestion de l'annulation de la reservation"""
|
||||
def valider_reservation(request: HttpRequest):
|
||||
"""Validation d'une réservation"""
|
||||
if request.method == 'POST':
|
||||
reservation_id= request.POST['id_reservation']
|
||||
reservation_id = request.POST.get('id_reservation')
|
||||
|
||||
try:
|
||||
reservation = Reservation.objects.get(id=reservation_id)
|
||||
except reservation.DoesNotExist:
|
||||
messages.error(request, "La resevertion selectionné n'existe pas.")
|
||||
reservation = Reservation.objects.get(id=reservation_id)
|
||||
except Reservation.DoesNotExist:
|
||||
messages.error(request, "La réservation sélectionnée n'existe pas.")
|
||||
return redirect("salle")
|
||||
|
||||
|
||||
reservation.statut = 'validee'
|
||||
reservation.save()
|
||||
if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION:
|
||||
@@ -198,16 +227,18 @@ def valider_reservation(request:HttpRequest):
|
||||
return redirect('gestion_salle:reservation-salle')
|
||||
|
||||
@login_required
|
||||
def refuser_reservation(request:HttpRequest):
|
||||
"""Vue de gestion de refus de la reservation"""
|
||||
data = json.loads(request.body)
|
||||
reservation_id = data.get("id_reservation")
|
||||
try:
|
||||
reservation = Reservation.objects.get(id=reservation_id)
|
||||
except Reservation.DoesNotExist and ValueError:
|
||||
return JsonResponse({"message": "La resevertion selectionné n'existe pas."})
|
||||
else:
|
||||
reservation.statut = "refusee"
|
||||
def refuser_reservation(request: HttpRequest):
|
||||
"""Refuser une réservation"""
|
||||
if request.method == 'POST':
|
||||
reservation_id = request.POST.get('id_reservation')
|
||||
|
||||
try:
|
||||
reservation = Reservation.objects.get(id=reservation_id)
|
||||
except Reservation.DoesNotExist:
|
||||
messages.error(request, "La réservation n'existe pas.")
|
||||
return redirect("salle")
|
||||
|
||||
reservation.statut = 'refusee'
|
||||
reservation.save()
|
||||
|
||||
if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION:
|
||||
@@ -217,4 +248,4 @@ def refuser_reservation(request:HttpRequest):
|
||||
destinataires = [reservation.employe.user.email]
|
||||
)
|
||||
|
||||
return JsonResponse({"message": "Réservation refusée avec succès."})
|
||||
return JsonResponse({"message": "Réservation refusée avec succès."})
|
||||
|
||||
301
notification/services.py
Normal file
301
notification/services.py
Normal file
@@ -0,0 +1,301 @@
|
||||
from django.core.mail import send_mail
|
||||
from django.conf import settings
|
||||
|
||||
def send_notification_email(user, sujet, message):
|
||||
"""
|
||||
Fonction générique pour envoyer un email simple
|
||||
"""
|
||||
if not user.email:
|
||||
return
|
||||
send_mail(
|
||||
sujet,
|
||||
message,
|
||||
settings.DEFAULT_FROM_EMAIL,
|
||||
[user.email],
|
||||
fail_silently=False,
|
||||
)
|
||||
|
||||
def email_reservation_creee(reservation):
|
||||
user = reservation.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre réservation a été enregistrée avec succès.
|
||||
|
||||
Salle : {reservation.salle}
|
||||
Date : {reservation.date_debut}
|
||||
Heure : {reservation.heure_debut} - {reservation.heure_fin}
|
||||
Motif : {reservation.motif_reservation}
|
||||
|
||||
Merci.
|
||||
"""
|
||||
send_notification_email(user, "Confirmation de votre réservation", message)
|
||||
|
||||
def email_reservation_directeur(reservation, directeur):
|
||||
message = f"""
|
||||
Bonjour {directeur.get_full_name()},
|
||||
|
||||
Une nouvelle réservation attend votre validation.
|
||||
|
||||
Employé : {reservation.employe.user.get_full_name()}
|
||||
Salle : {reservation.salle}
|
||||
Date : Du {reservation.date_debut} au {reservation.date_fin}
|
||||
Heure : {reservation.heure_debut} - {reservation.heure_fin}
|
||||
Motif : {reservation.motif_reservation}
|
||||
|
||||
Connectez-vous à la plateforme pour valider.
|
||||
|
||||
https://support.cerfig.org/login/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(directeur, "Nouvelle réservation à valider", message)
|
||||
|
||||
def email_reservation_zoom(reservation, admin):
|
||||
message = f"""
|
||||
Bonjour {admin.get_full_name()},
|
||||
|
||||
Une demande de lien Zoom a été faite.
|
||||
|
||||
Employé : {reservation.employe.user.get_full_name()}
|
||||
Date : {reservation.date_debut} au {reservation.date_fin}
|
||||
Heure : {reservation.heure_debut} - {reservation.heure_fin}
|
||||
Motif : {reservation.motif_reservation}
|
||||
|
||||
Veuillez créer le lien Zoom.
|
||||
|
||||
https://support.cerfig.org/login/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(admin, "Création lien Zoom requise", message)
|
||||
|
||||
def email_statut_reservation(reservation):
|
||||
user = reservation.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Le statut de votre réservation a été mis à jour.
|
||||
|
||||
Salle : {reservation.salle}
|
||||
Date : {reservation.date_debut}
|
||||
Statut : {reservation.statut}
|
||||
|
||||
Merci de vous connecter
|
||||
|
||||
https://support.cerfig.org
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
|
||||
"""
|
||||
send_notification_email(user, "Mise à jour de votre réservation", message)
|
||||
|
||||
def email_reservation_validee(reservation):
|
||||
user = reservation.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre réservation a été VALIDÉE.
|
||||
|
||||
Salle : {reservation.salle}
|
||||
Date : {reservation.date_debut}
|
||||
|
||||
https://support.cerfig.org/login/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(user, "Réservation validée", message)
|
||||
|
||||
def email_reservation_refusee(reservation):
|
||||
user = reservation.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre réservation a été REFUSÉE.
|
||||
|
||||
Salle : {reservation.salle}
|
||||
Date : {reservation.date_debut}
|
||||
|
||||
https://support.cerfig.org/login/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(user, "Réservation refusée", message)
|
||||
|
||||
def email_contrat_cree(contrat):
|
||||
user = contrat.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Un nouveau contrat a été créé pour vous.
|
||||
|
||||
Date début : {contrat.date_debut}
|
||||
Date fin : {contrat.date_fin}
|
||||
|
||||
Bienvenue !
|
||||
Merci de vous connecter pour plus informations sur votre contrat
|
||||
|
||||
https://support.cerfig.org
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
|
||||
send_notification_email(user, "Nouveau contrat", message)
|
||||
|
||||
def email_statut_contrat(contrat):
|
||||
user = contrat.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre contrat a été modifier par les ressource humaine .
|
||||
veuillez vous connecter pour voir les nouvelles informations sur votre contrat
|
||||
|
||||
https://support.cerfig.org/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(user, "Mise à jour du contrat", message)
|
||||
|
||||
def email_affectation_projet(affectation):
|
||||
user = affectation.employe.user
|
||||
projet = affectation.projet
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre affectation au projet a été mise à jour.
|
||||
|
||||
Projet : {projet.nom}
|
||||
Description : {projet.description}
|
||||
Date début : {projet.date_debut}
|
||||
Date fin : {projet.date_fin}
|
||||
Rôle : {affectation.role}
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
|
||||
"""
|
||||
send_notification_email(user, "Affectation projet", message)
|
||||
|
||||
def email_expiration_fichier(projet, fichier, chef_projet):
|
||||
message = f"""
|
||||
Bonjour {chef_projet.get_full_name()},
|
||||
|
||||
Un fichier est proche de son expiration.
|
||||
|
||||
Projet : {projet.nom}
|
||||
Fichier : {fichier.nom}
|
||||
Date début : {fichier.date_debut}
|
||||
Date fin : {fichier.date_fin}
|
||||
|
||||
https://support.cerfig.org/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
|
||||
"""
|
||||
send_notification_email(chef_projet, "Fichier proche expiration", message)
|
||||
|
||||
def email_conge_cree(conge):
|
||||
user = conge.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre demande de congé a été enregistrée avec succès.
|
||||
|
||||
Type de congé : {conge.type_conge}
|
||||
Date début : {conge.date_debut}
|
||||
Date fin : {conge.date_fin}
|
||||
Motif : {conge.motif}
|
||||
|
||||
Votre demande est en attente de validation.
|
||||
|
||||
https://support.cerfig.org/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(user, "Demande de congé enregistrée", message)
|
||||
|
||||
def email_conge_responsable(conge, responsable):
|
||||
message = f"""
|
||||
Bonjour {responsable.get_full_name()},
|
||||
|
||||
Une demande de congé nécessite votre validation.
|
||||
|
||||
Employé : {conge.employe.user.get_full_name()}
|
||||
Type : {conge.type_conge}
|
||||
Date : {conge.date_debut} au {conge.date_fin}
|
||||
Motif : {conge.motif}
|
||||
|
||||
Merci de valider ou refuser.
|
||||
|
||||
https://support.cerfig.org/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(responsable, "Validation congé (Niveau 1)", message)
|
||||
|
||||
def email_conge_directeur(conge, directeur):
|
||||
message = f"""
|
||||
Bonjour {directeur.get_full_name()},
|
||||
|
||||
Une demande de congé a été validée par le responsable et attend votre validation finale.
|
||||
|
||||
Employé : {conge.employe.user.get_full_name()}
|
||||
Type : {conge.type_conge}
|
||||
Date : {conge.date_debut} au {conge.date_fin}
|
||||
|
||||
Merci de valider définitivement.
|
||||
|
||||
https://support.cerfig.org/login/
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(directeur, "Validation congé (Niveau 2)", message)
|
||||
|
||||
def email_conge_valide(conge):
|
||||
user = conge.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre demande de congé a été VALIDÉE.
|
||||
|
||||
Type : {conge.type_conge}
|
||||
Date : {conge.date_debut} au {conge.date_fin}
|
||||
|
||||
Bon repos a vous
|
||||
|
||||
https://support.cerfig.org
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(user, "Congé validé", message)
|
||||
|
||||
def email_conge_refuse(conge):
|
||||
user = conge.employe.user
|
||||
message = f"""
|
||||
Bonjour {user.get_full_name()},
|
||||
|
||||
Votre demande de congé a été REFUSÉE.
|
||||
|
||||
Type : {conge.type_conge}
|
||||
Date : {conge.date_debut} au {conge.date_fin}
|
||||
|
||||
Motif : {conge.motif_refus or "Non précisé"}
|
||||
|
||||
https://support.cerfig.org
|
||||
|
||||
Cordialement,
|
||||
SIRH
|
||||
"""
|
||||
send_notification_email(user, "Congé refusé", message)
|
||||
Reference in New Issue
Block a user