5 Commits

Author SHA1 Message Date
267994044c feature: Affichage de tous les detail de la resevation
Some checks failed
Organisation/sirh/pipeline/head There was a failure building this commit
2026-06-08 11:06:22 +00:00
a4b4a68dd2 feature: Affichage du message sur l'tat du contrat 2026-05-11 13:13:08 +02:00
07dc097d27 feature: Affichage des details de la reservation 2026-05-11 13:12:42 +02:00
4146563f41 Fonctionnalite : Ajout de la liste des contrats 2026-05-11 13:05:06 +02:00
c0cdca48fa Fonctionnalite: Ajout des contrats 2026-05-11 12:43:16 +02:00
45 changed files with 403 additions and 67 deletions

1
Jenkinsfile vendored
View File

@@ -39,7 +39,6 @@ pipeline
python manage.py makemigrations python manage.py makemigrations
python manage.py migrate python manage.py migrate
python manage.py collectstatic --noinput
echo $SUDO_PASSWORD | sudo -S chown -R www-data:www-data /var/www/sirh echo $SUDO_PASSWORD | sudo -S chown -R www-data:www-data /var/www/sirh
echo "Deploiement reussi" echo "Deploiement reussi"

View File

@@ -19,12 +19,13 @@
{% endfor %} {% endfor %}
{% endif %} {% endif %}
{% if not expiration_contrat %} {% if not has_contrat %}
<div class="alert alert-danger fade show alert-dismissible mt-2"> <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. <strong>Important :</strong> Les informations sur votre contrat n'ont pas été renseignées, veuillez contacter les ressources humaines.
</div> </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. <strong>Important :</strong> Votre contrat de travail expire dans {{ contrat_nb_jours_restant }} jours, veuillez contacter les ressources humaines.
</div> </div>
{% endif %} {% endif %}

View File

@@ -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>

View File

@@ -129,26 +129,41 @@ def mon_profil(request):
try: try:
employe = Employe.objects.get(user__username=request.user) employe = Employe.objects.get(user__username=request.user)
except Employe.DoesNotExist: 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") return redirect("gestion_conges:conge")
contrats = Contrat.objects.filter(employe=employe, statut='actif').first() contrats = Contrat.objects.filter(employe=employe, statut='actif').first()
projets = Affectation.objects.filter( projets = Affectation.objects.filter(
employe=employe, employe=employe,
date_fin_daffectation__gte=timezone.now().date() date_fin_daffectation__gte=timezone.now().date()
).select_related('projet') ).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( return render(
request, request,
'gestion_employe/monprofil.html', 'gestion_employe/monprofil.html',
{ {
'employe': employe, 'employe': employe,
'contrats': [{ 'contrats': [{
**model_to_dict(contrats), **model_to_dict(contrats),
"type_contrat": dict(Contrat.TYPE_CONTRAT).get(contrats.type_contrat), "type_contrat": dict(Contrat.TYPE_CONTRAT).get(contrats.type_contrat),
"statut": dict(Contrat.STATUT_CONTRAT).get(contrats.statut), "statut": dict(Contrat.STATUT_CONTRAT).get(contrats.statut),
"fichier_contrat": contrats.fichier_contrat.url if contrats.fichier_contrat else "", "fichier_contrat": contrats.fichier_contrat.url if contrats.fichier_contrat else "",
} if contrats else []], }] if contrats else [],
'projets': [ 'projets': [
{ {
**model_to_dict(a.projet), **model_to_dict(a.projet),
@@ -158,12 +173,13 @@ def mon_profil(request):
"pourcentage_temps_affectation": a.pourcentage_temps_affectation "pourcentage_temps_affectation": a.pourcentage_temps_affectation
} for a in projets } for a in projets
], ],
"formation_form": FormationForm(), "formation_form": FormationForm(),
"expiration_contrat": contrats.nombre_jours_restant <= fonctions_utilitaire.DUREE_FIN_CONTRAT if contrats else False, "has_contrat": has_contrat,
"contrat_nb_jours_restant": contrats.nombre_jours_restant if contrats else None "expiration_contrat": expiration_contrat,
"contrat_nb_jours_restant": contrat_nb_jours_restant
} }
) )
@login_required @login_required
def modifier_mot_passe(request): def modifier_mot_passe(request):
"""Vue pour permettre à un utilisateur de modifier son mot de passe et ses informations de profil""" """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: except Employe.DoesNotExist:
messages.error(request, "Employé non trouvé.") messages.error(request, "Employé non trouvé.")
return redirect('employe-index') return redirect('employe-index')
contrat_actif = Contrat.objects.filter( contrat_actif = Contrat.objects.filter(
employe=employe, employe=employe,
date_fin__gte=date.today() date_fin__gte=date.today()

View File

@@ -23,7 +23,6 @@ btnEnregistrerBailleur.addEventListener('click', function() {
}); });
}); });
document.addEventListener("DOMContentLoaded", function () {
table = new Tabulator("#table-bailleurs", { table = new Tabulator("#table-bailleurs", {
ajaxURL: "/gestion-projet/bailleurs/", ajaxURL: "/gestion-projet/bailleurs/",
@@ -46,6 +45,3 @@ document.addEventListener("DOMContentLoaded", function () {
} }
} }
}); });
});

View File

@@ -48,4 +48,3 @@
</div> </div>
</div> </div>
</div> </div>

View File

@@ -335,7 +335,6 @@ def activites_projet(request):
} }
return render(request, 'gestion_projet/suivi_activite.html', context) return render(request, 'gestion_projet/suivi_activite.html', context)
@login_required @login_required
def ajouter_activite_projet(request): def ajouter_activite_projet(request):
"""Vue pour ajouter une activité à un projet spécifique via un formulaire""" """Vue pour ajouter une activité à un projet spécifique via un formulaire"""

View File

@@ -29,10 +29,21 @@ const calendrier = Schedule(document.getElementById('planning-reservation'), {
$("besoin_ordinateur").checked=data.besoin_ordinateur; $("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){ if (data.besoin_zoom === false){
$("lien_zoom_container").className = "d-none"; $("lien_zoom_container").className = "d-none";
} }
if(data.statut !== "annulee"){
$("motif_refus_container").className = "d-none";
}else{
$("motif_refus").value=data.motif_refus;
}
}) })
} }
}); });
@@ -159,11 +170,12 @@ const tableau_reservation_attente = new Tabulator("#tableau-reservation-attente"
tableau_reservation_attente.on("rowClick", (row, rowData) => { tableau_reservation_attente.on("rowClick", (row, rowData) => {
const data = rowData.getData(); const data = rowData.getData();
console.log(data);
if(data.besoin_zoom === false){ if(data.besoin_zoom === false){
$("lien_zoom_container").className = 'd-none'; $("lien_zoom_container").className = 'd-none';
} }
const id_user = $("current-user-id").dataset.userid;
$("id_reservation_detail").value = data.id; $("id_reservation_detail").value = data.id;
$("id_reservation_refus").value = data.id; $("id_reservation_refus").value = data.id;
$("id_reservation_zoom").value = data.id; $("id_reservation_zoom").value = data.id;
@@ -173,6 +185,7 @@ tableau_reservation_attente.on("rowClick", (row, rowData) => {
$("statut-reservation").innerHTML=data.statut; $("statut-reservation").innerHTML=data.statut;
$("date_debut").value = data.date_debut; $("date_debut").value = data.date_debut;
$("date_fin").value = data.date_fin; $("date_fin").value = data.date_fin;
$("heure_debut").value=data.heure_debut; $("heure_debut").value=data.heure_debut;
$("heure_fin").value=data.heure_fin; $("heure_fin").value=data.heure_fin;
$("motif_reservation").value=data.motif_reservation; $("motif_reservation").value=data.motif_reservation;
@@ -180,10 +193,6 @@ tableau_reservation_attente.on("rowClick", (row, rowData) => {
$("besoin_ordinateur").checked=data.besoin_ordi; $("besoin_ordinateur").checked=data.besoin_ordi;
$("lien_zoom").value=data.lien_zoom; $("lien_zoom").value=data.lien_zoom;
if (id_user != data.employe_id){
$("bouton-annuler").className = "d-none";
}
const modal = new bootstrap.Modal($("modalDetailReservation")); const modal = new bootstrap.Modal($("modalDetailReservation"));
bootstrap.Modal.getOrCreateInstance($("modalReservationAttente")).hide(); bootstrap.Modal.getOrCreateInstance($("modalReservationAttente")).hide();
modal.show(); modal.show();

View File

@@ -1,5 +1,4 @@
<!-- Modal d'affichage des détails d'une reservation --> <!-- Modal d'affichage des détails d'une reservation -->
{% load tags_personnaliser %}
<div class="modal fade" id="modalDetailReservation" tabindex="-1" aria-labelledby="modalDetailReservationLabel" aria-hidden="true"> <div class="modal fade" id="modalDetailReservation" tabindex="-1" aria-labelledby="modalDetailReservationLabel" aria-hidden="true">
<div class="modal-dialog"> <div class="modal-dialog">
<div class="modal-content"> <div class="modal-content">
@@ -23,11 +22,7 @@
</div> </div>
<div class="form-group mb-2"> <div class="form-group mb-2">
<label>Date de debut :</label> <label>Date de debut :</label>
<input type='date' class="form-control" id="date_debut" readonly > <input type='date' class="form-control" id="date_evenement" readonly >
</div>
<div class="form-group mb-2">
<label>Date de fin :</label>
<input type='date' class="form-control" id="date_fin" readonly >
</div> </div>
<div class="form-group mb-2"> <div class="form-group mb-2">
<label>Heure de début :</label> <label>Heure de début :</label>
@@ -57,11 +52,11 @@
{% if appartient_au_departement_informatique %} {% if appartient_au_departement_informatique %}
<button class="btn btn-primary" id="ajoutZoom">Ajout du lien zoom</button> <button class="btn btn-primary" id="ajoutZoom">Ajout du lien zoom</button>
{% endif %} {% endif %}
{% if user|has_group:'direction' %} {% if appartient_direction and reservation.statut == "en_attente" %}
<button class="btn btn-danger" id="refuserReservation" data-lienrefus="{% url 'gestion_salle:refuser-reservation' %}">Refuser</button> <button class="btn btn-danger" id="refuserReservation" data-lienrefus="{% url 'gestion_salle:refuser-reservation' %}">Refuser</button>
<button class="btn btn-success" id="bouton-valider">Valider</button> <button class="btn btn-success" id="bouton-valider">Valider</button>
{% endif %} {% endif %}
<span id="current-user-id" data-userid="{{ request.user.id }}"></span> <span id="current-user-id" data-user-id="{{ request.user.id }}"></span>
<button class="btn btn-danger" id="bouton-annuler">Annuler</button> <button class="btn btn-danger" id="bouton-annuler">Annuler</button>
</div> </div>
</form> </form>

View File

@@ -8,6 +8,9 @@ from fonction_utilitaire import fonctions_utilitaire
from gestion_employe.models import Employe from gestion_employe.models import Employe
from gestion_salle.forms import ReservationForm from gestion_salle.forms import ReservationForm
from .models import Reservation from .models import Reservation
from datetime import timedelta
from django.contrib.auth.models import User
@login_required @login_required
@@ -71,15 +74,13 @@ def index(request: HttpRequest):
current_date += timedelta(days=1) current_date += timedelta(days=1)
messages.success(request, "Réservation(s) créée(s) avec succès.") messages.success(request, "Réservation(s) créée(s) avec succès.")
print(fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION)
if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION: if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION:
fonctions_utilitaire.envoyer_mail( fonctions_utilitaire.envoyer_mail(
sujet = "Reservation de salle", sujet = "Reservation de salle",
message = f""" 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}". 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.""", Veuillez vous connecter à la plateforme pour plus de détails.""",
destinataires = fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION destinataires = list(fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION)
) )
return redirect('gestion_salle:reservation-salle') return redirect('gestion_salle:reservation-salle')
@@ -102,7 +103,6 @@ Veuillez vous connecter à la plateforme pour plus de détails.""",
} }
return render(request, "gestion_salle/index.html", context) return render(request, "gestion_salle/index.html", context)
def liste_reservation(request:HttpRequest): def liste_reservation(request:HttpRequest):
"""Vue d'affichage des creneaux disponibles""" """Vue d'affichage des creneaux disponibles"""
reservations = Reservation.objects.filter(statut = "validee") reservations = Reservation.objects.filter(statut = "validee")
@@ -133,9 +133,7 @@ def liste_reservation_attente(request):
{ {
**model_to_dict(reservation), **model_to_dict(reservation),
"employe": f"{reservation.employe.user.first_name} {reservation.employe.user.last_name}", "employe": f"{reservation.employe.user.first_name} {reservation.employe.user.last_name}",
"employe_id": reservation.employe.user.id, "salle": dict(Reservation.TYPE_CHOICES).get(reservation.salle)
"salle": dict(Reservation.TYPE_CHOICES).get(reservation.salle),
'statut': dict(Reservation.STATUT).get(reservation.statut),
} for reservation in reservations } for reservation in reservations
] ]
@@ -160,7 +158,6 @@ def detail_reservation(request:HttpRequest, reservation_id:int):
'besoin_ordinateur': reservation.besoin_ordi, 'besoin_ordinateur': reservation.besoin_ordi,
'lien_zoom': reservation.lien_zoom or '', 'lien_zoom': reservation.lien_zoom or '',
} }
return JsonResponse(reservation_json, safe=True) return JsonResponse(reservation_json, safe=True)
@login_required @login_required
@@ -219,7 +216,6 @@ def valider_reservation(request: HttpRequest):
reservation.statut = 'validee' reservation.statut = 'validee'
reservation.save() reservation.save()
if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION: if fonctions_utilitaire.EMAIL_ASSISTANTE_DE_DIRECTION:
fonctions_utilitaire.envoyer_mail( fonctions_utilitaire.envoyer_mail(
sujet = "Reservation de salle", sujet = "Reservation de salle",

301
notification/services.py Normal file
View 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)