{
    "id": "event_a4c22d3ca872c3ae",
    "timestamp": 1777292253,
    "branch_id": "main",
    "parent_event_id": "event_1d594ac189671de7",
    "type": "patch_apply",
    "label": "supprimer js bootstrap modernizr manquants",
    "source": "patch",
    "author": "CNOC",
    "session_id": "43305eb2706f6eee2531a5a173355a18",
    "payload": [
        {
            "path": "pointages/admin.php",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 6264,
                "sha1": "b691f5cb970c626d45b282fe11bd9ac23057bc23",
                "content_b64": "<?php
/* doc-project | pointages/admin.php | Protège l’accès administrateur via autorisation permanente par appareil et permet l’ajout d’un salarié en base. | Expose: aucun | Dépend de: config.php, includes/device_auth.php, index.php, connexion.php | Impacte: état de session, cookie d’appareil, accès à l’interface, insertion BDD, redirection | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_utilisateurs(Nom, Prenom, DateDeNaissance, Role, Email, CodePin, DateDeCreation, Statut) */



session_start();
require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_device_authorized($pdo);

?>
<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">





<title>Ajout nouveau salarié</title>

<!--



Template 2089 Meteor



http://www.tooplate.com/view/2089-meteor



-->

<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">



<link rel="stylesheet" href="css/bootstrap.min.css">

<link rel="stylesheet" href="css/bootstrap-theme.min.css">

<link rel="stylesheet" href="css/fontAwesome.css">

<link rel="stylesheet" href="css/fontAwesome.css">
<link rel="stylesheet" href="css/hero-slider.css">
<link rel="stylesheet" href="css/tooplate-style.css">
<?php
$cssPath = __DIR__ . '/css/style.css';
$cssVersion = file_exists($cssPath) ? filemtime($cssPath) : time();
?>
<link rel="stylesheet" href="css/style.css?v=<?php echo (int)$cssVersion; ?>">



<link href="https://fonts.googleapis.com/css?family=Open+Sans:300,400,600,700,800" rel="stylesheet">



<script src="js/vendor/modernizr-2.8.3-respond-1.4.2.min.js"></script>

</head>

<body class="page-admin theme-dark">







<div id="about" class="page-section">

<div class="container">

<div class="row">

<div class="col-md-12">

<div class="section-heading">

<h1>Ajout nouvel employé</h1>





<h4></h4><div class="line-dec"></div>                        



<center><font color="white">











<!-- Formulaire pour ajouter un employé -->

<br>
<font color="black">
<form method="post" action="">
<!-- Ajouter les champs manquants ici -->
<input type="text" name="nom" placeholder="Nom" required><br><br>
<input type="text" name="prenom" placeholder="Prénom" required><br><br>

<input type="text" id="dateDeNaissance" name="dateDeNaissance" oninput="formaterDate(this)" inputmode="numeric" maxlength="10" placeholder="JJ/MM/AAAA"><br><br>
<input type="text" name="role" placeholder="Role" ><br><br>
<input type="email" name="email" placeholder="Email" ><br><br>
<input type="text" name="codePin" placeholder="Code Pin" ><br><br>

<input type="submit" name="submit" value="Ajouter">
</font>
</form>







<br><br><br>

<font color="white">

<?php
if (isset($_POST['submit'])) {
    $dateParts = explode('/', $_POST['dateDeNaissance']);
    $dateDeNaissanceFormatted = $dateParts[2] . '-' . $dateParts[1] . '-' . $dateParts[0];

    // Récupérer les données du formulaire
    $nom = $_POST['nom'];
    $prenom = $_POST['prenom'];
    $dateDeNaissance = $_POST['dateDeNaissance'];
    $role = $_POST['role'];
    $email = $_POST['email'];
    $codePin = $_POST['codePin'];
    
    // Construire et exécuter la requête INSERT
    $stmt = $pdo->prepare("INSERT INTO z_ptg_aqp_utilisateurs (Nom, Prenom, DateDeNaissance, Role, Email, CodePin, DateDeCreation, Statut) VALUES (?, ?, ?, ?, ?, ?, NOW(), 'actif')");
    $stmt->execute([$nom, $prenom, $dateDeNaissanceFormatted, $role, $email, $codePin]);
    
    // Rediriger avec toast (affiché sur index.php)
    $toastMessage = rawurlencode('Employé ajouté avec succès');
    echo "<script>window.location.href = 'index.php?toast={$toastMessage}&toastType=success';</script>";
}
?>






<br><br><br><br><br>

<font color="black">

<button onclick="window.location.href = 'index.php';"> Retourner page employés </button>

</font>





</font>

</center>

</div></div></div></div>



<p><a href="index.php">Retour accueil</a></p>

</div>



















<script src="//ajax.googleapis.com/ajax/libs/jquery/1.11.2/jquery.min.js"></script>

<script>window.jQuery || document.write('<script src="js/vendor/jquery-1.11.2.min.js"><\/script>')</script>



<script src="js/vendor/bootstrap.min.js"></script>



<script src="js/plugins.js"></script>

<script src="js/main.js"></script>





<script type="text/javascript">

$(document).ready(function() {
    
    // navigation click actions 
    
    $('.scroll-link').on('click', function(event){
        
        event.preventDefault();
        
        var sectionID = $(this).attr("data-id");
        
        scrollToID('#' + sectionID, 750);
        
    });
    
    // scroll to top action
    
    $('.scroll-top').on('click', function(event) {
        
        event.preventDefault();
        
        $('html, body').animate({scrollTop:0}, 'slow');         
        
    });
    
    // mobile nav toggle
    
    $('#nav-toggle').on('click', function (event) {
        
        event.preventDefault();
        
        $('#main-nav').toggleClass("open");
        
    });
    
});

// scroll function

function scrollToID(id, speed){
    
    var offSet = 50;
    
    var targetOffset = $(id).offset().top - offSet;
    
    var mainNav = $('#main-nav');
    
    $('html,body').animate({scrollTop:targetOffset}, speed);
    
    if (mainNav.hasClass("open")) {
        
        mainNav.css("height", "1px").removeClass("in").addClass("collapse");
        
        mainNav.removeClass("open");
        
    }
    
}

if (typeof console === "undefined") {
    
    console = {
        
        log: function() { }
        
    };
    
}


function formaterDate(input) {
    var valeur = input.value;
    var chiffres = valeur.replace(/\D/g, ''); // Supprime tous les caractères non-numériques
    
    // Ajoute des '/' après le jour et le mois
    chiffres = chiffres.substring(0, 2) + (chiffres.length > 2 ? '/' : '') + chiffres.substring(2, 4) + (chiffres.length > 4 ? '/' : '') + chiffres.substring(4, 8);
    
    input.value = chiffres;
}


</script>

</body>

</html>"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 6607,
                "sha1": "d6906f9c89917042c845ba35b47847fec59fa8e7",
                "content_b64": "<?php
/* doc-project | pointages/admin.php | Protège l’accès administrateur via autorisation permanente par appareil, permet l’ajout d’un salarié et charge uniquement des assets locaux versionnés sans Modernizr/Respond ni Bootstrap JS. | Expose: aucun | Dépend de: config.php, includes/device_auth.php, includes/asset_version.php, index.php, connexion.php, js/vendor/jquery-1.11.2.min.js, js/plugins.js, js/main.js | Impacte: état de session, cookie d’appareil, accès à l’interface, insertion BDD, redirection, interactions UI natives | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_utilisateurs(Nom, Prenom, DateDeNaissance, Role, Email, CodePin, DateDeCreation, Statut) */



session_start();
require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_once __DIR__ . "/includes/asset_version.php";
require_device_authorized($pdo);

?>
<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">





<title>Ajout nouveau salarié</title>

<!--



Template 2089 Meteor



http://www.tooplate.com/view/2089-meteor



-->

<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">



<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap-theme.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/fontAwesome.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/hero-slider.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/tooplate-style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/style.css'), ENT_QUOTES, 'UTF-8'); ?>">

</head>

<body class="page-admin theme-dark">







<div id="about" class="page-section">

<div class="container">

<div class="row">

<div class="col-md-12">

<div class="section-heading">

<h1>Ajout nouvel employé</h1>





<h4></h4><div class="line-dec"></div>                        



<center><font color="white">











<!-- Formulaire pour ajouter un employé -->

<br>
<font color="black">
<form method="post" action="">
<!-- Ajouter les champs manquants ici -->
<input type="text" name="nom" placeholder="Nom" required><br><br>
<input type="text" name="prenom" placeholder="Prénom" required><br><br>

<input type="text" id="dateDeNaissance" name="dateDeNaissance" oninput="formaterDate(this)" inputmode="numeric" maxlength="10" placeholder="JJ/MM/AAAA"><br><br>
<input type="text" name="role" placeholder="Role" ><br><br>
<input type="email" name="email" placeholder="Email" ><br><br>
<input type="text" name="codePin" placeholder="Code Pin" ><br><br>

<input type="submit" name="submit" value="Ajouter">
</font>
</form>







<br><br><br>

<font color="white">

<?php
if (isset($_POST['submit'])) {
    $dateParts = explode('/', $_POST['dateDeNaissance']);
    $dateDeNaissanceFormatted = $dateParts[2] . '-' . $dateParts[1] . '-' . $dateParts[0];

    // Récupérer les données du formulaire
    $nom = $_POST['nom'];
    $prenom = $_POST['prenom'];
    $dateDeNaissance = $_POST['dateDeNaissance'];
    $role = $_POST['role'];
    $email = $_POST['email'];
    $codePin = $_POST['codePin'];
    
    // Construire et exécuter la requête INSERT
    $stmt = $pdo->prepare("INSERT INTO z_ptg_aqp_utilisateurs (Nom, Prenom, DateDeNaissance, Role, Email, CodePin, DateDeCreation, Statut) VALUES (?, ?, ?, ?, ?, ?, NOW(), 'actif')");
    $stmt->execute([$nom, $prenom, $dateDeNaissanceFormatted, $role, $email, $codePin]);
    
    // Rediriger avec toast (affiché sur index.php)
    $toastMessage = rawurlencode('Employé ajouté avec succès');
    echo "<script>window.location.href = 'index.php?toast={$toastMessage}&toastType=success';</script>";
}
?>






<br><br><br><br><br>

<font color="black">

<button onclick="window.location.href = 'index.php';"> Retourner page employés </button>

</font>





</font>

</center>

</div></div></div></div>



<p><a href="index.php">Retour accueil</a></p>

</div>



















<script src="<?php echo htmlspecialchars(asset_version_url('js/vendor/jquery-1.11.2.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/plugins.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/main.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>





<script type="text/javascript">

$(document).ready(function() {
    
    // navigation click actions 
    
    $('.scroll-link').on('click', function(event){
        
        event.preventDefault();
        
        var sectionID = $(this).attr("data-id");
        
        scrollToID('#' + sectionID, 750);
        
    });
    
    // scroll to top action
    
    $('.scroll-top').on('click', function(event) {
        
        event.preventDefault();
        
        $('html, body').animate({scrollTop:0}, 'slow');         
        
    });
    
    // mobile nav toggle
    
    $('#nav-toggle').on('click', function (event) {
        
        event.preventDefault();
        
        $('#main-nav').toggleClass("open");
        
    });
    
});

// scroll function

function scrollToID(id, speed){
    
    var offSet = 50;
    
    var targetOffset = $(id).offset().top - offSet;
    
    var mainNav = $('#main-nav');
    
    $('html,body').animate({scrollTop:targetOffset}, speed);
    
    if (mainNav.hasClass("open")) {
        
        mainNav.css("height", "1px").removeClass("in").addClass("collapse");
        
        mainNav.removeClass("open");
        
    }
    
}

if (typeof console === "undefined") {
    
    console = {
        
        log: function() { }
        
    };
    
}


function formaterDate(input) {
    var valeur = input.value;
    var chiffres = valeur.replace(/\D/g, ''); // Supprime tous les caractères non-numériques
    
    // Ajoute des '/' après le jour et le mois
    chiffres = chiffres.substring(0, 2) + (chiffres.length > 2 ? '/' : '') + chiffres.substring(2, 4) + (chiffres.length > 4 ? '/' : '') + chiffres.substring(4, 8);
    
    input.value = chiffres;
}


</script>

</body>

</html>"
            }
        },
        {
            "path": "pointages/css/tooplate-style.css",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 16415,
                "sha1": "28f9e70eb95b1a44dc27895ecef364aa5278587e",
                "content_b64": "/* doc-project | pointages/css/tooplate-style.css | Définit la mise en forme globale du site vitrine et de ses sections (header, hero, portfolio, blog, contact, footer) | Expose: aucun | Dépend de: ../img/*.png, bootstrap.min.css | Impacte: UI, responsive, navigation, modales/lightbox | Tables: aucune */
/*
Template 2089 Meteor

http://www.tooplate.com/view/2089-meteor

*/

* {
    margin: 0;
    padding: 0;
    font-family: 'Open Sans', arial, sans-serif;
}
ul {
	list-style: none;
}
p {
	color: #5a5a5a;
	font-size: 13px;
	letter-spacing: 0.25px;
	line-height: 24px;
}
.primary-button a {
	display: inline-block;
	background-color: #a2cf8d;
	padding: 12px 18px;
	color: #000;
	font-size: 18px;
	font-weight : bolder;
	letter-spacing: 1px;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
	transition: all 0.5s;
    width: 50%;
}
.primary-button a:hover {
	background-color: #5c9d7d;
}
.white-button a {
	display: inline-block;
	background-color: #fff;
	padding: 12px 18px;
	color: #121212;
	font-size: 11px;
	font-weight: 500;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
	transition: all 0.5s;
}
.d1{
  width: 50%;
  border: 1px solid white;
}
.white-button a:hover {
	background-color: #121212;
	color: #fff;
}
.section-heading h4 {
	text-align: center;
	font-size: 17px;
	text-transform: uppercase;
	font-weight: 700;
	color: #fff;
}
.section-heading h1 {
	text-align: center;
	font-size: 25px;
	text-transform: uppercase;
	font-weight: 700;
	color: #fff;
}
.section-heading .line-dec {
	width: 60px;
	height: 3px;
	background-color: #fff;
	margin: 0 auto;
}
.line-dec2 {
	width: 60px;
	height: 3px;
	background-color: #000;
	margin: 0 auto;
}
.header {
    position: fixed;
    top: 0;
    left: 0;
    width: 100%;
    padding: 0;
    z-index: 10000;
    transition: all 0.2s ease-in-out;
    height: 80px;
    background-color:transparent;  
    
    text-align: center;
    line-height: 40px;
}

.navbar-inverse .navbar-brand, .navbar-inverse .navbar-nav>li>a {
    text-shadow: none;
}

.navbar-brand {
	line-height: 80px;
	padding: 0px;
}

.navbar-nav>li>a {
	color: #fff!important;
	font-size: 13px;
	font-weight: 300;
	text-transform: uppercase;
	text-shadow: none;
	line-height: 80px;
	padding: 0px;
	letter-spacing: 0.5px;
}

.navbar-nav>li {
	margin-left: 45px;
}

.header.active .navbar-nav>li>a {
	font-size: 13px;
	font-weight: 300;
	text-transform: uppercase;
	text-shadow: none;
	color: #121212!important;
}

.header.active {
    background-color: rgba(250, 250, 250, 0.98);
    -webkit-box-shadow: 0 1px 5px rgba(0, 0, 0, 0.25);
    -moz-box-shadow: 0 1px 5px rgba(0, 0, 0, 0.25);
    box-shadow: 0 1px 5px rgba(0, 0, 0, 0.25);
}
.navbar-brand .logo{
	margin-top: 22px;
	background-image: url(../img/white_logo.png);
	display: inline-block;
	width: 125px;
	height: 32px;
	background-repeat: no-repeat;
}
.header.active .navbar-brand .logo{
	margin-top: 22px;
	background-image: url(../img/black_logo.png);
	display: inline-block;
	width: 125px;
	height: 32px;
	background-repeat: no-repeat;
}
#main-nav {
	background-color: transparent;
	background-image: none!important;
}
.header .navbar-brand {
    font-weight: 900;
}
.navbar-inverse {
	background-image: none;
	background-color: transparent;
}
.header .navbar {
    margin: 0;
    border: none;
}
.page-section {
    padding: 80px 0;
}
.navbar-inverse .navbar-toggle {
	margin-top: 22px;
    border-color: #a2cf8d;
    background-color: #a2cf8d;
}
.navbar-inverse .navbar-toggle:hover, .navbar-inverse .navbar-toggle:focus {
	background-color: #a2cf8d;
}

.navbar-toggle {
    position: relative;
    float: right;
    padding: 9px 10px;
    margin-top: 15px!important;
    margin-right: 15px;
    margin-bottom: 8px;
    background-color: transparent;
    background-image: none;
    border: 1px solid transparent;
    border-radius: 4px;
}
@media (min-width: 768px) {
    .header .navbar-brand {
        padding-left: 0;
    }
    .navbar-nav {
        float: right;
    }
}
@media (max-width: 480px) {
    .page-section h1 {
        font-size: 3em;
    }
    .page-section h2 {
        font-size: 2em;
    }
    .page-section p {
        font-size: 1em;
        margin: 0 0 20px;
    }
}



/* HERO SLIDER */

.slide-caption {
	color: #fff;
}

.slide-caption span {
	font-size: 14px;
	text-transform: uppercase;
}

.slide-caption h2 {
	font-size: 36px;
	text-transform: uppercase;
	font-weight: 900;
}

.slide-caption p {
	font-size: 13px;
	letter-spacing: 0.25px;
	line-height: 24px;
}

ul.cd-hero-slider {
	margin-bottom: 0px!important;
}

/* WHAT WE DO */

#about {
	background-color: #212121;
	padding: 20px 0px;
}

#about .section-heading {
	margin-bottom: 20px;
}

.service-item {
	cursor: pointer;
	background-color: #fff;
	text-align: center;
	padding: 30px 20px;
	transition: all 0.7s;
	box-shadow: 0px 0px 15px #cdcdcd;
}

.service-item:hover {
	background-color: #a2cf8d;
}

.service-item h4 {
	font-size: 16px;
	font-weight: 700;
	letter-spacing: 0.5px;
	margin-bottom: 15px;
	margin-top: 25px;
}

.service-item:hover h4 {
	color: #fff;
}

.service-item:hover p {
	color: #fff;
}

.first-service .icon {
	background-image: url(../img/first-service-main.png);
	height: 32px;
	width: 32px;
	display: inline-block;
}

.first-service:hover .icon {
	background-image: url(../img/first-service-white.png);
}

.second-service .icon {
	background-image: url(../img/second-service-main.png);
	height: 32px;
	width: 22px;
	display: inline-block;
}

.second-service:hover .icon {
	background-image: url(../img/second-service-white.png);
}

.third-service .icon {
	background-image: url(../img/third-service-main.png);
	height: 32px;
	width: 32px;
	display: inline-block;
}

.third-service:hover .icon {
	background-image: url(../img/third-service-white.png);
}

.fourth-service .icon {
	background-image: url(../img/fourth-service-main.png);
	height: 32px;
	width: 32px;
	display: inline-block;
}

.fourth-service:hover .icon {
	background-image: url(../img/fourth-service-white.png);
}


#what-we-do {
	background-image: url(../img/what-we-do-bg.png);
	width: 100%;
	background-size: cover;
	background-repeat: no-repeat;
	padding: 80px 0px;
	color: #fff;
}

#what-we-do .right-image img {
	width: 100%;
	overflow: hidden;
}

#what-we-do h4 {
	font-size: 24px;
	font-weight: 700;
	letter-spacing: 0.5px;
	line-height: 30px;
	margin-bottom: 25px;
}

#what-we-do p {
	margin-bottom: 30px;
	color: #fff;
}

#what-we-do ul li {
	display: inline-block;
	margin-right: 15px;
	margin-bottom: 15px;
}



/* PORTFOLIO */

#portfolio {
	text-align: center;
}

.projects-holder .mix {
  display: none;
}

.filter-categories {
  text-align: center;
  margin-bottom: 10px;
  margin-top: 60px;
}

.filter-categories ul li {
  margin: 0px 10px 15px 10px;
  display: inline-block;
}

.filter-categories ul li span {
  cursor: pointer;
  color: #5a5a5a;
  display: inline-block;
  font-size: 14px;
  text-transform: capitalize;
  font-weight: 400;
}

.filter-categories ul li.active span {
  color: #a2cf8d;
}

.project-item {
	margin-top: 30px;
}

.project-item .thumb {
	position: relative;
	text-align: center;
	display: inline-block;
}

.project-item .thumb img {
	width: 100%;
	overflow: hidden;
}

.project-item .hover-effect {
	position: absolute;
	background-color: rgba(162, 207, 141, 0.95);
	width: 100%;
	top: 0;
	bottom: 0;
	opacity: 0;
	visibility: hidden;
	transition: all 0.5s;
}

.project-item .hover-effect i {
	text-align: center;
	margin-top: 35%;
	display: inline-block;
	width: 46px;
	height: 46px;
	background-color: #fff;
	line-height: 46px;
	color: #a2cf8d;
	border-radius: 50%;
	font-size: 18px;
}

.project-item .hover-effect i:hover {
	background-color: #5c9d7d;
	color: #fff;
	transition: all 0.5s;
}

.project-item:hover .hover-effect {
	opacity: 1;
	visibility: visible;
}


/* BLOG */

#blog {
	background-image: url(../img/blog-bg.png);
	background-repeat: no-repeat;
	background-size: cover;
	width: 100%;
}

#blog .section-heading  {
	margin-bottom: 60px;
}

#blog .section-heading h4 {
	color: #fff;
}

#blog .section-heading .line-dec {
	background-color: #fff;
}

#blog .pop, .pop2, .pop3 {
	background-color: #fff;
}

.blog-item {
	cursor: pointer;
	margin-bottom: 30px;
}

.blog-item img {
	width: 100%;
	overflow: hidden;
}

.blog-item .thumb {
	position: relative;
}

.blog-item .text-content {
	position: absolute;
	bottom: 0;
	background-color: rgba(0, 0, 0, 0.5);
	width: 100%;
	padding: 12px 20px;
}

.blog-item .text-content h4 {
	font-size: 16px;
	font-weight: 700;
	color: #fff;
	letter-spacing: 0.5px;
}

.blog-item .text-content span {
	font-size: 12px;
	color: #fff;
}

.blog-item .text-content em {
	font-style: normal;
	font-weight: 700;
	margin-right: 5px;
}

.pop, .pop2, .pop3 {
	z-index: 9999;
	text-align: center;
	padding: 30px;
    display:none;
    position:fixed;
    top:25%;
    left:10.3%;
    width:80%;
    height: auto;
    box-shadow: 0px 0px 25px #7a7a7a;
}

.pop span, .pop2 span, .pop3 span {
	margin-bottom: 20px;
	cursor: pointer;
	width: 44px;
	height: 44px;
	display: inline-block;
	line-height: 44px;
	color: #fff;
	text-align: center;
	background-color: #a2cf8d;
}

.pop span:hover, .pop2 span:hover, .pop3 span:hover {
	background-color: #5c9d7d;
	transition: all 0.5s;
}



/* FUN FACTS */

#fun-facts {
	padding: 100px 0px;
	background-color: #fff;
}

#fun-facts .fact-item {
	background-color: #f4f4f4;
	padding: 30px;
	text-align: center;
}

.fact-item .counter {
	font-size: 24px;
	font-weight: 700;
	color: #a2cf8d;
}

.fact-item span {
	display: inline-block;
	margin-top: 10px;
	font-style: 13px;
	text-transform: uppercase;
	letter-spacing: 0.5px;
	color: #7a7a7a;
}



/* CONTACT US */

#contact .section-heading h4 {
	color: #fff;
}

#contact .section-heading .line-dec {
	background-color: #fff;
}

#contact .section-heading {
	margin-bottom: 60px;
}

#contact {
	
	background-repeat: no-repeat;
	width: 100%;
	background-size: cover;
}

#contact .map img {
	width: 100%;
	overflow: hidden;
}

#contact input {
	border-radius: 0px;
	padding-left: 15px;
	font-size: 13px;
	color: #fff;
	background-color: rgba(250, 250, 250, 0.1);
	outline: none;
	border: none;
	box-shadow: none;
	line-height: 50px;
	height: 50px;
	width: 100%;
	margin-bottom: 30px;
}

#contact textarea {
	border-radius: 0px;
	padding-left: 15px;
	font-size: 13px;
	color: #fff;
	background-color: rgba(250, 250, 250, 0.1);
	outline: none;
	border: none;
	box-shadow: none;
	height: 165px;
	max-height: 220px;
	width: 100%;
	margin-bottom: 25px;
}

#contact button {
	display: inline-block;
	background-color: #000099;
	padding: 12px 18px;
	color: #fff;
	font-size: 11px;
	letter-spacing: 1px;
	font-weight: 900;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
}



#contact button:hover {
	color: #fff;
	background-color: #0000cc;
	transition: all 0.3s;
}
#contact2 button {
	display: inline-block;
	background-color: #990000;
	padding: 12px 18px;
	color: #fff;
	font-size: 11px;
	letter-spacing: 1px;
	font-weight: 900;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
}



#contact2 button:hover {
	color: #fff;
	background-color: #cc0000;
	transition: all 0.3s;
}

/* FOOTER */

footer {
	background-color: #202628;
	padding: 20px;
}

footer p {
	font-size: 12px;
	color: #fff;
	margin-bottom: 0px;
	margin-top: 20px;
}

footer em {
	font-style: normal;
	font-weight: 700;
	color: #a2cf8d;
}

footer ul {
	margin-top: 15px;
	float: right;
}

footer ul li {
	display: inline-block;
	margin-left: 5px;
}

footer ul li a {
	font-size: 16px;
	width: 36px;
	height: 36px;
	line-height: 36px;
	text-align: center;
	display: inline-block;
	background-color: #a2cf8d;
	color: #202628;
	border-radius: 50%;
}

footer ul li a:hover {
	color: #202628;
	background-color: #5c9d7d;
	transition: all 0.5s;
}




/* LIGHT BOX */

body:after {
  display: none;
}

body.lb-disable-scrolling {
  overflow: hidden;
}

.lightboxOverlay {
  position: absolute;
  top: 0;
  left: 0;
  z-index: 99999;
  background-color: black;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=80);
  opacity: 0.8;
  display: none;
}

.lightbox {
  position: absolute;
  margin-top: 5%;
  left: 0;
  width: 100%;
  z-index: 100000;
  text-align: center;
  line-height: 0;
  font-weight: normal;
}

.lightbox .lb-image {
  display: block;
  height: auto;
  max-width: inherit;
  max-height: none;
  border-radius: 3px;

  /* Image border */
  border: 4px solid white;
}

.lightbox a img {
  border: none;
}

.lb-outerContainer {
  position: relative;
  *zoom: 1;
  width: 250px;
  height: 250px;
  margin: 0 auto;
  border-radius: 4px;

  /* Background color behind image.
     This is visible during transitions. */
  background-color: white;
}

.lb-outerContainer:after {
  content: "";
  display: table;
  clear: both;
}

.lb-loader {
  position: absolute;
  top: 43%;
  left: 0;
  height: 25%;
  width: 100%;
  text-align: center;
  line-height: 0;
}

.lb-cancel {
  display: block;
  width: 32px;
  height: 32px;
  margin: 0 auto;
}

.lb-nav {
  position: absolute;
  top: 0;
  left: 0;
  height: 100%;
  width: 100%;
  z-index: 10;
}

.lb-container > .nav {
  left: 0;
}

.lb-nav a {
  outline: none;
  background-image: url('data:image/gif;base64,R0lGODlhAQABAPAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw==');
}

.lb-prev, .lb-next {
  height: 100%;
  cursor: pointer;
  display: block;
}

.lb-nav a.lb-prev {
  width: 34%;
  left: 0;
  float: left;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=0);
  opacity: 0;
  -webkit-transition: opacity 0.6s;
  -moz-transition: opacity 0.6s;
  -o-transition: opacity 0.6s;
  transition: opacity 0.6s;
}

.lb-nav a.lb-prev:hover {
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100);
  opacity: 1;
}

.lb-nav a.lb-next {
  width: 64%;
  right: 0;
  float: right;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=0);
  opacity: 0;
  -webkit-transition: opacity 0.6s;
  -moz-transition: opacity 0.6s;
  -o-transition: opacity 0.6s;
  transition: opacity 0.6s;
}

.lb-nav a.lb-next:hover {
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100);
  opacity: 1;
}

.lb-dataContainer {
  margin: 0 auto;
  padding-top: 5px;
  *zoom: 1;
  width: 100%;
  -moz-border-radius-bottomleft: 4px;
  -webkit-border-bottom-left-radius: 4px;
  border-bottom-left-radius: 4px;
  -moz-border-radius-bottomright: 4px;
  -webkit-border-bottom-right-radius: 4px;
  border-bottom-right-radius: 4px;
}

.lb-dataContainer:after {
  content: "";
  display: table;
  clear: both;
}

.lb-data {
  padding: 0 4px;
  color: #ccc;
}

.lb-data .lb-details {
  width: 85%;
  float: left;
  text-align: left;
  line-height: 1.1em;
}

.lb-data .lb-caption {
  font-size: 13px;
  font-weight: bold;
  line-height: 1em;
}

.lb-data .lb-caption a {
  color: #4ae;
}

.lb-data .lb-number {
  display: block;
  clear: left;
  padding-bottom: 1em;
  font-size: 12px;
  color: #999999;
}

.lb-data .lb-close {
  display: block;
  float: right;
  width: 30px;
  height: 30px;
  text-align: right;
  outline: none;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=70);
  opacity: 0.7;
  -webkit-transition: opacity 0.2s;
  -moz-transition: opacity 0.2s;
  -o-transition: opacity 0.2s;
  transition: opacity 0.2s;
}

.lb-data .lb-close:hover {
  cursor: pointer;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100);
  opacity: 1;
}



/* RESPONSIVE */

@media (max-width: 767px){
	#main-nav {
		border-color: rgba(250, 250, 250, 0.95);
		margin-top: 16px;
		background-color: rgba(250, 250, 250, 0.95);
		box-shadow: 0px 5px 10px #cdcdcd;
	}
	.navbar-nav {
		padding: 0px;
		margin-top: 0px;
		margin-top: 0px;
	}
	.navbar-nav>li {
		margin-left: 0px;
		border-bottom: 1px solid #eee;
	}
	.navbar-nav>li:last-child {
		border-bottom: none;
	}
	.navbar-nav>li>a {
		line-height: 50px;
		color: #121212!important;
	}
}

@media (max-width: 768px){
	.cd-hero .slide-caption span {
		font-size: 13px;
	}

	.cd-hero .slide-caption h2 {
		font-size: 28px;
	}
}


@media (max-width: 1015px){
	.slide-caption .content-caption p {
		padding: 0 10%;
	}
	.service-item {
		margin-bottom: 30px;
	}
	#what-we-do .right-image {
		margin-top: 60px;
	}
	#blog .pop {
		top: 15%;
		height: 80%;
		overflow-y: scroll;
	}
	.fact-item {
		margin-bottom: 15px;
	}
	.map {
		margin-bottom: 45px;
	}
	footer {
		text-align: center;
	}
	footer ul {
		float: none;
	}
}"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 16493,
                "sha1": "fd90ed76516a08e5dd9f721e9bbf8a7bb597212c",
                "content_b64": "/* doc-project | pointages/css/tooplate-style.css | Définit la mise en forme globale du site vitrine et de ses sections (header, hero, portfolio, blog, contact, footer) en complément du CSS Bootstrap local, sans dépendance JavaScript Bootstrap. | Expose: aucun | Dépend de: ../img/*.png, bootstrap.min.css | Impacte: UI, responsive, navigation, modales/lightbox | Tables: aucune */
/*
Template 2089 Meteor

http://www.tooplate.com/view/2089-meteor

*/

* {
    margin: 0;
    padding: 0;
    font-family: 'Open Sans', arial, sans-serif;
}
ul {
	list-style: none;
}
p {
	color: #5a5a5a;
	font-size: 13px;
	letter-spacing: 0.25px;
	line-height: 24px;
}
.primary-button a {
	display: inline-block;
	background-color: #a2cf8d;
	padding: 12px 18px;
	color: #000;
	font-size: 18px;
	font-weight : bolder;
	letter-spacing: 1px;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
	transition: all 0.5s;
    width: 50%;
}
.primary-button a:hover {
	background-color: #5c9d7d;
}
.white-button a {
	display: inline-block;
	background-color: #fff;
	padding: 12px 18px;
	color: #121212;
	font-size: 11px;
	font-weight: 500;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
	transition: all 0.5s;
}
.d1{
  width: 50%;
  border: 1px solid white;
}
.white-button a:hover {
	background-color: #121212;
	color: #fff;
}
.section-heading h4 {
	text-align: center;
	font-size: 17px;
	text-transform: uppercase;
	font-weight: 700;
	color: #fff;
}
.section-heading h1 {
	text-align: center;
	font-size: 25px;
	text-transform: uppercase;
	font-weight: 700;
	color: #fff;
}
.section-heading .line-dec {
	width: 60px;
	height: 3px;
	background-color: #fff;
	margin: 0 auto;
}
.line-dec2 {
	width: 60px;
	height: 3px;
	background-color: #000;
	margin: 0 auto;
}
.header {
    position: fixed;
    top: 0;
    left: 0;
    width: 100%;
    padding: 0;
    z-index: 10000;
    transition: all 0.2s ease-in-out;
    height: 80px;
    background-color:transparent;  
    
    text-align: center;
    line-height: 40px;
}

.navbar-inverse .navbar-brand, .navbar-inverse .navbar-nav>li>a {
    text-shadow: none;
}

.navbar-brand {
	line-height: 80px;
	padding: 0px;
}

.navbar-nav>li>a {
	color: #fff!important;
	font-size: 13px;
	font-weight: 300;
	text-transform: uppercase;
	text-shadow: none;
	line-height: 80px;
	padding: 0px;
	letter-spacing: 0.5px;
}

.navbar-nav>li {
	margin-left: 45px;
}

.header.active .navbar-nav>li>a {
	font-size: 13px;
	font-weight: 300;
	text-transform: uppercase;
	text-shadow: none;
	color: #121212!important;
}

.header.active {
    background-color: rgba(250, 250, 250, 0.98);
    -webkit-box-shadow: 0 1px 5px rgba(0, 0, 0, 0.25);
    -moz-box-shadow: 0 1px 5px rgba(0, 0, 0, 0.25);
    box-shadow: 0 1px 5px rgba(0, 0, 0, 0.25);
}
.navbar-brand .logo{
	margin-top: 22px;
	background-image: url(../img/white_logo.png);
	display: inline-block;
	width: 125px;
	height: 32px;
	background-repeat: no-repeat;
}
.header.active .navbar-brand .logo{
	margin-top: 22px;
	background-image: url(../img/black_logo.png);
	display: inline-block;
	width: 125px;
	height: 32px;
	background-repeat: no-repeat;
}
#main-nav {
	background-color: transparent;
	background-image: none!important;
}
.header .navbar-brand {
    font-weight: 900;
}
.navbar-inverse {
	background-image: none;
	background-color: transparent;
}
.header .navbar {
    margin: 0;
    border: none;
}
.page-section {
    padding: 80px 0;
}
.navbar-inverse .navbar-toggle {
	margin-top: 22px;
    border-color: #a2cf8d;
    background-color: #a2cf8d;
}
.navbar-inverse .navbar-toggle:hover, .navbar-inverse .navbar-toggle:focus {
	background-color: #a2cf8d;
}

.navbar-toggle {
    position: relative;
    float: right;
    padding: 9px 10px;
    margin-top: 15px!important;
    margin-right: 15px;
    margin-bottom: 8px;
    background-color: transparent;
    background-image: none;
    border: 1px solid transparent;
    border-radius: 4px;
}
@media (min-width: 768px) {
    .header .navbar-brand {
        padding-left: 0;
    }
    .navbar-nav {
        float: right;
    }
}
@media (max-width: 480px) {
    .page-section h1 {
        font-size: 3em;
    }
    .page-section h2 {
        font-size: 2em;
    }
    .page-section p {
        font-size: 1em;
        margin: 0 0 20px;
    }
}



/* HERO SLIDER */

.slide-caption {
	color: #fff;
}

.slide-caption span {
	font-size: 14px;
	text-transform: uppercase;
}

.slide-caption h2 {
	font-size: 36px;
	text-transform: uppercase;
	font-weight: 900;
}

.slide-caption p {
	font-size: 13px;
	letter-spacing: 0.25px;
	line-height: 24px;
}

ul.cd-hero-slider {
	margin-bottom: 0px!important;
}

/* WHAT WE DO */

#about {
	background-color: #212121;
	padding: 20px 0px;
}

#about .section-heading {
	margin-bottom: 20px;
}

.service-item {
	cursor: pointer;
	background-color: #fff;
	text-align: center;
	padding: 30px 20px;
	transition: all 0.7s;
	box-shadow: 0px 0px 15px #cdcdcd;
}

.service-item:hover {
	background-color: #a2cf8d;
}

.service-item h4 {
	font-size: 16px;
	font-weight: 700;
	letter-spacing: 0.5px;
	margin-bottom: 15px;
	margin-top: 25px;
}

.service-item:hover h4 {
	color: #fff;
}

.service-item:hover p {
	color: #fff;
}

.first-service .icon {
	background-image: url(../img/first-service-main.png);
	height: 32px;
	width: 32px;
	display: inline-block;
}

.first-service:hover .icon {
	background-image: url(../img/first-service-white.png);
}

.second-service .icon {
	background-image: url(../img/second-service-main.png);
	height: 32px;
	width: 22px;
	display: inline-block;
}

.second-service:hover .icon {
	background-image: url(../img/second-service-white.png);
}

.third-service .icon {
	background-image: url(../img/third-service-main.png);
	height: 32px;
	width: 32px;
	display: inline-block;
}

.third-service:hover .icon {
	background-image: url(../img/third-service-white.png);
}

.fourth-service .icon {
	background-image: url(../img/fourth-service-main.png);
	height: 32px;
	width: 32px;
	display: inline-block;
}

.fourth-service:hover .icon {
	background-image: url(../img/fourth-service-white.png);
}


#what-we-do {
	background-image: url(../img/what-we-do-bg.png);
	width: 100%;
	background-size: cover;
	background-repeat: no-repeat;
	padding: 80px 0px;
	color: #fff;
}

#what-we-do .right-image img {
	width: 100%;
	overflow: hidden;
}

#what-we-do h4 {
	font-size: 24px;
	font-weight: 700;
	letter-spacing: 0.5px;
	line-height: 30px;
	margin-bottom: 25px;
}

#what-we-do p {
	margin-bottom: 30px;
	color: #fff;
}

#what-we-do ul li {
	display: inline-block;
	margin-right: 15px;
	margin-bottom: 15px;
}



/* PORTFOLIO */

#portfolio {
	text-align: center;
}

.projects-holder .mix {
  display: none;
}

.filter-categories {
  text-align: center;
  margin-bottom: 10px;
  margin-top: 60px;
}

.filter-categories ul li {
  margin: 0px 10px 15px 10px;
  display: inline-block;
}

.filter-categories ul li span {
  cursor: pointer;
  color: #5a5a5a;
  display: inline-block;
  font-size: 14px;
  text-transform: capitalize;
  font-weight: 400;
}

.filter-categories ul li.active span {
  color: #a2cf8d;
}

.project-item {
	margin-top: 30px;
}

.project-item .thumb {
	position: relative;
	text-align: center;
	display: inline-block;
}

.project-item .thumb img {
	width: 100%;
	overflow: hidden;
}

.project-item .hover-effect {
	position: absolute;
	background-color: rgba(162, 207, 141, 0.95);
	width: 100%;
	top: 0;
	bottom: 0;
	opacity: 0;
	visibility: hidden;
	transition: all 0.5s;
}

.project-item .hover-effect i {
	text-align: center;
	margin-top: 35%;
	display: inline-block;
	width: 46px;
	height: 46px;
	background-color: #fff;
	line-height: 46px;
	color: #a2cf8d;
	border-radius: 50%;
	font-size: 18px;
}

.project-item .hover-effect i:hover {
	background-color: #5c9d7d;
	color: #fff;
	transition: all 0.5s;
}

.project-item:hover .hover-effect {
	opacity: 1;
	visibility: visible;
}


/* BLOG */

#blog {
	background-image: url(../img/blog-bg.png);
	background-repeat: no-repeat;
	background-size: cover;
	width: 100%;
}

#blog .section-heading  {
	margin-bottom: 60px;
}

#blog .section-heading h4 {
	color: #fff;
}

#blog .section-heading .line-dec {
	background-color: #fff;
}

#blog .pop, .pop2, .pop3 {
	background-color: #fff;
}

.blog-item {
	cursor: pointer;
	margin-bottom: 30px;
}

.blog-item img {
	width: 100%;
	overflow: hidden;
}

.blog-item .thumb {
	position: relative;
}

.blog-item .text-content {
	position: absolute;
	bottom: 0;
	background-color: rgba(0, 0, 0, 0.5);
	width: 100%;
	padding: 12px 20px;
}

.blog-item .text-content h4 {
	font-size: 16px;
	font-weight: 700;
	color: #fff;
	letter-spacing: 0.5px;
}

.blog-item .text-content span {
	font-size: 12px;
	color: #fff;
}

.blog-item .text-content em {
	font-style: normal;
	font-weight: 700;
	margin-right: 5px;
}

.pop, .pop2, .pop3 {
	z-index: 9999;
	text-align: center;
	padding: 30px;
    display:none;
    position:fixed;
    top:25%;
    left:10.3%;
    width:80%;
    height: auto;
    box-shadow: 0px 0px 25px #7a7a7a;
}

.pop span, .pop2 span, .pop3 span {
	margin-bottom: 20px;
	cursor: pointer;
	width: 44px;
	height: 44px;
	display: inline-block;
	line-height: 44px;
	color: #fff;
	text-align: center;
	background-color: #a2cf8d;
}

.pop span:hover, .pop2 span:hover, .pop3 span:hover {
	background-color: #5c9d7d;
	transition: all 0.5s;
}



/* FUN FACTS */

#fun-facts {
	padding: 100px 0px;
	background-color: #fff;
}

#fun-facts .fact-item {
	background-color: #f4f4f4;
	padding: 30px;
	text-align: center;
}

.fact-item .counter {
	font-size: 24px;
	font-weight: 700;
	color: #a2cf8d;
}

.fact-item span {
	display: inline-block;
	margin-top: 10px;
	font-style: 13px;
	text-transform: uppercase;
	letter-spacing: 0.5px;
	color: #7a7a7a;
}



/* CONTACT US */

#contact .section-heading h4 {
	color: #fff;
}

#contact .section-heading .line-dec {
	background-color: #fff;
}

#contact .section-heading {
	margin-bottom: 60px;
}

#contact {
	
	background-repeat: no-repeat;
	width: 100%;
	background-size: cover;
}

#contact .map img {
	width: 100%;
	overflow: hidden;
}

#contact input {
	border-radius: 0px;
	padding-left: 15px;
	font-size: 13px;
	color: #fff;
	background-color: rgba(250, 250, 250, 0.1);
	outline: none;
	border: none;
	box-shadow: none;
	line-height: 50px;
	height: 50px;
	width: 100%;
	margin-bottom: 30px;
}

#contact textarea {
	border-radius: 0px;
	padding-left: 15px;
	font-size: 13px;
	color: #fff;
	background-color: rgba(250, 250, 250, 0.1);
	outline: none;
	border: none;
	box-shadow: none;
	height: 165px;
	max-height: 220px;
	width: 100%;
	margin-bottom: 25px;
}

#contact button {
	display: inline-block;
	background-color: #000099;
	padding: 12px 18px;
	color: #fff;
	font-size: 11px;
	letter-spacing: 1px;
	font-weight: 900;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
}



#contact button:hover {
	color: #fff;
	background-color: #0000cc;
	transition: all 0.3s;
}
#contact2 button {
	display: inline-block;
	background-color: #990000;
	padding: 12px 18px;
	color: #fff;
	font-size: 11px;
	letter-spacing: 1px;
	font-weight: 900;
	text-transform: uppercase;
	text-decoration: none;
	border-radius: 20px;
}



#contact2 button:hover {
	color: #fff;
	background-color: #cc0000;
	transition: all 0.3s;
}

/* FOOTER */

footer {
	background-color: #202628;
	padding: 20px;
}

footer p {
	font-size: 12px;
	color: #fff;
	margin-bottom: 0px;
	margin-top: 20px;
}

footer em {
	font-style: normal;
	font-weight: 700;
	color: #a2cf8d;
}

footer ul {
	margin-top: 15px;
	float: right;
}

footer ul li {
	display: inline-block;
	margin-left: 5px;
}

footer ul li a {
	font-size: 16px;
	width: 36px;
	height: 36px;
	line-height: 36px;
	text-align: center;
	display: inline-block;
	background-color: #a2cf8d;
	color: #202628;
	border-radius: 50%;
}

footer ul li a:hover {
	color: #202628;
	background-color: #5c9d7d;
	transition: all 0.5s;
}




/* LIGHT BOX */

body:after {
  display: none;
}

body.lb-disable-scrolling {
  overflow: hidden;
}

.lightboxOverlay {
  position: absolute;
  top: 0;
  left: 0;
  z-index: 99999;
  background-color: black;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=80);
  opacity: 0.8;
  display: none;
}

.lightbox {
  position: absolute;
  margin-top: 5%;
  left: 0;
  width: 100%;
  z-index: 100000;
  text-align: center;
  line-height: 0;
  font-weight: normal;
}

.lightbox .lb-image {
  display: block;
  height: auto;
  max-width: inherit;
  max-height: none;
  border-radius: 3px;

  /* Image border */
  border: 4px solid white;
}

.lightbox a img {
  border: none;
}

.lb-outerContainer {
  position: relative;
  *zoom: 1;
  width: 250px;
  height: 250px;
  margin: 0 auto;
  border-radius: 4px;

  /* Background color behind image.
     This is visible during transitions. */
  background-color: white;
}

.lb-outerContainer:after {
  content: "";
  display: table;
  clear: both;
}

.lb-loader {
  position: absolute;
  top: 43%;
  left: 0;
  height: 25%;
  width: 100%;
  text-align: center;
  line-height: 0;
}

.lb-cancel {
  display: block;
  width: 32px;
  height: 32px;
  margin: 0 auto;
}

.lb-nav {
  position: absolute;
  top: 0;
  left: 0;
  height: 100%;
  width: 100%;
  z-index: 10;
}

.lb-container > .nav {
  left: 0;
}

.lb-nav a {
  outline: none;
  background-image: url('data:image/gif;base64,R0lGODlhAQABAPAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw==');
}

.lb-prev, .lb-next {
  height: 100%;
  cursor: pointer;
  display: block;
}

.lb-nav a.lb-prev {
  width: 34%;
  left: 0;
  float: left;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=0);
  opacity: 0;
  -webkit-transition: opacity 0.6s;
  -moz-transition: opacity 0.6s;
  -o-transition: opacity 0.6s;
  transition: opacity 0.6s;
}

.lb-nav a.lb-prev:hover {
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100);
  opacity: 1;
}

.lb-nav a.lb-next {
  width: 64%;
  right: 0;
  float: right;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=0);
  opacity: 0;
  -webkit-transition: opacity 0.6s;
  -moz-transition: opacity 0.6s;
  -o-transition: opacity 0.6s;
  transition: opacity 0.6s;
}

.lb-nav a.lb-next:hover {
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100);
  opacity: 1;
}

.lb-dataContainer {
  margin: 0 auto;
  padding-top: 5px;
  *zoom: 1;
  width: 100%;
  -moz-border-radius-bottomleft: 4px;
  -webkit-border-bottom-left-radius: 4px;
  border-bottom-left-radius: 4px;
  -moz-border-radius-bottomright: 4px;
  -webkit-border-bottom-right-radius: 4px;
  border-bottom-right-radius: 4px;
}

.lb-dataContainer:after {
  content: "";
  display: table;
  clear: both;
}

.lb-data {
  padding: 0 4px;
  color: #ccc;
}

.lb-data .lb-details {
  width: 85%;
  float: left;
  text-align: left;
  line-height: 1.1em;
}

.lb-data .lb-caption {
  font-size: 13px;
  font-weight: bold;
  line-height: 1em;
}

.lb-data .lb-caption a {
  color: #4ae;
}

.lb-data .lb-number {
  display: block;
  clear: left;
  padding-bottom: 1em;
  font-size: 12px;
  color: #999999;
}

.lb-data .lb-close {
  display: block;
  float: right;
  width: 30px;
  height: 30px;
  text-align: right;
  outline: none;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=70);
  opacity: 0.7;
  -webkit-transition: opacity 0.2s;
  -moz-transition: opacity 0.2s;
  -o-transition: opacity 0.2s;
  transition: opacity 0.2s;
}

.lb-data .lb-close:hover {
  cursor: pointer;
  filter: progid:DXImageTransform.Microsoft.Alpha(Opacity=100);
  opacity: 1;
}



/* RESPONSIVE */

@media (max-width: 767px){
	#main-nav {
		border-color: rgba(250, 250, 250, 0.95);
		margin-top: 16px;
		background-color: rgba(250, 250, 250, 0.95);
		box-shadow: 0px 5px 10px #cdcdcd;
	}
	.navbar-nav {
		padding: 0px;
		margin-top: 0px;
		margin-top: 0px;
	}
	.navbar-nav>li {
		margin-left: 0px;
		border-bottom: 1px solid #eee;
	}
	.navbar-nav>li:last-child {
		border-bottom: none;
	}
	.navbar-nav>li>a {
		line-height: 50px;
		color: #121212!important;
	}
}

@media (max-width: 768px){
	.cd-hero .slide-caption span {
		font-size: 13px;
	}

	.cd-hero .slide-caption h2 {
		font-size: 28px;
	}
}


@media (max-width: 1015px){
	.slide-caption .content-caption p {
		padding: 0 10%;
	}
	.service-item {
		margin-bottom: 30px;
	}
	#what-we-do .right-image {
		margin-top: 60px;
	}
	#blog .pop {
		top: 15%;
		height: 80%;
		overflow-y: scroll;
	}
	.fact-item {
		margin-bottom: 15px;
	}
	.map {
		margin-bottom: 45px;
	}
	footer {
		text-align: center;
	}
	footer ul {
		float: none;
	}
}"
            }
        },
        {
            "path": "pointages/index.php",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 22209,
                "sha1": "546c34eb127e511fac7e6278d28e83f529e69734",
                "content_b64": "<?php
/* doc-project | pointages/index.php | Affiche le tableau de pointage des salariés et protège l’accès via autorisation permanente par appareil. | Expose: asset_version, openEmployeeModal, closeEmployeeModal, submitEmployeePunch, goToEmployeeDetails | Dépend de: pointages/config.php, includes/device_auth.php, connexion.php, admin.php, enregistrement_arrivee.php, enregistrement_depart.php, pointage.php, js/vendor/jquery-1.11.2.min.js, js/vendor/bootstrap.min.js, js/plugins.js, js/main.js, js/driver-modal.js | Impacte: session PHP, cookie d’appareil, redirection d’accès, lecture des pointages/employés, modales UI, appels AJAX | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_utilisateurs(UserID,Nom,Prenom,Statut), z_ptg_aqp_pointages(PointageID,UserID,DateHeureEntree,DateHeureSortie) */
// Démarrage ou reprise de la session
session_start();
date_default_timezone_set('Europe/Paris');

require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_device_authorized($pdo);

?>


<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">



<title>Pointages Salariés</title>



<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">


<?php
function asset_version(string $relativePath): string
{
  $fullPath = __DIR__ . '/' . ltrim($relativePath, '/');
  $version = file_exists($fullPath) ? filemtime($fullPath) : time();
  return $relativePath . '?v=' . (int)$version;
}
?>

<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/bootstrap.min.css'), ENT_QUOTES, 'UTF-8'); ?>">

<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/bootstrap-theme.min.css'), ENT_QUOTES, 'UTF-8'); ?>">

<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/fontAwesome.css'), ENT_QUOTES, 'UTF-8'); ?>">

<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/fontAwesome.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/hero-slider.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/tooplate-style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/home-buttons.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version('css/driver-modal.css'), ENT_QUOTES, 'UTF-8'); ?>">

<link href="https://fonts.googleapis.com/css?family=Open+Sans:300,400,600,700,800" rel="stylesheet">

<script src="<?php echo htmlspecialchars(asset_version('js/vendor/modernizr-2.8.3-respond-1.4.2.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>


<script src="<?php echo htmlspecialchars(asset_version('js/vendor/modernizr-2.8.3-respond-1.4.2.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>

</head>

<body class="page-index theme-dark">



<div id="about" class="">

<div class="section-heading">
<h1>Pointages salariés</h1>
<div class="line-dec"></div>
<center>
<br>
<font color="black" size="4" >
<strong>
<?php
// Récupère le dernier pointage de chaque employé pour déterminer s'il est "pointé"
$currentMonth = (int)date('n');
$currentYear = (int)date('Y');
$nowParis = date('Y-m-d H:i:s');

$stmt = $pdo->prepare("
  SELECT
    u.UserID,
    u.Nom,
    u.Prenom,
    p.DateHeureEntree,
    p.DateHeureSortie,
    mh.seconds_worked,
    mh.max_punch_seconds
  FROM z_ptg_aqp_utilisateurs u
  LEFT JOIN z_ptg_aqp_pointages p
    ON p.PointageID = (
      SELECT p2.PointageID
      FROM z_ptg_aqp_pointages p2
      WHERE p2.UserID = u.UserID
      ORDER BY p2.DateHeureEntree DESC, p2.PointageID DESC
      LIMIT 1
    )
  LEFT JOIN (
    SELECT
      UserID,
      SUM(
        TIMESTAMPDIFF(
          SECOND,
          DateHeureEntree,
          COALESCE(DateHeureSortie, :nowParisSum)
        )
      ) AS seconds_worked,
      MAX(
        TIMESTAMPDIFF(
          SECOND,
          DateHeureEntree,
          COALESCE(DateHeureSortie, :nowParisMax)
        )
      ) AS max_punch_seconds
    FROM z_ptg_aqp_pointages
    WHERE MONTH(DateHeureEntree) = :currentMonth
      AND YEAR(DateHeureEntree) = :currentYear
    GROUP BY UserID
  ) mh ON mh.UserID = u.UserID
  WHERE u.Statut = 'actif'
");
$stmt->execute([
  ':nowParisSum' => $nowParis,
  ':nowParisMax' => $nowParis,
  ':currentMonth' => $currentMonth,
  ':currentYear' => $currentYear,
]);
$employesActifs = $stmt->fetchAll(PDO::FETCH_ASSOC);

echo '<table class="employee-grid">';
$compteur = 0;

foreach ($employesActifs as $employe) {
  // Logique similaire pour afficher les employés
  if ($compteur % 2 == 0) {
    echo '<tr>';
  }
  $nomRaw = isset($employe['Nom']) ? (string)$employe['Nom'] : '';
  $prenomRaw = isset($employe['Prenom']) ? (string)$employe['Prenom'] : '';
  $nom = mb_strtoupper($nomRaw, 'UTF-8');
  $prenom = mb_strtoupper($prenomRaw, 'UTF-8');
  $prenomJs = htmlspecialchars(
    json_encode($prenomRaw, JSON_UNESCAPED_UNICODE | JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT),
    ENT_QUOTES,
    'UTF-8'
  );

  // Un employé est "pointé" s'il a un dernier pointage dont DateHeureSortie est NULL
  // (et qu'il existe bien une DateHeureEntree).
  $isPointed = (!empty($employe['DateHeureEntree']) && $employe['DateHeureSortie'] === null);

  // Détection "pointage > 8H" :
  // - via la durée max des pointages du mois (max_punch_seconds),
  // - et via le dernier pointage (en cours ou terminé), même s'il est hors mois (cas rare, mais robuste).
  $maxPunchSeconds = isset($employe['max_punch_seconds']) ? (int)$employe['max_punch_seconds'] : 0;
  $lastPunchSeconds = 0;
  if (!empty($employe['DateHeureEntree'])) {
    $startTs = strtotime($employe['DateHeureEntree']);
    $endTs = ($employe['DateHeureSortie'] === null || $employe['DateHeureSortie'] === '')
      ? strtotime($nowParis)
      : strtotime($employe['DateHeureSortie']);
    if ($startTs !== false && $endTs !== false && $endTs >= $startTs) {
      $lastPunchSeconds = (int)($endTs - $startTs);
    }
  }
  $hasPunchOver8h = ($maxPunchSeconds >= 28800) || ($lastPunchSeconds >= 28800);

  $btnClass = $isPointed ? 'employee-btn status-pointed' : 'employee-btn status-depointed';
  if ($hasPunchOver8h) {
    $btnClass .= ' status-problem';
  }
  // Icône uniquement quand pointé
  $iconHtml = '';
  if ($isPointed) {
    $iconHtml = '<i class="fa fa-check-circle status-icon" aria-hidden="true" title="Pointé"></i>';
  }

  $secondsWorked = isset($employe['seconds_worked']) ? (float)$employe['seconds_worked'] : 0.0;
  $monthHours = $secondsWorked > 0 ? ($secondsWorked / 3600) : 0.0;
  $monthHoursText = number_format($monthHours, 2, '.', '') . 'H';
  $monthHoursHtml = '<span class="month-hours">' . htmlspecialchars($monthHoursText, ENT_QUOTES, 'UTF-8') . '</span>';

  echo '<td class="employee-grid-cell">'
     . '<button type="button" class="' . $btnClass . '" onclick="openEmployeeModal(' . (int)$employe['UserID'] . ', ' . ($isPointed ? '1' : '0') . ', ' . $prenomJs . ')">'
     . $iconHtml
     . '<span class="employee-name">' . htmlspecialchars($nom, ENT_QUOTES, 'UTF-8') . '</span>'
     . '<span class="employee-firstname">' . htmlspecialchars($prenom, ENT_QUOTES, 'UTF-8') . '</span>'
     . $monthHoursHtml
     . '</button>'
     . '</td>';

  if ($compteur % 2 == 1) {
    echo '</tr>';
  }
  $compteur++;
}
 
 echo '</table>';
 ?>
 
 <div class="driver-phone-actions" style="text-align:center; margin:18px 0 6px;">
   <button
     type="button"
     id="driverPhoneButton"
     class="driver-phone-trigger"
     onclick="openDriverPhoneModal()"
   >
     Envoyer SMS livreur
   </button>
 </div>

 </strong>
 </font>
 </center>
 </div>
 </div>


<br><br>

<p><a href="admin.php">Gestion employés</a></p><br>





</div>




<!-- Modal PIN (design refait) -->
<div id="modalBackground" onclick="closePinModal()" aria-hidden="true"></div>
<div id="pinModal" role="dialog" aria-modal="true" aria-labelledby="pinModalTitle">
  <form id="pinForm" onsubmit="verifierPin(); return false;">
    <h2 id="pinModalTitle" class="pin-modal-title">Code PIN</h2>
    <p class="pin-modal-subtitle">Saisissez votre code à 4 chiffres</p>

    <div class="pin-input-wrapper" onclick="focusPinInput()">
      <label for="pinInput" class="sr-only">Code PIN (4 chiffres)</label>
      <div class="pin-boxes" aria-hidden="true">
        <div class="pin-box" data-idx="0"></div>
        <div class="pin-box" data-idx="1"></div>
        <div class="pin-box" data-idx="2"></div>
        <div class="pin-box" data-idx="3"></div>
      </div>
      <input
        type="tel"
        id="pinInput"
        class="pin-hidden-input"
        inputmode="numeric"
        pattern="[0-9]*"
        maxlength="4"
        autocomplete="off"
        autocapitalize="off"
        autocorrect="off"
        spellcheck="false"
        title="Seulement des chiffres."
        value=""
      >
    </div>

    <div class="pin-modal-actions">
      <button type="button" class="pin-modal-btn cancel" onclick="closePinModal()">Annuler</button>
      <button type="submit" class="pin-modal-btn confirm">Valider</button>
    </div>
  </form>
</div>

<!-- Modal Actions Employé (Arrivée/Départ + Voir détail) -->
<div id="employeeModalBackground" onclick="closeEmployeeModal()"></div>
<div id="employeeActionModal" role="dialog" aria-modal="true" aria-labelledby="employeeActionTitle">
  <h2 id="employeeActionTitle">Actions</h2>
  <button type="button" id="employeePunchButton" class="employee-modal-btn punch" onclick="submitEmployeePunch()"></button>
  <button type="button" class="employee-modal-btn details" onclick="goToEmployeeDetails()">Voir détail des heures</button>
</div>

<!-- Modal Livreur (sélection + envoi SMS) -->
<div id="driverModalBackground" class="driver-modal-backdrop" onclick="closeDriverPhoneModal()" aria-hidden="true"></div>
<div id="driverPhoneModal" class="driver-modal" role="dialog" aria-modal="true" aria-labelledby="driverPhoneModalTitle" aria-hidden="true">
  <form id="driverPhoneForm" onsubmit="submitDriverPhone(); return false;">
    <div class="driver-modal-header">
      <div class="driver-modal-header-text">
        <h2 id="driverPhoneModalTitle" class="driver-modal-title">Envoyer SMS livreur</h2>
        <p class="driver-modal-subtitle">Sélectionnez un livreur pour le point de vente de l’appareil</p>
      </div>
      <button type="button" class="driver-modal-close" onclick="closeDriverPhoneModal()" aria-label="Fermer">×</button>
    </div>

    <div class="driver-field">
      <label for="driverSelect" class="driver-label">Livreur</label>
      <select id="driverSelect" class="driver-select" required>
        <option value="">Chargement des livreurs…</option>
      </select>
      <small id="driverSelectedPhoneHelp" class="driver-help">Sélectionnez un livreur pour afficher son numéro.</small>
    </div>

    <div class="driver-field">
      <label for="driverPointVente" class="driver-label">Point de vente</label>
      <select id="driverPointVente" class="driver-select" disabled>
        <option value="lancon">Lançon</option>
        <option value="pelissanne">Pélissanne</option>
      </select>
    </div>

    <div class="driver-modal-actions driver-modal-actions-split">
      <button type="button" class="driver-modal-btn secondary" onclick="openDriverManageModal()">Gérer les livreurs</button>
    </div>

    <div class="driver-modal-actions">
      <button type="button" class="driver-modal-btn cancel" onclick="closeDriverPhoneModal()">Annuler</button>
      <button type="submit" class="driver-modal-btn confirm">Envoyer SMS</button>
    </div>
  </form>
</div>

<!-- Modal gestion livreurs -->
<div id="driverManageModalBackground" class="driver-modal-backdrop" onclick="closeDriverManageModal()" aria-hidden="true"></div>
<div id="driverManageModal" class="driver-modal driver-manage-modal" role="dialog" aria-modal="true" aria-labelledby="driverManageModalTitle" aria-hidden="true">
  <div class="driver-modal-header">
    <div class="driver-modal-header-text">
      <h2 id="driverManageModalTitle" class="driver-modal-title">Gérer les livreurs</h2>
      <p class="driver-modal-subtitle">Ajout, modification et suppression pour le point de vente courant</p>
    </div>
    <button type="button" class="driver-modal-close" onclick="closeDriverManageModal()" aria-label="Fermer">×</button>
  </div>

  <form id="driverManageForm">
    <input type="hidden" id="driverManageMode" value="create">
    <input type="hidden" id="driverEditOriginalPhone" value="">

    <div class="driver-field">
      <label for="driverManageSelect" class="driver-label">Livreur existant</label>
      <select id="driverManageSelect" class="driver-select">
        <option value="">Nouveau livreur</option>
      </select>
    </div>

    <div class="driver-field">
      <label for="driverManageName" class="driver-label">Nom affiché</label>
      <input
        type="text"
        id="driverManageName"
        class="driver-input"
        maxlength="120"
        placeholder="Ex: Karim"
        autocomplete="off"
      >
    </div>

    <div class="driver-field">
      <label for="driverManagePhone" class="driver-label">Téléphone</label>
      <input
        type="tel"
        id="driverManagePhone"
        class="driver-input"
        inputmode="numeric"
        pattern="[0-9]*"
        maxlength="16"
        placeholder="Ex: 0618529375"
        autocomplete="off"
        autocapitalize="off"
        autocorrect="off"
        spellcheck="false"
      >
      <small class="driver-help">Format accepté : 0618529375, 33618529375 ou 0033618529375.</small>
    </div>

    <div class="driver-modal-actions">
      <button type="button" class="driver-modal-btn danger" onclick="deleteSelectedDriver()">Supprimer</button>
      <button type="submit" class="driver-modal-btn confirm">Enregistrer</button>
    </div>
    <div class="driver-modal-actions">
      <button type="button" class="driver-modal-btn cancel" onclick="closeDriverManageModal()">Fermer</button>
    </div>
  </form>
</div>

<script src="//ajax.googleapis.com/ajax/libs/jquery/1.11.2/jquery.min.js"></script>









<script src="//ajax.googleapis.com/ajax/libs/jquery/1.11.2/jquery.min.js"></script>

<script>window.jQuery || document.write('<script src="<?php echo addslashes(asset_version('js/vendor/jquery-1.11.2.min.js')); ?>"><\/script>')</script>



<script src="<?php echo htmlspecialchars(asset_version('js/vendor/bootstrap.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>

<script src="<?php echo htmlspecialchars(asset_version('js/vendor/bootstrap.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>

<script src="<?php echo htmlspecialchars(asset_version('js/plugins.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>

<script src="<?php echo htmlspecialchars(asset_version('js/main.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>

<script src="<?php echo htmlspecialchars(asset_version('js/driver-modal.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>



<script type="text/javascript">

$(document).ready(function() {
  
  // navigation click actions 
  
  $('.scroll-link').on('click', function(event){
    
    event.preventDefault();
    
    var sectionID = $(this).attr("data-id");
    
    scrollToID('#' + sectionID, 750);
    
  });
  
  // scroll to top action
  
  $('.scroll-top').on('click', function(event) {
    
    event.preventDefault();
    
    $('html, body').animate({scrollTop:0}, 'slow');         
    
  });
  
  // mobile nav toggle
  
  $('#nav-toggle').on('click', function (event) {
    
    event.preventDefault();
    
    $('#main-nav').toggleClass("open");
    
  });
  
});

// scroll function

function scrollToID(id, speed){
  
  var offSet = 50;
  
  var targetOffset = $(id).offset().top - offSet;
  
  var mainNav = $('#main-nav');
  
  $('html,body').animate({scrollTop:targetOffset}, speed);
  
  if (mainNav.hasClass("open")) {
    
    mainNav.css("height", "1px").removeClass("in").addClass("collapse");
    
    mainNav.removeClass("open");
    
  }
  
}

if (typeof console === "undefined") {
  
  console = {
    
    log: function() { }
    
  };
  
}


function afficherModal() {
    window.location.href = "admin.php";
}


//function afficherModal(pointageId) {
  // Enregistrez l'ID du pointage pour l'utiliser plus tard
  //window.pointageIdPourModification = pointageId;
  // Affichez la fenêtre modale
  //document.getElementById('pinModal').style.display = 'block';
//}

function sanitizePinValue(val) {
  return String(val || '').replace(/\D/g, '').slice(0, 4);
}

function updatePinBoxes(pin) {
  var boxes = document.querySelectorAll('#pinModal .pin-box');
  for (var i = 0; i < boxes.length; i++) {
    boxes[i].textContent = (i < pin.length) ? '★' : '';
  }
}

function focusPinInput() {
  var input = document.getElementById('pinInput');
  if (!input) return;
  try { input.focus(); } catch (e) {}
}

function resetPinModal() {
  var input = document.getElementById('pinInput');
  if (input) input.value = '';
  updatePinBoxes('');
}

function openPinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (!modal || !bg) return;

  modal.style.display = 'block';
  bg.style.display = 'block';
  resetPinModal();

  window.setTimeout(function() {
    focusPinInput();
  }, 10);
}

function closePinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (modal) modal.style.display = 'none';
  if (bg) bg.style.display = 'none';
  resetPinModal();
}

// Bind input once (digits only + sync UI boxes)
(function bindPinOnce() {
  var input = document.getElementById('pinInput');
  if (!input || input.getAttribute('data-bound') === '1') return;
  input.setAttribute('data-bound', '1');
  input.addEventListener('input', function() {
    var clean = sanitizePinValue(input.value);
    if (input.value !== clean) input.value = clean;
    updatePinBoxes(clean);
  });
  input.addEventListener('keydown', function(e) {
    var key = e.key || '';
    if (key === 'Escape') {
      closePinModal();
    }
  });
})();

function verifierPin() {
    var input = document.getElementById('pinInput');
    var pin = sanitizePinValue(input ? input.value : '');
    if (input && input.value !== pin) input.value = pin;
    updatePinBoxes(pin);

    if (pin.length !== 4) {
        if (typeof window.showToast === "function") {
          window.showToast("Entrez un code PIN à 4 chiffres.", "error");
        } else {
          alert("Entrez un code PIN à 4 chiffres.");
        }
        try { focusPinInput(); } catch (e) {}
        return;
    }
    if (pin) {
        // Envoi du code PIN à un script PHP pour vérification
        var xhr = new XMLHttpRequest();
        xhr.onreadystatechange = function() {
            if (this.readyState == 4 && this.status == 200) {
                if (this.responseText == "OK") {
                    window.location.href = "admin.php";
                } else {
                    if (typeof window.showToast === "function") {
                      window.showToast("Code PIN incorrect.", "error");
                    } else {
                      alert("Code PIN incorrect.");
                    }
                    resetPinModal();
                    try { focusPinInput(); } catch (e) {}
                }
            }
        };
        xhr.open("POST", "verifier_pin.php", true);
        xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
        xhr.send("pin=" + pin);
    }
}

// --- Modal actions employé ---
var selectedEmployeId = null;
var selectedIsPointed = 0;
var selectedEmployePrenom = '';

function openEmployeeModal(employeId, isPointed, employePrenom) {
  selectedEmployeId = employeId;
  selectedIsPointed = isPointed ? 1 : 0;
  selectedEmployePrenom = (typeof employePrenom === 'string') ? employePrenom : '';

  var punchBtn = document.getElementById('employeePunchButton');
  punchBtn.textContent = (selectedIsPointed === 1) ? 'Départ' : 'Arrivée';
  if (selectedIsPointed === 1) {
    punchBtn.classList.add('is-depart');
  } else {
    punchBtn.classList.remove('is-depart');
  }

  document.getElementById('employeeModalBackground').style.display = 'block';
  document.getElementById('employeeActionModal').style.display = 'block';
}

function closeEmployeeModal() {
  document.getElementById('employeeActionModal').style.display = 'none';
  document.getElementById('employeeModalBackground').style.display = 'none';
  var punchBtn = document.getElementById('employeePunchButton');
  if (punchBtn) punchBtn.classList.remove('is-depart');
  selectedEmployeId = null;
  selectedIsPointed = 0;
  selectedEmployePrenom = '';
}

function submitEmployeePunch() {
  if (!selectedEmployeId) return;

  var url = (selectedIsPointed === 1) ? 'enregistrement_depart.php' : 'enregistrement_arrivee.php';
  $.ajax({
    type: "POST",
    url: url,
    data: { employe: selectedEmployeId },
    success: function(response) {
      var firstnamePart = selectedEmployePrenom ? (" " + selectedEmployePrenom) : "";
      var msg = (selectedIsPointed === 1)
        ? ("Départ" + firstnamePart + " enregistré")
        : ("Arrivée" + firstnamePart + " enregistrée");
      window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
    },
    error: function() {
      if (typeof window.showToast === "function") {
        window.showToast("Erreur lors de l'enregistrement du pointage.", "error");
      } else {
        alert("Erreur lors de l'enregistrement du pointage.");
      }
    }
  });
}

 function goToEmployeeDetails() {
   if (!selectedEmployeId) return;
   window.location.href = "pointage.php?employe=" + selectedEmployeId + "&view=1";
 }

</script>

</body>

</html>
 </body>
 
 </html>"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 21144,
                "sha1": "e9531fa86e2dee8f602b272137af0763b70bf93b",
                "content_b64": "<?php
/* doc-project | pointages/index.php | Affiche le tableau de pointage des salariés et protège l’accès via autorisation permanente par appareil avec assets locaux versionnés, sans Modernizr/Respond ni Bootstrap JS. | Expose: openEmployeeModal, closeEmployeeModal, submitEmployeePunch, goToEmployeeDetails | Dépend de: pointages/config.php, includes/device_auth.php, includes/asset_version.php, connexion.php, admin.php, enregistrement_arrivee.php, enregistrement_depart.php, pointage.php, js/vendor/jquery-1.11.2.min.js, js/plugins.js, js/main.js, js/driver-modal.js | Impacte: session PHP, cookie d’appareil, redirection d’accès, lecture des pointages/employés, modales UI natives, appels AJAX | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_utilisateurs(UserID,Nom,Prenom,Statut), z_ptg_aqp_pointages(PointageID,UserID,DateHeureEntree,DateHeureSortie) */
// Démarrage ou reprise de la session
session_start();
date_default_timezone_set('Europe/Paris');

require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_once __DIR__ . "/includes/asset_version.php";
require_device_authorized($pdo);

?>


<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">



<title>Pointages Salariés</title>



<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">

<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap-theme.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/fontAwesome.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/hero-slider.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/tooplate-style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/home-buttons.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/driver-modal.css'), ENT_QUOTES, 'UTF-8'); ?>">

</head>

<body class="page-index theme-dark">



<div id="about" class="">

<div class="section-heading">
<h1>Pointages salariés</h1>
<div class="line-dec"></div>
<center>
<br>
<font color="black" size="4" >
<strong>
<?php
// Récupère le dernier pointage de chaque employé pour déterminer s'il est "pointé"
$currentMonth = (int)date('n');
$currentYear = (int)date('Y');
$nowParis = date('Y-m-d H:i:s');

$stmt = $pdo->prepare("
  SELECT
    u.UserID,
    u.Nom,
    u.Prenom,
    p.DateHeureEntree,
    p.DateHeureSortie,
    mh.seconds_worked,
    mh.max_punch_seconds
  FROM z_ptg_aqp_utilisateurs u
  LEFT JOIN z_ptg_aqp_pointages p
    ON p.PointageID = (
      SELECT p2.PointageID
      FROM z_ptg_aqp_pointages p2
      WHERE p2.UserID = u.UserID
      ORDER BY p2.DateHeureEntree DESC, p2.PointageID DESC
      LIMIT 1
    )
  LEFT JOIN (
    SELECT
      UserID,
      SUM(
        TIMESTAMPDIFF(
          SECOND,
          DateHeureEntree,
          COALESCE(DateHeureSortie, :nowParisSum)
        )
      ) AS seconds_worked,
      MAX(
        TIMESTAMPDIFF(
          SECOND,
          DateHeureEntree,
          COALESCE(DateHeureSortie, :nowParisMax)
        )
      ) AS max_punch_seconds
    FROM z_ptg_aqp_pointages
    WHERE MONTH(DateHeureEntree) = :currentMonth
      AND YEAR(DateHeureEntree) = :currentYear
    GROUP BY UserID
  ) mh ON mh.UserID = u.UserID
  WHERE u.Statut = 'actif'
");
$stmt->execute([
  ':nowParisSum' => $nowParis,
  ':nowParisMax' => $nowParis,
  ':currentMonth' => $currentMonth,
  ':currentYear' => $currentYear,
]);
$employesActifs = $stmt->fetchAll(PDO::FETCH_ASSOC);

echo '<table class="employee-grid">';
$compteur = 0;

foreach ($employesActifs as $employe) {
  // Logique similaire pour afficher les employés
  if ($compteur % 2 == 0) {
    echo '<tr>';
  }
  $nomRaw = isset($employe['Nom']) ? (string)$employe['Nom'] : '';
  $prenomRaw = isset($employe['Prenom']) ? (string)$employe['Prenom'] : '';
  $nom = mb_strtoupper($nomRaw, 'UTF-8');
  $prenom = mb_strtoupper($prenomRaw, 'UTF-8');
  $prenomJs = htmlspecialchars(
    json_encode($prenomRaw, JSON_UNESCAPED_UNICODE | JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT),
    ENT_QUOTES,
    'UTF-8'
  );

  // Un employé est "pointé" s'il a un dernier pointage dont DateHeureSortie est NULL
  // (et qu'il existe bien une DateHeureEntree).
  $isPointed = (!empty($employe['DateHeureEntree']) && $employe['DateHeureSortie'] === null);

  // Détection "pointage > 8H" :
  // - via la durée max des pointages du mois (max_punch_seconds),
  // - et via le dernier pointage (en cours ou terminé), même s'il est hors mois (cas rare, mais robuste).
  $maxPunchSeconds = isset($employe['max_punch_seconds']) ? (int)$employe['max_punch_seconds'] : 0;
  $lastPunchSeconds = 0;
  if (!empty($employe['DateHeureEntree'])) {
    $startTs = strtotime($employe['DateHeureEntree']);
    $endTs = ($employe['DateHeureSortie'] === null || $employe['DateHeureSortie'] === '')
      ? strtotime($nowParis)
      : strtotime($employe['DateHeureSortie']);
    if ($startTs !== false && $endTs !== false && $endTs >= $startTs) {
      $lastPunchSeconds = (int)($endTs - $startTs);
    }
  }
  $hasPunchOver8h = ($maxPunchSeconds >= 28800) || ($lastPunchSeconds >= 28800);

  $btnClass = $isPointed ? 'employee-btn status-pointed' : 'employee-btn status-depointed';
  if ($hasPunchOver8h) {
    $btnClass .= ' status-problem';
  }
  // Icône uniquement quand pointé
  $iconHtml = '';
  if ($isPointed) {
    $iconHtml = '<i class="fa fa-check-circle status-icon" aria-hidden="true" title="Pointé"></i>';
  }

  $secondsWorked = isset($employe['seconds_worked']) ? (float)$employe['seconds_worked'] : 0.0;
  $monthHours = $secondsWorked > 0 ? ($secondsWorked / 3600) : 0.0;
  $monthHoursText = number_format($monthHours, 2, '.', '') . 'H';
  $monthHoursHtml = '<span class="month-hours">' . htmlspecialchars($monthHoursText, ENT_QUOTES, 'UTF-8') . '</span>';

  echo '<td class="employee-grid-cell">'
     . '<button type="button" class="' . $btnClass . '" onclick="openEmployeeModal(' . (int)$employe['UserID'] . ', ' . ($isPointed ? '1' : '0') . ', ' . $prenomJs . ')">'
     . $iconHtml
     . '<span class="employee-name">' . htmlspecialchars($nom, ENT_QUOTES, 'UTF-8') . '</span>'
     . '<span class="employee-firstname">' . htmlspecialchars($prenom, ENT_QUOTES, 'UTF-8') . '</span>'
     . $monthHoursHtml
     . '</button>'
     . '</td>';

  if ($compteur % 2 == 1) {
    echo '</tr>';
  }
  $compteur++;
}
 
 echo '</table>';
 ?>
 
 <div class="driver-phone-actions" style="text-align:center; margin:18px 0 6px;">
   <button
     type="button"
     id="driverPhoneButton"
     class="driver-phone-trigger"
     onclick="openDriverPhoneModal()"
   >
     Envoyer SMS livreur
   </button>
 </div>

 </strong>
 </font>
 </center>
 </div>
 </div>


<br><br>

<p><a href="admin.php">Gestion employés</a></p><br>





</div>




<!-- Modal PIN (design refait) -->
<div id="modalBackground" onclick="closePinModal()" aria-hidden="true"></div>
<div id="pinModal" role="dialog" aria-modal="true" aria-labelledby="pinModalTitle">
  <form id="pinForm" onsubmit="verifierPin(); return false;">
    <h2 id="pinModalTitle" class="pin-modal-title">Code PIN</h2>
    <p class="pin-modal-subtitle">Saisissez votre code à 4 chiffres</p>

    <div class="pin-input-wrapper" onclick="focusPinInput()">
      <label for="pinInput" class="sr-only">Code PIN (4 chiffres)</label>
      <div class="pin-boxes" aria-hidden="true">
        <div class="pin-box" data-idx="0"></div>
        <div class="pin-box" data-idx="1"></div>
        <div class="pin-box" data-idx="2"></div>
        <div class="pin-box" data-idx="3"></div>
      </div>
      <input
        type="tel"
        id="pinInput"
        class="pin-hidden-input"
        inputmode="numeric"
        pattern="[0-9]*"
        maxlength="4"
        autocomplete="off"
        autocapitalize="off"
        autocorrect="off"
        spellcheck="false"
        title="Seulement des chiffres."
        value=""
      >
    </div>

    <div class="pin-modal-actions">
      <button type="button" class="pin-modal-btn cancel" onclick="closePinModal()">Annuler</button>
      <button type="submit" class="pin-modal-btn confirm">Valider</button>
    </div>
  </form>
</div>

<!-- Modal Actions Employé (Arrivée/Départ + Voir détail) -->
<div id="employeeModalBackground" onclick="closeEmployeeModal()"></div>
<div id="employeeActionModal" role="dialog" aria-modal="true" aria-labelledby="employeeActionTitle">
  <h2 id="employeeActionTitle">Actions</h2>
  <button type="button" id="employeePunchButton" class="employee-modal-btn punch" onclick="submitEmployeePunch()"></button>
  <button type="button" class="employee-modal-btn details" onclick="goToEmployeeDetails()">Voir détail des heures</button>
</div>

<!-- Modal Livreur (sélection + envoi SMS) -->
<div id="driverModalBackground" class="driver-modal-backdrop" onclick="closeDriverPhoneModal()" aria-hidden="true"></div>
<div id="driverPhoneModal" class="driver-modal" role="dialog" aria-modal="true" aria-labelledby="driverPhoneModalTitle" aria-hidden="true">
  <form id="driverPhoneForm" onsubmit="submitDriverPhone(); return false;">
    <div class="driver-modal-header">
      <div class="driver-modal-header-text">
        <h2 id="driverPhoneModalTitle" class="driver-modal-title">Envoyer SMS livreur</h2>
        <p class="driver-modal-subtitle">Sélectionnez un livreur pour le point de vente de l’appareil</p>
      </div>
      <button type="button" class="driver-modal-close" onclick="closeDriverPhoneModal()" aria-label="Fermer">×</button>
    </div>

    <div class="driver-field">
      <label for="driverSelect" class="driver-label">Livreur</label>
      <select id="driverSelect" class="driver-select" required>
        <option value="">Chargement des livreurs…</option>
      </select>
      <small id="driverSelectedPhoneHelp" class="driver-help">Sélectionnez un livreur pour afficher son numéro.</small>
    </div>

    <div class="driver-field">
      <label for="driverPointVente" class="driver-label">Point de vente</label>
      <select id="driverPointVente" class="driver-select" disabled>
        <option value="lancon">Lançon</option>
        <option value="pelissanne">Pélissanne</option>
      </select>
    </div>

    <div class="driver-modal-actions driver-modal-actions-split">
      <button type="button" class="driver-modal-btn secondary" onclick="openDriverManageModal()">Gérer les livreurs</button>
    </div>

    <div class="driver-modal-actions">
      <button type="button" class="driver-modal-btn cancel" onclick="closeDriverPhoneModal()">Annuler</button>
      <button type="submit" class="driver-modal-btn confirm">Envoyer SMS</button>
    </div>
  </form>
</div>

<!-- Modal gestion livreurs -->
<div id="driverManageModalBackground" class="driver-modal-backdrop" onclick="closeDriverManageModal()" aria-hidden="true"></div>
<div id="driverManageModal" class="driver-modal driver-manage-modal" role="dialog" aria-modal="true" aria-labelledby="driverManageModalTitle" aria-hidden="true">
  <div class="driver-modal-header">
    <div class="driver-modal-header-text">
      <h2 id="driverManageModalTitle" class="driver-modal-title">Gérer les livreurs</h2>
      <p class="driver-modal-subtitle">Ajout, modification et suppression pour le point de vente courant</p>
    </div>
    <button type="button" class="driver-modal-close" onclick="closeDriverManageModal()" aria-label="Fermer">×</button>
  </div>

  <form id="driverManageForm">
    <input type="hidden" id="driverManageMode" value="create">
    <input type="hidden" id="driverEditOriginalPhone" value="">

    <div class="driver-field">
      <label for="driverManageSelect" class="driver-label">Livreur existant</label>
      <select id="driverManageSelect" class="driver-select">
        <option value="">Nouveau livreur</option>
      </select>
    </div>

    <div class="driver-field">
      <label for="driverManageName" class="driver-label">Nom affiché</label>
      <input
        type="text"
        id="driverManageName"
        class="driver-input"
        maxlength="120"
        placeholder="Ex: Karim"
        autocomplete="off"
      >
    </div>

    <div class="driver-field">
      <label for="driverManagePhone" class="driver-label">Téléphone</label>
      <input
        type="tel"
        id="driverManagePhone"
        class="driver-input"
        inputmode="numeric"
        pattern="[0-9]*"
        maxlength="16"
        placeholder="Ex: 0618529375"
        autocomplete="off"
        autocapitalize="off"
        autocorrect="off"
        spellcheck="false"
      >
      <small class="driver-help">Format accepté : 0618529375, 33618529375 ou 0033618529375.</small>
    </div>

    <div class="driver-modal-actions">
      <button type="button" class="driver-modal-btn danger" onclick="deleteSelectedDriver()">Supprimer</button>
      <button type="submit" class="driver-modal-btn confirm">Enregistrer</button>
    </div>
    <div class="driver-modal-actions">
      <button type="button" class="driver-modal-btn cancel" onclick="closeDriverManageModal()">Fermer</button>
    </div>
  </form>
</div>

<script src="<?php echo htmlspecialchars(asset_version_url('js/vendor/jquery-1.11.2.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/plugins.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/main.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/driver-modal.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>



<script type="text/javascript">

$(document).ready(function() {
  
  // navigation click actions 
  
  $('.scroll-link').on('click', function(event){
    
    event.preventDefault();
    
    var sectionID = $(this).attr("data-id");
    
    scrollToID('#' + sectionID, 750);
    
  });
  
  // scroll to top action
  
  $('.scroll-top').on('click', function(event) {
    
    event.preventDefault();
    
    $('html, body').animate({scrollTop:0}, 'slow');         
    
  });
  
  // mobile nav toggle
  
  $('#nav-toggle').on('click', function (event) {
    
    event.preventDefault();
    
    $('#main-nav').toggleClass("open");
    
  });
  
});

// scroll function

function scrollToID(id, speed){
  
  var offSet = 50;
  
  var targetOffset = $(id).offset().top - offSet;
  
  var mainNav = $('#main-nav');
  
  $('html,body').animate({scrollTop:targetOffset}, speed);
  
  if (mainNav.hasClass("open")) {
    
    mainNav.css("height", "1px").removeClass("in").addClass("collapse");
    
    mainNav.removeClass("open");
    
  }
  
}

if (typeof console === "undefined") {
  
  console = {
    
    log: function() { }
    
  };
  
}


function afficherModal() {
    window.location.href = "admin.php";
}


//function afficherModal(pointageId) {
  // Enregistrez l'ID du pointage pour l'utiliser plus tard
  //window.pointageIdPourModification = pointageId;
  // Affichez la fenêtre modale
  //document.getElementById('pinModal').style.display = 'block';
//}

function sanitizePinValue(val) {
  return String(val || '').replace(/\D/g, '').slice(0, 4);
}

function updatePinBoxes(pin) {
  var boxes = document.querySelectorAll('#pinModal .pin-box');
  for (var i = 0; i < boxes.length; i++) {
    boxes[i].textContent = (i < pin.length) ? '★' : '';
  }
}

function focusPinInput() {
  var input = document.getElementById('pinInput');
  if (!input) return;
  try { input.focus(); } catch (e) {}
}

function resetPinModal() {
  var input = document.getElementById('pinInput');
  if (input) input.value = '';
  updatePinBoxes('');
}

function openPinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (!modal || !bg) return;

  modal.style.display = 'block';
  bg.style.display = 'block';
  resetPinModal();

  window.setTimeout(function() {
    focusPinInput();
  }, 10);
}

function closePinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (modal) modal.style.display = 'none';
  if (bg) bg.style.display = 'none';
  resetPinModal();
}

// Bind input once (digits only + sync UI boxes)
(function bindPinOnce() {
  var input = document.getElementById('pinInput');
  if (!input || input.getAttribute('data-bound') === '1') return;
  input.setAttribute('data-bound', '1');
  input.addEventListener('input', function() {
    var clean = sanitizePinValue(input.value);
    if (input.value !== clean) input.value = clean;
    updatePinBoxes(clean);
  });
  input.addEventListener('keydown', function(e) {
    var key = e.key || '';
    if (key === 'Escape') {
      closePinModal();
    }
  });
})();

function verifierPin() {
    var input = document.getElementById('pinInput');
    var pin = sanitizePinValue(input ? input.value : '');
    if (input && input.value !== pin) input.value = pin;
    updatePinBoxes(pin);

    if (pin.length !== 4) {
        if (typeof window.showToast === "function") {
          window.showToast("Entrez un code PIN à 4 chiffres.", "error");
        } else {
          alert("Entrez un code PIN à 4 chiffres.");
        }
        try { focusPinInput(); } catch (e) {}
        return;
    }
    if (pin) {
        // Envoi du code PIN à un script PHP pour vérification
        var xhr = new XMLHttpRequest();
        xhr.onreadystatechange = function() {
            if (this.readyState == 4 && this.status == 200) {
                if (this.responseText == "OK") {
                    window.location.href = "admin.php";
                } else {
                    if (typeof window.showToast === "function") {
                      window.showToast("Code PIN incorrect.", "error");
                    } else {
                      alert("Code PIN incorrect.");
                    }
                    resetPinModal();
                    try { focusPinInput(); } catch (e) {}
                }
            }
        };
        xhr.open("POST", "verifier_pin.php", true);
        xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
        xhr.send("pin=" + pin);
    }
}

// --- Modal actions employé ---
var selectedEmployeId = null;
var selectedIsPointed = 0;
var selectedEmployePrenom = '';

function openEmployeeModal(employeId, isPointed, employePrenom) {
  selectedEmployeId = employeId;
  selectedIsPointed = isPointed ? 1 : 0;
  selectedEmployePrenom = (typeof employePrenom === 'string') ? employePrenom : '';

  var punchBtn = document.getElementById('employeePunchButton');
  punchBtn.textContent = (selectedIsPointed === 1) ? 'Départ' : 'Arrivée';
  if (selectedIsPointed === 1) {
    punchBtn.classList.add('is-depart');
  } else {
    punchBtn.classList.remove('is-depart');
  }

  document.getElementById('employeeModalBackground').style.display = 'block';
  document.getElementById('employeeActionModal').style.display = 'block';
}

function closeEmployeeModal() {
  document.getElementById('employeeActionModal').style.display = 'none';
  document.getElementById('employeeModalBackground').style.display = 'none';
  var punchBtn = document.getElementById('employeePunchButton');
  if (punchBtn) punchBtn.classList.remove('is-depart');
  selectedEmployeId = null;
  selectedIsPointed = 0;
  selectedEmployePrenom = '';
}

function submitEmployeePunch() {
  if (!selectedEmployeId) return;

  var url = (selectedIsPointed === 1) ? 'enregistrement_depart.php' : 'enregistrement_arrivee.php';
  $.ajax({
    type: "POST",
    url: url,
    data: { employe: selectedEmployeId },
    success: function(response) {
      var firstnamePart = selectedEmployePrenom ? (" " + selectedEmployePrenom) : "";
      var msg = (selectedIsPointed === 1)
        ? ("Départ" + firstnamePart + " enregistré")
        : ("Arrivée" + firstnamePart + " enregistrée");
      window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
    },
    error: function() {
      if (typeof window.showToast === "function") {
        window.showToast("Erreur lors de l'enregistrement du pointage.", "error");
      } else {
        alert("Erreur lors de l'enregistrement du pointage.");
      }
    }
  });
}

 function goToEmployeeDetails() {
   if (!selectedEmployeId) return;
   window.location.href = "pointage.php?employe=" + selectedEmployeId + "&view=1";
 }

</script>

</body>

</html>"
            }
        },
        {
            "path": "pointages/js/driver-modal.js",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 18432,
                "sha1": "9d96ad21c47e667794579464e00ea7c6cd887760",
                "content_b64": "/* -------------------------------------------------------------------------- */
/* doc-project | pointages/js/driver-modal.js | Gère l’ouverture des modales livreur, le chargement JSON par point de vente, l’envoi SMS et l’administration des livreurs. | Expose: openDriverPhoneModal, closeDriverPhoneModal, submitDriverPhone, openDriverManageModal, closeDriverManageModal, saveDriverManagerForm, deleteSelectedDriver | Dépend de: jQuery, api/livreurs.php, enregistrer_livreur.php, window.appDevicePointVente, window.openDevicePosChooser, window.showToast | Impacte: UI modales, navigation vers index.php avec toast, état de sélection des livreurs | Tables: aucune */
/* Modales Livreur : envoi SMS + gestion JSON par point de vente              */
/* Dépendances: jQuery (déjà présent sur la page)                             */
/* -------------------------------------------------------------------------- */
(function(window, document) {
  'use strict';

  var BG_ID = 'driverModalBackground';
  var MODAL_ID = 'driverPhoneModal';
  var FORM_ID = 'driverPhoneForm';
  var DRIVER_SELECT_ID = 'driverSelect';
  var DRIVER_SELECTED_PHONE_HELP_ID = 'driverSelectedPhoneHelp';
  var PV_ID = 'driverPointVente';
  var MANAGE_BG_ID = 'driverManageModalBackground';
  var MANAGE_MODAL_ID = 'driverManageModal';
  var MANAGE_FORM_ID = 'driverManageForm';
  var MANAGE_SELECT_ID = 'driverManageSelect';
  var MANAGE_MODE_ID = 'driverManageMode';
  var MANAGE_ORIGINAL_PHONE_ID = 'driverEditOriginalPhone';
  var MANAGE_NAME_ID = 'driverManageName';
  var MANAGE_PHONE_ID = 'driverManagePhone';
  var MANAGE_TRIGGER_SELECTOR = '.driver-modal-btn.secondary';
  var API_URL = 'api/livreurs.php';
  var driverCache = {
    lancon: [],
    pelissanne: []
  };

  function $(id) {
    return document.getElementById(id);
  }

  function safeToast(msg, type) {
    if (typeof window.showToast === 'function') {
      window.showToast(msg, type || 'info');
      return;
    }
    window.alert(msg);
  }

  function sanitizeDigits(val) {
    return String(val || '').replace(/\D/g, '');
  }

  function normalizeDriverPhone(raw) {
    var v = String(raw || '').trim();
    if (!v) return '';

    var digits = sanitizeDigits(v);

    // 0033XXXXXXXXX => 33XXXXXXXXX
    if (digits.indexOf('0033') === 0) {
      digits = '33' + digits.slice(4);
    }
    // 0XXXXXXXXX (10 chiffres) => 33XXXXXXXXX
    if (digits.length === 10 && digits.charAt(0) === '0') {
      digits = '33' + digits.slice(1);
    }
    // Validation minimale FR: 33 + 9 chiffres
    if (!/^33\d{9}$/.test(digits)) return '';
    return digits;
  }

  function getPointVente() {
    var pv = $(PV_ID);
    var devicePosApi = window.appDevicePointVente || null;
    var pointVente = pv ? String(pv.value || '').toLowerCase() : '';

    if (!pointVente && devicePosApi && typeof devicePosApi.get === 'function') {
      pointVente = devicePosApi.get() || '';
    }

    if (pointVente !== 'lancon' && pointVente !== 'pelissanne') {
      return '';
    }

    return pointVente;
  }

  function getPointVenteLabel(pointVente) {
    return pointVente === 'pelissanne' ? 'Pélissanne' : 'Lançon';
  }

  function getDriversForPointVente(pointVente) {
    if (pointVente !== 'lancon' && pointVente !== 'pelissanne') return [];
    return Array.isArray(driverCache[pointVente]) ? driverCache[pointVente] : [];
  }

  function findDriverByPhone(pointVente, phone) {
    var list = getDriversForPointVente(pointVente);
    var i;
    for (i = 0; i < list.length; i++) {
      if (String(list[i].telephone || '') === String(phone || '')) {
        return list[i];
      }
    }
    return null;
  }

  function updateSelectedDriverHelp() {
    var pointVente = getPointVente();
    var select = $(DRIVER_SELECT_ID);
    var help = $(DRIVER_SELECTED_PHONE_HELP_ID);
    var driver;
    if (!help || !select) return;

    driver = findDriverByPhone(pointVente, select.value);
    if (!driver) {
      help.textContent = 'Sélectionnez un livreur pour afficher son numéro.';
      return;
    }

    help.textContent = 'Numéro sélectionné : +' + String(driver.telephone || '');
  }

  function renderDriverSelect(pointVente) {
    var select = $(DRIVER_SELECT_ID);
    var list = getDriversForPointVente(pointVente);
    var i;
    var option;

    if (!select) return;
    select.innerHTML = '';

    option = document.createElement('option');
    option.value = '';
    option.textContent = list.length ? 'Choisir un livreur' : 'Aucun livreur enregistré';
    select.appendChild(option);

    for (i = 0; i < list.length; i++) {
      option = document.createElement('option');
      option.value = list[i].telephone;
      option.textContent = list[i].nom + ' - +' + list[i].telephone;
      select.appendChild(option);
    }

    updateSelectedDriverHelp();
  }

  function renderDriverManagerSelect(pointVente) {
    var select = $(MANAGE_SELECT_ID);
    var list = getDriversForPointVente(pointVente);
    var i;
    var option;

    if (!select) return;
    select.innerHTML = '';

    option = document.createElement('option');
    option.value = '';
    option.textContent = 'Nouveau livreur';
    select.appendChild(option);

    for (i = 0; i < list.length; i++) {
      option = document.createElement('option');
      option.value = list[i].telephone;
      option.textContent = list[i].nom + ' - +' + list[i].telephone;
      select.appendChild(option);
    }
  }

  function resetDriverManagerForm() {
    var mode = $(MANAGE_MODE_ID);
    var originalPhone = $(MANAGE_ORIGINAL_PHONE_ID);
    var nameInput = $(MANAGE_NAME_ID);
    var phoneInput = $(MANAGE_PHONE_ID);
    var select = $(MANAGE_SELECT_ID);

    if (mode) mode.value = 'create';
    if (originalPhone) originalPhone.value = '';
    if (nameInput) nameInput.value = '';
    if (phoneInput) phoneInput.value = '';
    if (select) select.value = '';
  }

  function fillDriverManagerForm(pointVente, phone) {
    var driver = findDriverByPhone(pointVente, phone);
    var mode = $(MANAGE_MODE_ID);
    var originalPhone = $(MANAGE_ORIGINAL_PHONE_ID);
    var nameInput = $(MANAGE_NAME_ID);
    var phoneInput = $(MANAGE_PHONE_ID);

    if (!driver) {
      resetDriverManagerForm();
      return;
    }

    if (mode) mode.value = 'update';
    if (originalPhone) originalPhone.value = String(driver.telephone || '');
    if (nameInput) nameInput.value = String(driver.nom || '');
    if (phoneInput) phoneInput.value = '0' + String(driver.telephone || '').replace(/^33/, '');
  }

  function fetchDrivers(pointVente, onDone) {
    if (!window.jQuery || !window.jQuery.ajax) {
      safeToast('Erreur: jQuery indisponible.', 'error');
      return;
    }

    window.jQuery.ajax({
      type: 'GET',
      url: API_URL,
      dataType: 'json',
      cache: false,
      data: { point_vente: pointVente },
      success: function(resp) {
        if (!resp || !resp.ok || !resp.data) {
          safeToast((resp && resp.error) ? resp.error : 'Impossible de charger les livreurs.', 'error');
          return;
        }

        driverCache[pointVente] = Array.isArray(resp.data.drivers) ? resp.data.drivers : [];
        renderDriverSelect(pointVente);
        renderDriverManagerSelect(pointVente);

        if (typeof onDone === 'function') {
          onDone(driverCache[pointVente]);
        }
      },
      error: function() {
        safeToast('Erreur réseau lors du chargement des livreurs.', 'error');
      }
    });
  }

  function isManageOpen() {
    var modal = $(MANAGE_MODAL_ID);
    return !!(modal && modal.style.display === 'block');
  }

  function ensureManageModalMountedAtBody() {
    var bg = $(MANAGE_BG_ID);
    var modal = $(MANAGE_MODAL_ID);

    if (bg && bg.parentNode !== document.body) {
      document.body.appendChild(bg);
    }
    if (modal && modal.parentNode !== document.body) {
      document.body.appendChild(modal);
    }
  }

  function stopEventPropagation(e) {
    if (e) e.stopPropagation();
  }

  function isOpen() {
    var modal = $(MODAL_ID);
    return !!(modal && modal.style.display === 'block');
  }

  function openModal() {
    var bg = $(BG_ID);
    var modal = $(MODAL_ID);
    var devicePosApi = window.appDevicePointVente || null;
    var savedPointVente = '';
    var pv = $(PV_ID);
    if (!bg || !modal) return;

    if (devicePosApi && typeof devicePosApi.get === 'function') {
      savedPointVente = devicePosApi.get();
    }
    if (!savedPointVente) {
      safeToast('Veuillez d’abord définir le point de vente par défaut de l’appareil.', 'error');
      if (typeof window.openDevicePosChooser === 'function') {
        window.openDevicePosChooser();
      }
      return;
    }
    if (pv) pv.value = savedPointVente;

    bg.style.display = 'block';
    modal.style.display = 'block';
    modal.classList.add('is-visible');
    modal.setAttribute('aria-hidden', 'false');
    bg.setAttribute('aria-hidden', 'false');
    document.body.classList.add('modal-open-driver');

    fetchDrivers(savedPointVente, function(drivers) {
      if (!drivers.length) {
        safeToast('Aucun livreur enregistré pour ' + getPointVenteLabel(savedPointVente) + '.', 'info');
      }
      window.setTimeout(function() {
        try { if ($(DRIVER_SELECT_ID)) $(DRIVER_SELECT_ID).focus(); } catch (e) {}
      }, 10);
    });
  }

  function closeModal() {
    var bg = $(BG_ID);
    var modal = $(MODAL_ID);

    if (isManageOpen()) {
      closeManageModal();
    }
    if (modal) {
      modal.style.display = 'none';
      modal.classList.remove('is-visible');
      modal.setAttribute('aria-hidden', 'true');
    }
    if (bg) {
      bg.style.display = 'none';
      bg.setAttribute('aria-hidden', 'true');
    }
    document.body.classList.remove('modal-open-driver');
  }

  function submitModal() {
    var pointVente = getPointVente();
    var select = $(DRIVER_SELECT_ID);
    var phone = select ? String(select.value || '') : '';

    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }

    if (!phone) {
      safeToast('Veuillez sélectionner un livreur.', 'error');
      try { if (select) select.focus(); } catch (e) {}
      return;
    }

    if (!window.jQuery || !window.jQuery.ajax) {
      safeToast('Erreur: jQuery indisponible pour l’envoi.', 'error');
      return;
    }

    window.jQuery.ajax({
      type: 'POST',
      url: 'enregistrer_livreur.php',
      dataType: 'json',
      data: { phone: phone, point_vente: pointVente },
      success: function(resp) {
        if (resp && resp.ok) {
          closeModal();
          var pvLabel = (pointVente === 'pelissanne') ? 'Pélissanne' : 'Lançon';
          var driver = findDriverByPhone(pointVente, phone);
          var driverName = driver ? driver.nom : phone;
          var msg = 'SMS en attente : ' + driverName + ' (+' + phone + ', ' + pvLabel + ')';
          window.location.href = 'index.php?toast=' + encodeURIComponent(msg) + '&toastType=success';
          return;
        }
        var err = (resp && resp.error) ? resp.error : "Erreur lors de l'enregistrement du livreur.";
        safeToast(err, 'error');
      },
      error: function() {
        safeToast("Erreur réseau lors de l'enregistrement du livreur.", 'error');
      }
    });
  }

  function openManageModal() {
    var bg = $(MANAGE_BG_ID);
    var modal = $(MANAGE_MODAL_ID);
    var pointVente = getPointVente();

    if (!bg || !modal) return;

    ensureManageModalMountedAtBody();
    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }

    renderDriverManagerSelect(pointVente);
    resetDriverManagerForm();

    bg.style.display = 'block';
    modal.style.display = 'block';
    modal.classList.add('is-visible');
    modal.setAttribute('aria-hidden', 'false');
    bg.setAttribute('aria-hidden', 'false');
    document.body.classList.add('driver-manage-modal-open');

    window.setTimeout(function() {
      try { if ($(MANAGE_SELECT_ID)) $(MANAGE_SELECT_ID).focus(); } catch (e) {}
    }, 10);
  }

  function closeManageModal() {
    var bg = $(MANAGE_BG_ID);
    var modal = $(MANAGE_MODAL_ID);
    if (modal) {
      modal.style.display = 'none';
      modal.classList.remove('is-visible');
      modal.setAttribute('aria-hidden', 'true');
    }
    if (bg) {
      bg.style.display = 'none';
      bg.setAttribute('aria-hidden', 'true');
    }
    document.body.classList.remove('driver-manage-modal-open');
    resetDriverManagerForm();

    window.setTimeout(function() {
      var trigger;
      if (!isOpen()) return;
      trigger = document.querySelector(MANAGE_TRIGGER_SELECTOR);
      try {
        if (trigger) trigger.focus();
      } catch (e) {}
    }, 10);
  }

  function saveDriverManagerForm() {
    var pointVente = getPointVente();
    var mode = $(MANAGE_MODE_ID);
    var originalPhone = $(MANAGE_ORIGINAL_PHONE_ID);
    var nameInput = $(MANAGE_NAME_ID);
    var phoneInput = $(MANAGE_PHONE_ID);
    var name = nameInput ? String(nameInput.value || '').trim() : '';
    var phone = normalizeDriverPhone(phoneInput ? phoneInput.value : '');
    var payload;

    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }
    if (!name) {
      safeToast('Le nom du livreur est obligatoire.', 'error');
      try { if (nameInput) nameInput.focus(); } catch (e) {}
      return;
    }
    if (!phone) {
      safeToast('Numéro invalide. Exemple: 0618529375.', 'error');
      try { if (phoneInput) phoneInput.focus(); } catch (e) {}
      return;
    }

    payload = {
      action: (mode && mode.value === 'update') ? 'update' : 'create',
      point_vente: pointVente,
      nom: name,
      telephone: phone,
      original_phone: originalPhone ? String(originalPhone.value || '') : ''
    };

    window.jQuery.ajax({
      type: 'POST',
      url: API_URL,
      dataType: 'json',
      data: payload,
      success: function(resp) {
        if (!resp || !resp.ok) {
          safeToast((resp && resp.error) ? resp.error : 'Impossible d’enregistrer le livreur.', 'error');
          return;
        }
        safeToast('Livreur enregistré.', 'success');
        fetchDrivers(pointVente, function() {
          renderDriverSelect(pointVente);
          renderDriverManagerSelect(pointVente);
          resetDriverManagerForm();
        });
      },
      error: function(xhr) {
        var resp = xhr && xhr.responseJSON ? xhr.responseJSON : null;
        var msg = (resp && resp.error) ? resp.error : 'Erreur réseau lors de l’enregistrement.';
        safeToast(msg, 'error');
      }
    });
  }

  function deleteDriver() {
    var pointVente = getPointVente();
    var select = $(MANAGE_SELECT_ID);
    var phone = select ? String(select.value || '') : '';

    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }
    if (!phone) {
      safeToast('Sélectionnez un livreur à supprimer.', 'error');
      return;
    }

    window.jQuery.ajax({
      type: 'POST',
      url: API_URL,
      dataType: 'json',
      data: {
        action: 'delete',
        point_vente: pointVente,
        telephone: phone
      },
      success: function(resp) {
        if (!resp || !resp.ok) {
          safeToast((resp && resp.error) ? resp.error : 'Impossible de supprimer le livreur.', 'error');
          return;
        }
        safeToast('Livreur supprimé.', 'success');
        fetchDrivers(pointVente, function() {
          renderDriverSelect(pointVente);
          renderDriverManagerSelect(pointVente);
          resetDriverManagerForm();
        });
      },
      error: function() {
        safeToast('Erreur réseau lors de la suppression.', 'error');
      }
    });
  }

  function bindOnce() {
    var form = $(FORM_ID);
    var modal = $(MODAL_ID);
    var bg = $(BG_ID);
    var select = $(DRIVER_SELECT_ID);
    var manageForm = $(MANAGE_FORM_ID);
    var manageSelect = $(MANAGE_SELECT_ID);
    var managePhone = $(MANAGE_PHONE_ID);
    var manageBg = $(MANAGE_BG_ID);
    if (!form || !modal || !bg || !select || !manageForm || !manageSelect || !manageBg) return;
    ensureManageModalMountedAtBody();

    if (modal.getAttribute('data-bound') === '1') return;
    modal.setAttribute('data-bound', '1');

    if (managePhone) {
      managePhone.addEventListener('input', function() {
        var clean = sanitizeDigits(managePhone.value).slice(0, 16);
        if (managePhone.value !== clean) managePhone.value = clean;
      });
    }

    select.addEventListener('change', function() {
      updateSelectedDriverHelp();
    });

    manageSelect.addEventListener('change', function() {
      fillDriverManagerForm(getPointVente(), manageSelect.value);
    });

    // ESC close
    document.addEventListener('keydown', function(e) {
      var key = e.key || '';
      if (key !== 'Escape') return;
      if (isManageOpen()) {
        closeManageModal();
        return;
      }
      if (isOpen()) closeModal();
    });

    bg.addEventListener('click', function() {
      if (!isOpen()) return;
      closeModal();
    });
    manageBg.addEventListener('click', function() {
      closeManageModal();
    });

    modal.addEventListener('click', stopEventPropagation);
    manageBg.addEventListener('click', function() {
      closeManageModal();
    });
    $(MANAGE_MODAL_ID).addEventListener('click', stopEventPropagation);
    form.addEventListener('click', stopEventPropagation);
    manageForm.addEventListener('click', stopEventPropagation);
    form.addEventListener('submit', function(e) {
      e.preventDefault();
      submitModal();
      return false;
    });
    manageForm.addEventListener('submit', function(e) {
      e.preventDefault();
      saveDriverManagerForm();
      return false;
    });
  }

  window.openDriverPhoneModal = function() { openModal(); };
  window.closeDriverPhoneModal = function() { closeModal(); };
  window.submitDriverPhone = function() { submitModal(); };
  window.openDriverManageModal = function() { openManageModal(); };
  window.closeDriverManageModal = function() { closeManageModal(); };
  window.saveDriverManagerForm = function() { saveDriverManagerForm(); };
  window.deleteSelectedDriver = function() { deleteDriver(); };

  if (document.readyState === 'loading') {
    document.addEventListener('DOMContentLoaded', bindOnce);
  } else {
    bindOnce();
  }

})(window, document);
"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 18464,
                "sha1": "93fe5e17ad4d741bcea9a4fcf2b46fe3da602f6b",
                "content_b64": "/* -------------------------------------------------------------------------- */
/* doc-project | pointages/js/driver-modal.js | Gère l’ouverture des modales livreur natives, le chargement JSON par point de vente, l’envoi SMS et l’administration des livreurs sans Bootstrap JS. | Expose: openDriverPhoneModal, closeDriverPhoneModal, submitDriverPhone, openDriverManageModal, closeDriverManageModal, saveDriverManagerForm, deleteSelectedDriver | Dépend de: jQuery local, api/livreurs.php, enregistrer_livreur.php, window.appDevicePointVente, window.openDevicePosChooser, window.showToast | Impacte: UI modales, navigation vers index.php avec toast, état de sélection des livreurs | Tables: aucune */
/* Modales Livreur : envoi SMS + gestion JSON par point de vente              */
/* Dépendances: jQuery (déjà présent sur la page)                             */
/* -------------------------------------------------------------------------- */
(function(window, document) {
  'use strict';

  var BG_ID = 'driverModalBackground';
  var MODAL_ID = 'driverPhoneModal';
  var FORM_ID = 'driverPhoneForm';
  var DRIVER_SELECT_ID = 'driverSelect';
  var DRIVER_SELECTED_PHONE_HELP_ID = 'driverSelectedPhoneHelp';
  var PV_ID = 'driverPointVente';
  var MANAGE_BG_ID = 'driverManageModalBackground';
  var MANAGE_MODAL_ID = 'driverManageModal';
  var MANAGE_FORM_ID = 'driverManageForm';
  var MANAGE_SELECT_ID = 'driverManageSelect';
  var MANAGE_MODE_ID = 'driverManageMode';
  var MANAGE_ORIGINAL_PHONE_ID = 'driverEditOriginalPhone';
  var MANAGE_NAME_ID = 'driverManageName';
  var MANAGE_PHONE_ID = 'driverManagePhone';
  var MANAGE_TRIGGER_SELECTOR = '.driver-modal-btn.secondary';
  var API_URL = 'api/livreurs.php';
  var driverCache = {
    lancon: [],
    pelissanne: []
  };

  function $(id) {
    return document.getElementById(id);
  }

  function safeToast(msg, type) {
    if (typeof window.showToast === 'function') {
      window.showToast(msg, type || 'info');
      return;
    }
    window.alert(msg);
  }

  function sanitizeDigits(val) {
    return String(val || '').replace(/\D/g, '');
  }

  function normalizeDriverPhone(raw) {
    var v = String(raw || '').trim();
    if (!v) return '';

    var digits = sanitizeDigits(v);

    // 0033XXXXXXXXX => 33XXXXXXXXX
    if (digits.indexOf('0033') === 0) {
      digits = '33' + digits.slice(4);
    }
    // 0XXXXXXXXX (10 chiffres) => 33XXXXXXXXX
    if (digits.length === 10 && digits.charAt(0) === '0') {
      digits = '33' + digits.slice(1);
    }
    // Validation minimale FR: 33 + 9 chiffres
    if (!/^33\d{9}$/.test(digits)) return '';
    return digits;
  }

  function getPointVente() {
    var pv = $(PV_ID);
    var devicePosApi = window.appDevicePointVente || null;
    var pointVente = pv ? String(pv.value || '').toLowerCase() : '';

    if (!pointVente && devicePosApi && typeof devicePosApi.get === 'function') {
      pointVente = devicePosApi.get() || '';
    }

    if (pointVente !== 'lancon' && pointVente !== 'pelissanne') {
      return '';
    }

    return pointVente;
  }

  function getPointVenteLabel(pointVente) {
    return pointVente === 'pelissanne' ? 'Pélissanne' : 'Lançon';
  }

  function getDriversForPointVente(pointVente) {
    if (pointVente !== 'lancon' && pointVente !== 'pelissanne') return [];
    return Array.isArray(driverCache[pointVente]) ? driverCache[pointVente] : [];
  }

  function findDriverByPhone(pointVente, phone) {
    var list = getDriversForPointVente(pointVente);
    var i;
    for (i = 0; i < list.length; i++) {
      if (String(list[i].telephone || '') === String(phone || '')) {
        return list[i];
      }
    }
    return null;
  }

  function updateSelectedDriverHelp() {
    var pointVente = getPointVente();
    var select = $(DRIVER_SELECT_ID);
    var help = $(DRIVER_SELECTED_PHONE_HELP_ID);
    var driver;
    if (!help || !select) return;

    driver = findDriverByPhone(pointVente, select.value);
    if (!driver) {
      help.textContent = 'Sélectionnez un livreur pour afficher son numéro.';
      return;
    }

    help.textContent = 'Numéro sélectionné : +' + String(driver.telephone || '');
  }

  function renderDriverSelect(pointVente) {
    var select = $(DRIVER_SELECT_ID);
    var list = getDriversForPointVente(pointVente);
    var i;
    var option;

    if (!select) return;
    select.innerHTML = '';

    option = document.createElement('option');
    option.value = '';
    option.textContent = list.length ? 'Choisir un livreur' : 'Aucun livreur enregistré';
    select.appendChild(option);

    for (i = 0; i < list.length; i++) {
      option = document.createElement('option');
      option.value = list[i].telephone;
      option.textContent = list[i].nom + ' - +' + list[i].telephone;
      select.appendChild(option);
    }

    updateSelectedDriverHelp();
  }

  function renderDriverManagerSelect(pointVente) {
    var select = $(MANAGE_SELECT_ID);
    var list = getDriversForPointVente(pointVente);
    var i;
    var option;

    if (!select) return;
    select.innerHTML = '';

    option = document.createElement('option');
    option.value = '';
    option.textContent = 'Nouveau livreur';
    select.appendChild(option);

    for (i = 0; i < list.length; i++) {
      option = document.createElement('option');
      option.value = list[i].telephone;
      option.textContent = list[i].nom + ' - +' + list[i].telephone;
      select.appendChild(option);
    }
  }

  function resetDriverManagerForm() {
    var mode = $(MANAGE_MODE_ID);
    var originalPhone = $(MANAGE_ORIGINAL_PHONE_ID);
    var nameInput = $(MANAGE_NAME_ID);
    var phoneInput = $(MANAGE_PHONE_ID);
    var select = $(MANAGE_SELECT_ID);

    if (mode) mode.value = 'create';
    if (originalPhone) originalPhone.value = '';
    if (nameInput) nameInput.value = '';
    if (phoneInput) phoneInput.value = '';
    if (select) select.value = '';
  }

  function fillDriverManagerForm(pointVente, phone) {
    var driver = findDriverByPhone(pointVente, phone);
    var mode = $(MANAGE_MODE_ID);
    var originalPhone = $(MANAGE_ORIGINAL_PHONE_ID);
    var nameInput = $(MANAGE_NAME_ID);
    var phoneInput = $(MANAGE_PHONE_ID);

    if (!driver) {
      resetDriverManagerForm();
      return;
    }

    if (mode) mode.value = 'update';
    if (originalPhone) originalPhone.value = String(driver.telephone || '');
    if (nameInput) nameInput.value = String(driver.nom || '');
    if (phoneInput) phoneInput.value = '0' + String(driver.telephone || '').replace(/^33/, '');
  }

  function fetchDrivers(pointVente, onDone) {
    if (!window.jQuery || !window.jQuery.ajax) {
      safeToast('Erreur: jQuery indisponible.', 'error');
      return;
    }

    window.jQuery.ajax({
      type: 'GET',
      url: API_URL,
      dataType: 'json',
      cache: false,
      data: { point_vente: pointVente },
      success: function(resp) {
        if (!resp || !resp.ok || !resp.data) {
          safeToast((resp && resp.error) ? resp.error : 'Impossible de charger les livreurs.', 'error');
          return;
        }

        driverCache[pointVente] = Array.isArray(resp.data.drivers) ? resp.data.drivers : [];
        renderDriverSelect(pointVente);
        renderDriverManagerSelect(pointVente);

        if (typeof onDone === 'function') {
          onDone(driverCache[pointVente]);
        }
      },
      error: function() {
        safeToast('Erreur réseau lors du chargement des livreurs.', 'error');
      }
    });
  }

  function isManageOpen() {
    var modal = $(MANAGE_MODAL_ID);
    return !!(modal && modal.style.display === 'block');
  }

  function ensureManageModalMountedAtBody() {
    var bg = $(MANAGE_BG_ID);
    var modal = $(MANAGE_MODAL_ID);

    if (bg && bg.parentNode !== document.body) {
      document.body.appendChild(bg);
    }
    if (modal && modal.parentNode !== document.body) {
      document.body.appendChild(modal);
    }
  }

  function stopEventPropagation(e) {
    if (e) e.stopPropagation();
  }

  function isOpen() {
    var modal = $(MODAL_ID);
    return !!(modal && modal.style.display === 'block');
  }

  function openModal() {
    var bg = $(BG_ID);
    var modal = $(MODAL_ID);
    var devicePosApi = window.appDevicePointVente || null;
    var savedPointVente = '';
    var pv = $(PV_ID);
    if (!bg || !modal) return;

    if (devicePosApi && typeof devicePosApi.get === 'function') {
      savedPointVente = devicePosApi.get();
    }
    if (!savedPointVente) {
      safeToast('Veuillez d’abord définir le point de vente par défaut de l’appareil.', 'error');
      if (typeof window.openDevicePosChooser === 'function') {
        window.openDevicePosChooser();
      }
      return;
    }
    if (pv) pv.value = savedPointVente;

    bg.style.display = 'block';
    modal.style.display = 'block';
    modal.classList.add('is-visible');
    modal.setAttribute('aria-hidden', 'false');
    bg.setAttribute('aria-hidden', 'false');
    document.body.classList.add('modal-open-driver');

    fetchDrivers(savedPointVente, function(drivers) {
      if (!drivers.length) {
        safeToast('Aucun livreur enregistré pour ' + getPointVenteLabel(savedPointVente) + '.', 'info');
      }
      window.setTimeout(function() {
        try { if ($(DRIVER_SELECT_ID)) $(DRIVER_SELECT_ID).focus(); } catch (e) {}
      }, 10);
    });
  }

  function closeModal() {
    var bg = $(BG_ID);
    var modal = $(MODAL_ID);

    if (isManageOpen()) {
      closeManageModal();
    }
    if (modal) {
      modal.style.display = 'none';
      modal.classList.remove('is-visible');
      modal.setAttribute('aria-hidden', 'true');
    }
    if (bg) {
      bg.style.display = 'none';
      bg.setAttribute('aria-hidden', 'true');
    }
    document.body.classList.remove('modal-open-driver');
  }

  function submitModal() {
    var pointVente = getPointVente();
    var select = $(DRIVER_SELECT_ID);
    var phone = select ? String(select.value || '') : '';

    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }

    if (!phone) {
      safeToast('Veuillez sélectionner un livreur.', 'error');
      try { if (select) select.focus(); } catch (e) {}
      return;
    }

    if (!window.jQuery || !window.jQuery.ajax) {
      safeToast('Erreur: jQuery indisponible pour l’envoi.', 'error');
      return;
    }

    window.jQuery.ajax({
      type: 'POST',
      url: 'enregistrer_livreur.php',
      dataType: 'json',
      data: { phone: phone, point_vente: pointVente },
      success: function(resp) {
        if (resp && resp.ok) {
          closeModal();
          var pvLabel = (pointVente === 'pelissanne') ? 'Pélissanne' : 'Lançon';
          var driver = findDriverByPhone(pointVente, phone);
          var driverName = driver ? driver.nom : phone;
          var msg = 'SMS en attente : ' + driverName + ' (+' + phone + ', ' + pvLabel + ')';
          window.location.href = 'index.php?toast=' + encodeURIComponent(msg) + '&toastType=success';
          return;
        }
        var err = (resp && resp.error) ? resp.error : "Erreur lors de l'enregistrement du livreur.";
        safeToast(err, 'error');
      },
      error: function() {
        safeToast("Erreur réseau lors de l'enregistrement du livreur.", 'error');
      }
    });
  }

  function openManageModal() {
    var bg = $(MANAGE_BG_ID);
    var modal = $(MANAGE_MODAL_ID);
    var pointVente = getPointVente();

    if (!bg || !modal) return;

    ensureManageModalMountedAtBody();
    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }

    renderDriverManagerSelect(pointVente);
    resetDriverManagerForm();

    bg.style.display = 'block';
    modal.style.display = 'block';
    modal.classList.add('is-visible');
    modal.setAttribute('aria-hidden', 'false');
    bg.setAttribute('aria-hidden', 'false');
    document.body.classList.add('driver-manage-modal-open');

    window.setTimeout(function() {
      try { if ($(MANAGE_SELECT_ID)) $(MANAGE_SELECT_ID).focus(); } catch (e) {}
    }, 10);
  }

  function closeManageModal() {
    var bg = $(MANAGE_BG_ID);
    var modal = $(MANAGE_MODAL_ID);
    if (modal) {
      modal.style.display = 'none';
      modal.classList.remove('is-visible');
      modal.setAttribute('aria-hidden', 'true');
    }
    if (bg) {
      bg.style.display = 'none';
      bg.setAttribute('aria-hidden', 'true');
    }
    document.body.classList.remove('driver-manage-modal-open');
    resetDriverManagerForm();

    window.setTimeout(function() {
      var trigger;
      if (!isOpen()) return;
      trigger = document.querySelector(MANAGE_TRIGGER_SELECTOR);
      try {
        if (trigger) trigger.focus();
      } catch (e) {}
    }, 10);
  }

  function saveDriverManagerForm() {
    var pointVente = getPointVente();
    var mode = $(MANAGE_MODE_ID);
    var originalPhone = $(MANAGE_ORIGINAL_PHONE_ID);
    var nameInput = $(MANAGE_NAME_ID);
    var phoneInput = $(MANAGE_PHONE_ID);
    var name = nameInput ? String(nameInput.value || '').trim() : '';
    var phone = normalizeDriverPhone(phoneInput ? phoneInput.value : '');
    var payload;

    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }
    if (!name) {
      safeToast('Le nom du livreur est obligatoire.', 'error');
      try { if (nameInput) nameInput.focus(); } catch (e) {}
      return;
    }
    if (!phone) {
      safeToast('Numéro invalide. Exemple: 0618529375.', 'error');
      try { if (phoneInput) phoneInput.focus(); } catch (e) {}
      return;
    }

    payload = {
      action: (mode && mode.value === 'update') ? 'update' : 'create',
      point_vente: pointVente,
      nom: name,
      telephone: phone,
      original_phone: originalPhone ? String(originalPhone.value || '') : ''
    };

    window.jQuery.ajax({
      type: 'POST',
      url: API_URL,
      dataType: 'json',
      data: payload,
      success: function(resp) {
        if (!resp || !resp.ok) {
          safeToast((resp && resp.error) ? resp.error : 'Impossible d’enregistrer le livreur.', 'error');
          return;
        }
        safeToast('Livreur enregistré.', 'success');
        fetchDrivers(pointVente, function() {
          renderDriverSelect(pointVente);
          renderDriverManagerSelect(pointVente);
          resetDriverManagerForm();
        });
      },
      error: function(xhr) {
        var resp = xhr && xhr.responseJSON ? xhr.responseJSON : null;
        var msg = (resp && resp.error) ? resp.error : 'Erreur réseau lors de l’enregistrement.';
        safeToast(msg, 'error');
      }
    });
  }

  function deleteDriver() {
    var pointVente = getPointVente();
    var select = $(MANAGE_SELECT_ID);
    var phone = select ? String(select.value || '') : '';

    if (!pointVente) {
      safeToast('Point de vente invalide.', 'error');
      return;
    }
    if (!phone) {
      safeToast('Sélectionnez un livreur à supprimer.', 'error');
      return;
    }

    window.jQuery.ajax({
      type: 'POST',
      url: API_URL,
      dataType: 'json',
      data: {
        action: 'delete',
        point_vente: pointVente,
        telephone: phone
      },
      success: function(resp) {
        if (!resp || !resp.ok) {
          safeToast((resp && resp.error) ? resp.error : 'Impossible de supprimer le livreur.', 'error');
          return;
        }
        safeToast('Livreur supprimé.', 'success');
        fetchDrivers(pointVente, function() {
          renderDriverSelect(pointVente);
          renderDriverManagerSelect(pointVente);
          resetDriverManagerForm();
        });
      },
      error: function() {
        safeToast('Erreur réseau lors de la suppression.', 'error');
      }
    });
  }

  function bindOnce() {
    var form = $(FORM_ID);
    var modal = $(MODAL_ID);
    var bg = $(BG_ID);
    var select = $(DRIVER_SELECT_ID);
    var manageForm = $(MANAGE_FORM_ID);
    var manageSelect = $(MANAGE_SELECT_ID);
    var managePhone = $(MANAGE_PHONE_ID);
    var manageBg = $(MANAGE_BG_ID);
    if (!form || !modal || !bg || !select || !manageForm || !manageSelect || !manageBg) return;
    ensureManageModalMountedAtBody();

    if (modal.getAttribute('data-bound') === '1') return;
    modal.setAttribute('data-bound', '1');

    if (managePhone) {
      managePhone.addEventListener('input', function() {
        var clean = sanitizeDigits(managePhone.value).slice(0, 16);
        if (managePhone.value !== clean) managePhone.value = clean;
      });
    }

    select.addEventListener('change', function() {
      updateSelectedDriverHelp();
    });

    manageSelect.addEventListener('change', function() {
      fillDriverManagerForm(getPointVente(), manageSelect.value);
    });

    // ESC close
    document.addEventListener('keydown', function(e) {
      var key = e.key || '';
      if (key !== 'Escape') return;
      if (isManageOpen()) {
        closeManageModal();
        return;
      }
      if (isOpen()) closeModal();
    });

    bg.addEventListener('click', function() {
      if (!isOpen()) return;
      closeModal();
    });
    manageBg.addEventListener('click', function() {
      closeManageModal();
    });

    modal.addEventListener('click', stopEventPropagation);
    manageBg.addEventListener('click', function() {
      closeManageModal();
    });
    $(MANAGE_MODAL_ID).addEventListener('click', stopEventPropagation);
    form.addEventListener('click', stopEventPropagation);
    manageForm.addEventListener('click', stopEventPropagation);
    form.addEventListener('submit', function(e) {
      e.preventDefault();
      submitModal();
      return false;
    });
    manageForm.addEventListener('submit', function(e) {
      e.preventDefault();
      saveDriverManagerForm();
      return false;
    });
  }

  window.openDriverPhoneModal = function() { openModal(); };
  window.closeDriverPhoneModal = function() { closeModal(); };
  window.submitDriverPhone = function() { submitModal(); };
  window.openDriverManageModal = function() { openManageModal(); };
  window.closeDriverManageModal = function() { closeManageModal(); };
  window.saveDriverManagerForm = function() { saveDriverManagerForm(); };
  window.deleteSelectedDriver = function() { deleteDriver(); };

  if (document.readyState === 'loading') {
    document.addEventListener('DOMContentLoaded', bindOnce);
  } else {
    bindOnce();
  }

})(window, document);
"
            }
        },
        {
            "path": "pointages/js/main.js",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 20813,
                "sha1": "b79d2d6c61cafbc8a267fa352e40bfc09cbcb0a6",
                "content_b64": "/* doc-project | pointages/js/main.js | Gère les interactions UI globales, les toasts, les modales de confirmation et le choix du point de vente par défaut. | Expose: showToast, consumeToastFromUrl, showConfirmModal, appDevicePointVente, openDevicePosChooser | Dépend de: jQuery, window.localStorage, window.history, CustomEvent, document.body.page-index | Impacte: UI, état local de l’app, navigation via query params, événements DOM | Tables: aucune */
/* 

Template 2089 Meteor

http://www.tooplate.com/view/2089-meteor

*/

jQuery(document).ready(function($) {

	'use strict';


        $('.counter').each(function() {
          var $this = $(this),
              countTo = $this.attr('data-count');
          
          $({ countNum: $this.text()}).animate({
            countNum: countTo
          },

          {

            duration: 8000,
            easing:'linear',
            step: function() {
              $this.text(Math.floor(this.countNum));
            },
            complete: function() {
              $this.text(this.countNum);
              //alert('finished');
            }

          });  
          
        });



        $(".b1").click(function () {
            $(".pop").fadeIn(300);
            
        });
		
		$(".b2").click(function () {
            $(".pop2").fadeIn(300);
            
        });
		
		$(".b3").click(function () {
            $(".pop3").fadeIn(300);
            
        });

        $(".pop > span, .pop").click(function () {
            $(".pop").fadeOut(300);
        });
		
		$(".pop2 > span, .pop2").click(function () {
            $(".pop2").fadeOut(300);
        });
		
		$(".pop3 > span, .pop3").click(function () {
            $(".pop3").fadeOut(300);
        });


        $(window).on("scroll", function() {
            if($(window).scrollTop() > 100) {
                $(".header").addClass("active");
            } else {
                //remove the background property so it comes transparent again (defined in your css)
               $(".header").removeClass("active");
            }
        });


	/************** Mixitup (Filter Projects) *********************/
    	$('.projects-holder').mixitup({
            effects: ['fade','grayscale'],
            easing: 'snap',
            transitionSpeed: 400
        });

});

/* -------------------------------------------------------------------------- */
/* App UI helpers: Toasts + Modale de confirmation (sans Bootstrap Toast)      */
/* -------------------------------------------------------------------------- */
(function(window, document, $) {
  'use strict';

  var TOAST_CONTAINER_ID = 'appToastContainer';
  var CONFIRM_BACKDROP_ID = 'appConfirmBackdrop';
  var CONFIRM_MODAL_ID = 'appConfirmModal';
  var DEVICE_POS_STORAGE_KEY = 'defaultPointVenteDevice';
  var DEVICE_POS_MODAL_ID = 'devicePosModal';
  var DEVICE_POS_BACKDROP_ID = 'devicePosBackdrop';
  var DEVICE_POS_ALLOWED_VALUES = ['lancon', 'pelissanne'];

  function ensureToastContainer() {
    var container = document.getElementById(TOAST_CONTAINER_ID);
    if (container) return container;

    container = document.createElement('div');
    container.id = TOAST_CONTAINER_ID;
    container.className = 'app-toast-container';
    document.body.appendChild(container);
    return container;
  }

  function safeString(val) {
    if (val === null || typeof val === 'undefined') return '';
    return String(val);
  }

  function normalizeToastType(type) {
    var t = (type || 'info').toLowerCase();
    if (t !== 'success' && t !== 'error' && t !== 'info') t = 'info';
    return t;
  }

  function showToastInternal(message, type, options) {
    var msg = safeString(message).trim();
    if (!msg) return;

    var toastType = normalizeToastType(type);
    var opts = options || {};
    var duration = typeof opts.duration === 'number' ? opts.duration : 3200;
    var centeredInfoMessage = 'Demander au FOU de réactiver';

    var container = ensureToastContainer();
    var toast = document.createElement('div');
    toast.className = 'app-toast ' + toastType;
    if (msg === centeredInfoMessage) {
      toast.className += ' app-toast-centered-message';
    }

    var text = document.createElement('div');
    text.className = 'app-toast-message';
    text.textContent = msg;

    toast.appendChild(text);
    container.appendChild(toast);

    var closed = false;
    var timeoutId = null;

    function closeToast() {
      if (closed) return;
      closed = true;
      if (timeoutId) window.clearTimeout(timeoutId);
      toast.className = toast.className.replace(/\bis-visible\b/g, '').trim();
      window.setTimeout(function() {
        if (toast && toast.parentNode) toast.parentNode.removeChild(toast);
      }, 220);
    }

    // Optionnel: tap/clic sur le toast pour le fermer, sans bouton dédié
    toast.addEventListener('click', function(e) {
      e.preventDefault();
      closeToast();
    });

    window.setTimeout(function() {
      toast.className += ' is-visible';
    }, 10);

    if (duration > 0) {
      timeoutId = window.setTimeout(function() {
        closeToast();
      }, duration);
    }
  }

  window.showToast = function(message, type, options) {
    showToastInternal(message, type, options);
  };

  function normalizePointVente(value) {
    var normalized = safeString(value).trim().toLowerCase();
    for (var i = 0; i < DEVICE_POS_ALLOWED_VALUES.length; i++) {
      if (normalized === DEVICE_POS_ALLOWED_VALUES[i]) return normalized;
    }
    return '';
  }

  function getDefaultPointVenteLabel(value) {
    return value === 'pelissanne' ? 'Pélissanne' : 'Lançon';
  }

  function getDefaultPointVente() {
    try {
      return normalizePointVente(window.localStorage.getItem(DEVICE_POS_STORAGE_KEY));
    } catch (e) {
      return '';
    }
  }

  function dispatchDefaultPointVenteChanged(value) {
    try {
      document.dispatchEvent(new CustomEvent('default-point-vente:changed', {
        detail: { value: value || '' }
      }));
    } catch (e) {
      var evt;
      if (document.createEvent) {
        evt = document.createEvent('Event');
        evt.initEvent('default-point-vente:changed', true, true);
        evt.detail = { value: value || '' };
        document.dispatchEvent(evt);
      }
    }
  }

  function setDefaultPointVente(value) {
    var normalized = normalizePointVente(value);
    if (!normalized) return false;
    try {
      window.localStorage.setItem(DEVICE_POS_STORAGE_KEY, normalized);
      dispatchDefaultPointVenteChanged(normalized);
      return true;
    } catch (e) {
      return false;
    }
  }

  function clearDefaultPointVente() {
    var hadValue = false;
    try {
      hadValue = !!window.localStorage.getItem(DEVICE_POS_STORAGE_KEY);
      window.localStorage.removeItem(DEVICE_POS_STORAGE_KEY);
    } catch (e) {}
    if (hadValue) {
      dispatchDefaultPointVenteChanged('');
    }
  }

  function clearAppLocalData() {
    try {
      clearDefaultPointVente();
      return true;
    } catch (e) {
      return false;
    }
  }

  function hasAppLocalData() {
    try {
      return !!window.localStorage.getItem(DEVICE_POS_STORAGE_KEY);
    } catch (e) {
      return false;
    }
  }

  window.appDevicePointVente = {
    storageKey: DEVICE_POS_STORAGE_KEY,
    get: getDefaultPointVente,
    set: setDefaultPointVente,
    clear: clearDefaultPointVente,
    normalize: normalizePointVente,
    label: getDefaultPointVenteLabel,
    clearAll: clearAppLocalData,
    hasStoredData: hasAppLocalData
  };

  function getQueryParam(name) {
    var query = window.location.search || '';
    var re = new RegExp('[?&]' + name + '=([^&]*)', 'i');
    var match = re.exec(query);
    if (!match || !match[1]) return null;
    try {
      return decodeURIComponent(match[1].replace(/\+/g, ' '));
    } catch (e) {
      return match[1];
    }
  }

  function removeQueryParams(paramNames) {
    if (!window.history || !window.history.replaceState) return;
    var href = window.location.href;
    var hashIndex = href.indexOf('#');
    var hash = '';
    if (hashIndex >= 0) {
      hash = href.substring(hashIndex);
      href = href.substring(0, hashIndex);
    }

    var qIndex = href.indexOf('?');
    if (qIndex < 0) return;

    var base = href.substring(0, qIndex);
    var query = href.substring(qIndex + 1);
    var parts = query.split('&');
    var kept = [];
    for (var i = 0; i < parts.length; i++) {
      var p = parts[i];
      if (!p) continue;
      var key = p.split('=')[0];
      var shouldRemove = false;
      for (var j = 0; j < paramNames.length; j++) {
        if (key === paramNames[j]) {
          shouldRemove = true;
          break;
        }
      }
      if (!shouldRemove) kept.push(p);
    }
    var newUrl = base + (kept.length ? '?' + kept.join('&') : '') + hash;
    window.history.replaceState({}, document.title, newUrl);
  }

  window.consumeToastFromUrl = function() {
    var toastMsg = getQueryParam('toast');
    if (!toastMsg) return;
    var toastType = getQueryParam('toastType') || 'info';
    showToastInternal(toastMsg, toastType, { duration: 3600 });
    removeQueryParams(['toast', 'toastType']);
  };

  var confirmState = {
    isOpen: false,
    callback: null
  };

  var confirmEls = null;

  function closeConfirm(result) {
    if (!confirmEls) return;
    confirmState.isOpen = false;
    confirmEls.backdrop.style.display = 'none';
    confirmEls.modal.style.display = 'none';

    var cb = confirmState.callback;
    confirmState.callback = null;
    if (typeof cb === 'function') cb(!!result);
  }

  function ensureConfirmModal() {
    if (confirmEls) return confirmEls;

    var backdrop = document.getElementById(CONFIRM_BACKDROP_ID);
    if (!backdrop) {
      backdrop = document.createElement('div');
      backdrop.id = CONFIRM_BACKDROP_ID;
      backdrop.className = 'app-modal-backdrop';
      document.body.appendChild(backdrop);
    }

    var modal = document.getElementById(CONFIRM_MODAL_ID);
    var titleEl, messageEl, cancelBtn, confirmBtn;

    if (!modal) {
      modal = document.createElement('div');
      modal.id = CONFIRM_MODAL_ID;
      modal.className = 'app-modal';
      modal.setAttribute('role', 'dialog');
      modal.setAttribute('aria-modal', 'true');

      titleEl = document.createElement('h3');
      titleEl.id = 'appConfirmTitle';
      titleEl.textContent = 'Confirmation';

      messageEl = document.createElement('p');
      messageEl.id = 'appConfirmMessage';
      messageEl.textContent = '';

      var actions = document.createElement('div');
      actions.className = 'app-modal-actions';

      cancelBtn = document.createElement('button');
      cancelBtn.type = 'button';
      cancelBtn.className = 'app-modal-btn cancel';
      cancelBtn.id = 'appConfirmCancel';
      cancelBtn.textContent = 'Annuler';

      confirmBtn = document.createElement('button');
      confirmBtn.type = 'button';
      confirmBtn.className = 'app-modal-btn confirm';
      confirmBtn.id = 'appConfirmOk';
      confirmBtn.textContent = 'OK';

      actions.appendChild(cancelBtn);
      actions.appendChild(confirmBtn);

      modal.appendChild(titleEl);
      modal.appendChild(messageEl);
      modal.appendChild(actions);
      document.body.appendChild(modal);
    } else {
      titleEl = document.getElementById('appConfirmTitle');
      messageEl = document.getElementById('appConfirmMessage');
      cancelBtn = document.getElementById('appConfirmCancel');
      confirmBtn = document.getElementById('appConfirmOk');
    }

    confirmEls = {
      backdrop: backdrop,
      modal: modal,
      titleEl: titleEl,
      messageEl: messageEl,
      cancelBtn: cancelBtn,
      confirmBtn: confirmBtn
    };

    if (modal.getAttribute('data-bound') !== '1') {
      modal.setAttribute('data-bound', '1');

      backdrop.addEventListener('click', function() {
        if (!confirmState.isOpen) return;
        closeConfirm(false);
      });
      cancelBtn.addEventListener('click', function(e) {
        e.preventDefault();
        closeConfirm(false);
      });
      confirmBtn.addEventListener('click', function(e) {
        e.preventDefault();
        closeConfirm(true);
      });
      document.addEventListener('keydown', function(e) {
        if (!confirmState.isOpen) return;
        var key = e.keyCode || e.which;
        if (key === 27) {
          closeConfirm(false);
        }
      });
    }

    return confirmEls;
  }

  window.showConfirmModal = function(options, callback) {
    var opts = options || {};
    var cb = callback;

    if (typeof cb !== 'function' && typeof window.Promise !== 'undefined') {
      return new window.Promise(function(resolve) {
        window.showConfirmModal(opts, function(result) { resolve(!!result); });
      });
    }

    var els = ensureConfirmModal();
    confirmState.isOpen = true;
    confirmState.callback = typeof cb === 'function' ? cb : null;

    els.titleEl.textContent = safeString(opts.title || 'Confirmation');
    els.messageEl.textContent = safeString(opts.message || '');
    els.confirmBtn.textContent = safeString(opts.confirmText || 'OK');
    els.cancelBtn.textContent = safeString(opts.cancelText || 'Annuler');

    els.backdrop.style.display = 'block';
    els.modal.style.display = 'block';

    window.setTimeout(function() {
      try { els.confirmBtn.focus(); } catch (e) {}
    }, 10);
  };

  var devicePosEls = null;
  var devicePosState = {
    isOpen: false
  };

  function isIndexPage() {
    return !!(document.body && document.body.classList.contains('page-index'));
  }

  function ensureDevicePosModal() {
    if (devicePosEls) return devicePosEls;

    var backdrop = document.getElementById(DEVICE_POS_BACKDROP_ID);
    if (!backdrop) {
      backdrop = document.createElement('div');
      backdrop.id = DEVICE_POS_BACKDROP_ID;
      backdrop.className = 'device-pos-backdrop';
      backdrop.setAttribute('aria-hidden', 'true');
      document.body.appendChild(backdrop);
    }

    var modal = document.getElementById(DEVICE_POS_MODAL_ID);
    if (!modal) {
      modal = document.createElement('div');
      modal.id = DEVICE_POS_MODAL_ID;
      modal.className = 'device-pos-modal';
      modal.setAttribute('role', 'dialog');
      modal.setAttribute('aria-modal', 'true');
      modal.setAttribute('aria-labelledby', 'devicePosTitle');
      modal.setAttribute('aria-describedby', 'devicePosMessage');
      modal.setAttribute('aria-hidden', 'true');
      modal.innerHTML = ''
        + '<h3 id="devicePosTitle">Point de vente par défaut</h3>'
        + '<p id="devicePosMessage">Sélectionnez le point de vente par défaut de cet appareil avant de continuer.</p>'
        + '<form id="devicePosForm">'
        + '  <label class="device-pos-label" for="devicePosSelect">Point de vente</label>'
        + '  <select id="devicePosSelect" class="device-pos-select" required>'
        + '    <option value="">Choisir un point de vente</option>'
        + '    <option value="lancon">Lançon</option>'
        + '    <option value="pelissanne">Pélissanne</option>'
        + '  </select>'
        + '  <div class="device-pos-actions">'
        + '    <button type="submit" class="device-pos-btn confirm">Enregistrer</button>'
        + '  </div>'
        + '</form>';
      document.body.appendChild(modal);
    }

    devicePosEls = {
      backdrop: backdrop,
      modal: modal,
      form: document.getElementById('devicePosForm'),
      select: document.getElementById('devicePosSelect')
    };

    if (modal.getAttribute('data-bound') !== '1') {
      modal.setAttribute('data-bound', '1');

      if (devicePosEls.form) {
        devicePosEls.form.addEventListener('submit', function(e) {
          var savedValue;
          e.preventDefault();
          savedValue = normalizePointVente(devicePosEls.select ? devicePosEls.select.value : '');
          if (!savedValue) {
            showToastInternal('Veuillez sélectionner un point de vente.', 'error', { duration: 3200 });
            return false;
          }

          if (!setDefaultPointVente(savedValue)) {
            showToastInternal('Impossible d’enregistrer le point de vente sur cet appareil.', 'error', { duration: 3600 });
            return false;
          }

          hideDevicePosModal();
          showToastInternal('Point de vente par défaut enregistré : ' + getDefaultPointVenteLabel(savedValue) + '.', 'success', { duration: 2600 });
          return false;
        });
      }
    }

    return devicePosEls;
  }

  function showDevicePosModal(forceOpen) {
    var els = ensureDevicePosModal();
    var currentValue = getDefaultPointVente();

    if (!forceOpen && currentValue) return;

    devicePosState.isOpen = true;
    if (els.select) {
      els.select.value = currentValue || '';
    }

    els.backdrop.style.display = 'block';
    els.modal.style.display = 'block';
    els.backdrop.setAttribute('aria-hidden', 'false');
    els.modal.setAttribute('aria-hidden', 'false');
    document.body.classList.add('device-pos-modal-open');

    window.setTimeout(function() {
      try { if (els.select) els.select.focus(); } catch (e) {}
    }, 10);
  }

  function hideDevicePosModal() {
    var els = ensureDevicePosModal();
    devicePosState.isOpen = false;
    els.backdrop.style.display = 'none';
    els.modal.style.display = 'none';
    els.backdrop.setAttribute('aria-hidden', 'true');
    els.modal.setAttribute('aria-hidden', 'true');
    document.body.classList.remove('device-pos-modal-open');
  }

  function openDevicePosChooser(forceOpen) {
    showDevicePosModal(!!forceOpen);
  }

  function ensureDevicePosChangeShortcut() {
    var existingBtn;
    var existingResetBtn;
    var heading;
    var btn;
    var resetBtn;
    var disabledResetMessage = 'Demander au FOU de réactiver';

    if (!isIndexPage()) return;

    existingBtn = document.getElementById('devicePosQuickChange');
    existingResetBtn = document.getElementById('deviceLocalDataReset');
    if (existingBtn && existingResetBtn) return;

    heading = document.querySelector('.section-heading');
    if (!heading || !heading.parentNode) return;

    btn = document.createElement('button');
    btn.type = 'button';
    btn.id = 'devicePosQuickChange';
    btn.className = 'device-pos-quick-change';
    btn.textContent = 'PDV appareil : ' + (getDefaultPointVente() ? getDefaultPointVenteLabel(getDefaultPointVente()) : 'non défini');
    btn.addEventListener('click', function(e) {
      e.preventDefault();
      openDevicePosChooser(true);
    });
    heading.parentNode.insertBefore(btn, heading.nextSibling);

    resetBtn = document.createElement('button');
    resetBtn.type = 'button';
    resetBtn.id = 'deviceLocalDataReset';
    resetBtn.className = 'device-pos-reset is-disabled';
    resetBtn.textContent = 'Supprimer les données locales';
    resetBtn.setAttribute('aria-disabled', 'true');
    resetBtn.setAttribute('title', disabledResetMessage);
    resetBtn.addEventListener('click', function(e) {
      e.preventDefault();
      e.stopPropagation();
      showToastInternal(disabledResetMessage, 'info', { duration: 2800 });
    });
    resetBtn.addEventListener('keydown', function(e) {
      var key = e.key || '';
      if (key !== 'Enter' && key !== ' ') return;

      e.preventDefault();
      e.stopPropagation();
      showToastInternal(disabledResetMessage, 'info', { duration: 2800 });
    });

    btn.insertAdjacentElement('afterend', resetBtn);
  }

  function refreshDevicePosQuickChangeLabel() {
    var btn = document.getElementById('devicePosQuickChange');
    var currentValue = getDefaultPointVente();
    if (!btn) return;
    btn.textContent = 'PDV appareil : ' + (currentValue ? getDefaultPointVenteLabel(currentValue) : 'non défini');
  }

  window.openDevicePosChooser = function() {
    openDevicePosChooser(true);
  };

  $(function() {
    if (typeof window.consumeToastFromUrl === 'function') {
      window.consumeToastFromUrl();
    }

    if (window.__flashToast && window.__flashToast.message) {
      showToastInternal(window.__flashToast.message, window.__flashToast.type || 'info', { duration: window.__flashToast.duration || 3600 });
      if (window.__flashToast.redirect) {
        window.setTimeout(function() {
          window.location.href = window.__flashToast.redirect;
        }, 900);
      }
      try { delete window.__flashToast; } catch (e) {}
    }

    if (isIndexPage()) {
      ensureDevicePosModal();
      ensureDevicePosChangeShortcut();
      refreshDevicePosQuickChangeLabel();

      if (!getDefaultPointVente()) {
        showDevicePosModal(false);
      }

      document.addEventListener('default-point-vente:changed', function() {
        refreshDevicePosQuickChangeLabel();
      });
    }
  });

})(window, document, jQuery);"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 20845,
                "sha1": "850d8541b6cecbb823ee39441fe27788a6b58c6c",
                "content_b64": "/* doc-project | pointages/js/main.js | Gère les interactions UI globales, les toasts, les modales natives de confirmation et le choix du point de vente par défaut sans Bootstrap JS. | Expose: showToast, consumeToastFromUrl, showConfirmModal, appDevicePointVente, openDevicePosChooser | Dépend de: jQuery local, window.localStorage, window.history, CustomEvent, document.body.page-index | Impacte: UI, état local de l’app, navigation via query params, événements DOM | Tables: aucune */
/* 

Template 2089 Meteor

http://www.tooplate.com/view/2089-meteor

*/

jQuery(document).ready(function($) {

	'use strict';


        $('.counter').each(function() {
          var $this = $(this),
              countTo = $this.attr('data-count');
          
          $({ countNum: $this.text()}).animate({
            countNum: countTo
          },

          {

            duration: 8000,
            easing:'linear',
            step: function() {
              $this.text(Math.floor(this.countNum));
            },
            complete: function() {
              $this.text(this.countNum);
              //alert('finished');
            }

          });  
          
        });



        $(".b1").click(function () {
            $(".pop").fadeIn(300);
            
        });
		
		$(".b2").click(function () {
            $(".pop2").fadeIn(300);
            
        });
		
		$(".b3").click(function () {
            $(".pop3").fadeIn(300);
            
        });

        $(".pop > span, .pop").click(function () {
            $(".pop").fadeOut(300);
        });
		
		$(".pop2 > span, .pop2").click(function () {
            $(".pop2").fadeOut(300);
        });
		
		$(".pop3 > span, .pop3").click(function () {
            $(".pop3").fadeOut(300);
        });


        $(window).on("scroll", function() {
            if($(window).scrollTop() > 100) {
                $(".header").addClass("active");
            } else {
                //remove the background property so it comes transparent again (defined in your css)
               $(".header").removeClass("active");
            }
        });


	/************** Mixitup (Filter Projects) *********************/
    	$('.projects-holder').mixitup({
            effects: ['fade','grayscale'],
            easing: 'snap',
            transitionSpeed: 400
        });

});

/* -------------------------------------------------------------------------- */
/* App UI helpers: Toasts + Modale de confirmation (sans Bootstrap Toast)      */
/* -------------------------------------------------------------------------- */
(function(window, document, $) {
  'use strict';

  var TOAST_CONTAINER_ID = 'appToastContainer';
  var CONFIRM_BACKDROP_ID = 'appConfirmBackdrop';
  var CONFIRM_MODAL_ID = 'appConfirmModal';
  var DEVICE_POS_STORAGE_KEY = 'defaultPointVenteDevice';
  var DEVICE_POS_MODAL_ID = 'devicePosModal';
  var DEVICE_POS_BACKDROP_ID = 'devicePosBackdrop';
  var DEVICE_POS_ALLOWED_VALUES = ['lancon', 'pelissanne'];

  function ensureToastContainer() {
    var container = document.getElementById(TOAST_CONTAINER_ID);
    if (container) return container;

    container = document.createElement('div');
    container.id = TOAST_CONTAINER_ID;
    container.className = 'app-toast-container';
    document.body.appendChild(container);
    return container;
  }

  function safeString(val) {
    if (val === null || typeof val === 'undefined') return '';
    return String(val);
  }

  function normalizeToastType(type) {
    var t = (type || 'info').toLowerCase();
    if (t !== 'success' && t !== 'error' && t !== 'info') t = 'info';
    return t;
  }

  function showToastInternal(message, type, options) {
    var msg = safeString(message).trim();
    if (!msg) return;

    var toastType = normalizeToastType(type);
    var opts = options || {};
    var duration = typeof opts.duration === 'number' ? opts.duration : 3200;
    var centeredInfoMessage = 'Demander au FOU de réactiver';

    var container = ensureToastContainer();
    var toast = document.createElement('div');
    toast.className = 'app-toast ' + toastType;
    if (msg === centeredInfoMessage) {
      toast.className += ' app-toast-centered-message';
    }

    var text = document.createElement('div');
    text.className = 'app-toast-message';
    text.textContent = msg;

    toast.appendChild(text);
    container.appendChild(toast);

    var closed = false;
    var timeoutId = null;

    function closeToast() {
      if (closed) return;
      closed = true;
      if (timeoutId) window.clearTimeout(timeoutId);
      toast.className = toast.className.replace(/\bis-visible\b/g, '').trim();
      window.setTimeout(function() {
        if (toast && toast.parentNode) toast.parentNode.removeChild(toast);
      }, 220);
    }

    // Optionnel: tap/clic sur le toast pour le fermer, sans bouton dédié
    toast.addEventListener('click', function(e) {
      e.preventDefault();
      closeToast();
    });

    window.setTimeout(function() {
      toast.className += ' is-visible';
    }, 10);

    if (duration > 0) {
      timeoutId = window.setTimeout(function() {
        closeToast();
      }, duration);
    }
  }

  window.showToast = function(message, type, options) {
    showToastInternal(message, type, options);
  };

  function normalizePointVente(value) {
    var normalized = safeString(value).trim().toLowerCase();
    for (var i = 0; i < DEVICE_POS_ALLOWED_VALUES.length; i++) {
      if (normalized === DEVICE_POS_ALLOWED_VALUES[i]) return normalized;
    }
    return '';
  }

  function getDefaultPointVenteLabel(value) {
    return value === 'pelissanne' ? 'Pélissanne' : 'Lançon';
  }

  function getDefaultPointVente() {
    try {
      return normalizePointVente(window.localStorage.getItem(DEVICE_POS_STORAGE_KEY));
    } catch (e) {
      return '';
    }
  }

  function dispatchDefaultPointVenteChanged(value) {
    try {
      document.dispatchEvent(new CustomEvent('default-point-vente:changed', {
        detail: { value: value || '' }
      }));
    } catch (e) {
      var evt;
      if (document.createEvent) {
        evt = document.createEvent('Event');
        evt.initEvent('default-point-vente:changed', true, true);
        evt.detail = { value: value || '' };
        document.dispatchEvent(evt);
      }
    }
  }

  function setDefaultPointVente(value) {
    var normalized = normalizePointVente(value);
    if (!normalized) return false;
    try {
      window.localStorage.setItem(DEVICE_POS_STORAGE_KEY, normalized);
      dispatchDefaultPointVenteChanged(normalized);
      return true;
    } catch (e) {
      return false;
    }
  }

  function clearDefaultPointVente() {
    var hadValue = false;
    try {
      hadValue = !!window.localStorage.getItem(DEVICE_POS_STORAGE_KEY);
      window.localStorage.removeItem(DEVICE_POS_STORAGE_KEY);
    } catch (e) {}
    if (hadValue) {
      dispatchDefaultPointVenteChanged('');
    }
  }

  function clearAppLocalData() {
    try {
      clearDefaultPointVente();
      return true;
    } catch (e) {
      return false;
    }
  }

  function hasAppLocalData() {
    try {
      return !!window.localStorage.getItem(DEVICE_POS_STORAGE_KEY);
    } catch (e) {
      return false;
    }
  }

  window.appDevicePointVente = {
    storageKey: DEVICE_POS_STORAGE_KEY,
    get: getDefaultPointVente,
    set: setDefaultPointVente,
    clear: clearDefaultPointVente,
    normalize: normalizePointVente,
    label: getDefaultPointVenteLabel,
    clearAll: clearAppLocalData,
    hasStoredData: hasAppLocalData
  };

  function getQueryParam(name) {
    var query = window.location.search || '';
    var re = new RegExp('[?&]' + name + '=([^&]*)', 'i');
    var match = re.exec(query);
    if (!match || !match[1]) return null;
    try {
      return decodeURIComponent(match[1].replace(/\+/g, ' '));
    } catch (e) {
      return match[1];
    }
  }

  function removeQueryParams(paramNames) {
    if (!window.history || !window.history.replaceState) return;
    var href = window.location.href;
    var hashIndex = href.indexOf('#');
    var hash = '';
    if (hashIndex >= 0) {
      hash = href.substring(hashIndex);
      href = href.substring(0, hashIndex);
    }

    var qIndex = href.indexOf('?');
    if (qIndex < 0) return;

    var base = href.substring(0, qIndex);
    var query = href.substring(qIndex + 1);
    var parts = query.split('&');
    var kept = [];
    for (var i = 0; i < parts.length; i++) {
      var p = parts[i];
      if (!p) continue;
      var key = p.split('=')[0];
      var shouldRemove = false;
      for (var j = 0; j < paramNames.length; j++) {
        if (key === paramNames[j]) {
          shouldRemove = true;
          break;
        }
      }
      if (!shouldRemove) kept.push(p);
    }
    var newUrl = base + (kept.length ? '?' + kept.join('&') : '') + hash;
    window.history.replaceState({}, document.title, newUrl);
  }

  window.consumeToastFromUrl = function() {
    var toastMsg = getQueryParam('toast');
    if (!toastMsg) return;
    var toastType = getQueryParam('toastType') || 'info';
    showToastInternal(toastMsg, toastType, { duration: 3600 });
    removeQueryParams(['toast', 'toastType']);
  };

  var confirmState = {
    isOpen: false,
    callback: null
  };

  var confirmEls = null;

  function closeConfirm(result) {
    if (!confirmEls) return;
    confirmState.isOpen = false;
    confirmEls.backdrop.style.display = 'none';
    confirmEls.modal.style.display = 'none';

    var cb = confirmState.callback;
    confirmState.callback = null;
    if (typeof cb === 'function') cb(!!result);
  }

  function ensureConfirmModal() {
    if (confirmEls) return confirmEls;

    var backdrop = document.getElementById(CONFIRM_BACKDROP_ID);
    if (!backdrop) {
      backdrop = document.createElement('div');
      backdrop.id = CONFIRM_BACKDROP_ID;
      backdrop.className = 'app-modal-backdrop';
      document.body.appendChild(backdrop);
    }

    var modal = document.getElementById(CONFIRM_MODAL_ID);
    var titleEl, messageEl, cancelBtn, confirmBtn;

    if (!modal) {
      modal = document.createElement('div');
      modal.id = CONFIRM_MODAL_ID;
      modal.className = 'app-modal';
      modal.setAttribute('role', 'dialog');
      modal.setAttribute('aria-modal', 'true');

      titleEl = document.createElement('h3');
      titleEl.id = 'appConfirmTitle';
      titleEl.textContent = 'Confirmation';

      messageEl = document.createElement('p');
      messageEl.id = 'appConfirmMessage';
      messageEl.textContent = '';

      var actions = document.createElement('div');
      actions.className = 'app-modal-actions';

      cancelBtn = document.createElement('button');
      cancelBtn.type = 'button';
      cancelBtn.className = 'app-modal-btn cancel';
      cancelBtn.id = 'appConfirmCancel';
      cancelBtn.textContent = 'Annuler';

      confirmBtn = document.createElement('button');
      confirmBtn.type = 'button';
      confirmBtn.className = 'app-modal-btn confirm';
      confirmBtn.id = 'appConfirmOk';
      confirmBtn.textContent = 'OK';

      actions.appendChild(cancelBtn);
      actions.appendChild(confirmBtn);

      modal.appendChild(titleEl);
      modal.appendChild(messageEl);
      modal.appendChild(actions);
      document.body.appendChild(modal);
    } else {
      titleEl = document.getElementById('appConfirmTitle');
      messageEl = document.getElementById('appConfirmMessage');
      cancelBtn = document.getElementById('appConfirmCancel');
      confirmBtn = document.getElementById('appConfirmOk');
    }

    confirmEls = {
      backdrop: backdrop,
      modal: modal,
      titleEl: titleEl,
      messageEl: messageEl,
      cancelBtn: cancelBtn,
      confirmBtn: confirmBtn
    };

    if (modal.getAttribute('data-bound') !== '1') {
      modal.setAttribute('data-bound', '1');

      backdrop.addEventListener('click', function() {
        if (!confirmState.isOpen) return;
        closeConfirm(false);
      });
      cancelBtn.addEventListener('click', function(e) {
        e.preventDefault();
        closeConfirm(false);
      });
      confirmBtn.addEventListener('click', function(e) {
        e.preventDefault();
        closeConfirm(true);
      });
      document.addEventListener('keydown', function(e) {
        if (!confirmState.isOpen) return;
        var key = e.keyCode || e.which;
        if (key === 27) {
          closeConfirm(false);
        }
      });
    }

    return confirmEls;
  }

  window.showConfirmModal = function(options, callback) {
    var opts = options || {};
    var cb = callback;

    if (typeof cb !== 'function' && typeof window.Promise !== 'undefined') {
      return new window.Promise(function(resolve) {
        window.showConfirmModal(opts, function(result) { resolve(!!result); });
      });
    }

    var els = ensureConfirmModal();
    confirmState.isOpen = true;
    confirmState.callback = typeof cb === 'function' ? cb : null;

    els.titleEl.textContent = safeString(opts.title || 'Confirmation');
    els.messageEl.textContent = safeString(opts.message || '');
    els.confirmBtn.textContent = safeString(opts.confirmText || 'OK');
    els.cancelBtn.textContent = safeString(opts.cancelText || 'Annuler');

    els.backdrop.style.display = 'block';
    els.modal.style.display = 'block';

    window.setTimeout(function() {
      try { els.confirmBtn.focus(); } catch (e) {}
    }, 10);
  };

  var devicePosEls = null;
  var devicePosState = {
    isOpen: false
  };

  function isIndexPage() {
    return !!(document.body && document.body.classList.contains('page-index'));
  }

  function ensureDevicePosModal() {
    if (devicePosEls) return devicePosEls;

    var backdrop = document.getElementById(DEVICE_POS_BACKDROP_ID);
    if (!backdrop) {
      backdrop = document.createElement('div');
      backdrop.id = DEVICE_POS_BACKDROP_ID;
      backdrop.className = 'device-pos-backdrop';
      backdrop.setAttribute('aria-hidden', 'true');
      document.body.appendChild(backdrop);
    }

    var modal = document.getElementById(DEVICE_POS_MODAL_ID);
    if (!modal) {
      modal = document.createElement('div');
      modal.id = DEVICE_POS_MODAL_ID;
      modal.className = 'device-pos-modal';
      modal.setAttribute('role', 'dialog');
      modal.setAttribute('aria-modal', 'true');
      modal.setAttribute('aria-labelledby', 'devicePosTitle');
      modal.setAttribute('aria-describedby', 'devicePosMessage');
      modal.setAttribute('aria-hidden', 'true');
      modal.innerHTML = ''
        + '<h3 id="devicePosTitle">Point de vente par défaut</h3>'
        + '<p id="devicePosMessage">Sélectionnez le point de vente par défaut de cet appareil avant de continuer.</p>'
        + '<form id="devicePosForm">'
        + '  <label class="device-pos-label" for="devicePosSelect">Point de vente</label>'
        + '  <select id="devicePosSelect" class="device-pos-select" required>'
        + '    <option value="">Choisir un point de vente</option>'
        + '    <option value="lancon">Lançon</option>'
        + '    <option value="pelissanne">Pélissanne</option>'
        + '  </select>'
        + '  <div class="device-pos-actions">'
        + '    <button type="submit" class="device-pos-btn confirm">Enregistrer</button>'
        + '  </div>'
        + '</form>';
      document.body.appendChild(modal);
    }

    devicePosEls = {
      backdrop: backdrop,
      modal: modal,
      form: document.getElementById('devicePosForm'),
      select: document.getElementById('devicePosSelect')
    };

    if (modal.getAttribute('data-bound') !== '1') {
      modal.setAttribute('data-bound', '1');

      if (devicePosEls.form) {
        devicePosEls.form.addEventListener('submit', function(e) {
          var savedValue;
          e.preventDefault();
          savedValue = normalizePointVente(devicePosEls.select ? devicePosEls.select.value : '');
          if (!savedValue) {
            showToastInternal('Veuillez sélectionner un point de vente.', 'error', { duration: 3200 });
            return false;
          }

          if (!setDefaultPointVente(savedValue)) {
            showToastInternal('Impossible d’enregistrer le point de vente sur cet appareil.', 'error', { duration: 3600 });
            return false;
          }

          hideDevicePosModal();
          showToastInternal('Point de vente par défaut enregistré : ' + getDefaultPointVenteLabel(savedValue) + '.', 'success', { duration: 2600 });
          return false;
        });
      }
    }

    return devicePosEls;
  }

  function showDevicePosModal(forceOpen) {
    var els = ensureDevicePosModal();
    var currentValue = getDefaultPointVente();

    if (!forceOpen && currentValue) return;

    devicePosState.isOpen = true;
    if (els.select) {
      els.select.value = currentValue || '';
    }

    els.backdrop.style.display = 'block';
    els.modal.style.display = 'block';
    els.backdrop.setAttribute('aria-hidden', 'false');
    els.modal.setAttribute('aria-hidden', 'false');
    document.body.classList.add('device-pos-modal-open');

    window.setTimeout(function() {
      try { if (els.select) els.select.focus(); } catch (e) {}
    }, 10);
  }

  function hideDevicePosModal() {
    var els = ensureDevicePosModal();
    devicePosState.isOpen = false;
    els.backdrop.style.display = 'none';
    els.modal.style.display = 'none';
    els.backdrop.setAttribute('aria-hidden', 'true');
    els.modal.setAttribute('aria-hidden', 'true');
    document.body.classList.remove('device-pos-modal-open');
  }

  function openDevicePosChooser(forceOpen) {
    showDevicePosModal(!!forceOpen);
  }

  function ensureDevicePosChangeShortcut() {
    var existingBtn;
    var existingResetBtn;
    var heading;
    var btn;
    var resetBtn;
    var disabledResetMessage = 'Demander au FOU de réactiver';

    if (!isIndexPage()) return;

    existingBtn = document.getElementById('devicePosQuickChange');
    existingResetBtn = document.getElementById('deviceLocalDataReset');
    if (existingBtn && existingResetBtn) return;

    heading = document.querySelector('.section-heading');
    if (!heading || !heading.parentNode) return;

    btn = document.createElement('button');
    btn.type = 'button';
    btn.id = 'devicePosQuickChange';
    btn.className = 'device-pos-quick-change';
    btn.textContent = 'PDV appareil : ' + (getDefaultPointVente() ? getDefaultPointVenteLabel(getDefaultPointVente()) : 'non défini');
    btn.addEventListener('click', function(e) {
      e.preventDefault();
      openDevicePosChooser(true);
    });
    heading.parentNode.insertBefore(btn, heading.nextSibling);

    resetBtn = document.createElement('button');
    resetBtn.type = 'button';
    resetBtn.id = 'deviceLocalDataReset';
    resetBtn.className = 'device-pos-reset is-disabled';
    resetBtn.textContent = 'Supprimer les données locales';
    resetBtn.setAttribute('aria-disabled', 'true');
    resetBtn.setAttribute('title', disabledResetMessage);
    resetBtn.addEventListener('click', function(e) {
      e.preventDefault();
      e.stopPropagation();
      showToastInternal(disabledResetMessage, 'info', { duration: 2800 });
    });
    resetBtn.addEventListener('keydown', function(e) {
      var key = e.key || '';
      if (key !== 'Enter' && key !== ' ') return;

      e.preventDefault();
      e.stopPropagation();
      showToastInternal(disabledResetMessage, 'info', { duration: 2800 });
    });

    btn.insertAdjacentElement('afterend', resetBtn);
  }

  function refreshDevicePosQuickChangeLabel() {
    var btn = document.getElementById('devicePosQuickChange');
    var currentValue = getDefaultPointVente();
    if (!btn) return;
    btn.textContent = 'PDV appareil : ' + (currentValue ? getDefaultPointVenteLabel(currentValue) : 'non défini');
  }

  window.openDevicePosChooser = function() {
    openDevicePosChooser(true);
  };

  $(function() {
    if (typeof window.consumeToastFromUrl === 'function') {
      window.consumeToastFromUrl();
    }

    if (window.__flashToast && window.__flashToast.message) {
      showToastInternal(window.__flashToast.message, window.__flashToast.type || 'info', { duration: window.__flashToast.duration || 3600 });
      if (window.__flashToast.redirect) {
        window.setTimeout(function() {
          window.location.href = window.__flashToast.redirect;
        }, 900);
      }
      try { delete window.__flashToast; } catch (e) {}
    }

    if (isIndexPage()) {
      ensureDevicePosModal();
      ensureDevicePosChangeShortcut();
      refreshDevicePosQuickChangeLabel();

      if (!getDefaultPointVente()) {
        showDevicePosModal(false);
      }

      document.addEventListener('default-point-vente:changed', function() {
        refreshDevicePosQuickChangeLabel();
      });
    }
  });

})(window, document, jQuery);"
            }
        },
        {
            "path": "pointages/modif_pointage.php",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 6039,
                "sha1": "69023f590ae15ed673716eca31cd4852132b2375",
                "content_b64": "<?php
session_start();
/* doc-project | pointages/modif_pointage.php | Gère l’accès sécurisé par appareil et l’affichage du formulaire de modification d’un pointage existant. | Expose: aucun | Dépend de: config.php, includes/device_auth.php, traitement_modif_pointage.php, index.php, connexion.php, pointage.php | Impacte: session PHP, cookie d’appareil, affichage UI, redirection d’accès | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_pointages(PointageID, UserID, DateHeureEntree, DateHeureSortie), z_ptg_aqp_utilisateurs(UserID, Nom, Prenom) */

require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_device_authorized($pdo);

date_default_timezone_set('Europe/Paris');

$pointage = null;
$employe = null;
$userId = null;

function formatDatetimeLocal($value) {
    $raw = is_string($value) ? trim($value) : '';
    if ($raw === '') return '';
    try {
        $dt = new DateTime($raw);
        $dt->setTimezone(new DateTimeZone('Europe/Paris'));
        return $dt->format('Y-m-d\TH:i');
    } catch (Exception $e) {
        return '';
    }
}

if (isset($_GET['pointageId'])) {
    $pointageId = (int)$_GET['pointageId'];

    // Récupération des informations de pointage
    $stmt = $pdo->prepare("SELECT * FROM z_ptg_aqp_pointages WHERE PointageID = :pointageId");
    $stmt->execute([':pointageId' => $pointageId]);
    $pointage = $stmt->fetch(PDO::FETCH_ASSOC);

    if ($pointage && isset($pointage['UserID'])) {
        $userId = (int)$pointage['UserID'];
        $stmt = $pdo->prepare("SELECT Nom, Prenom FROM z_ptg_aqp_utilisateurs WHERE UserID = :userID LIMIT 1");
        $stmt->execute([':userID' => $userId]);
        $employe = $stmt->fetch(PDO::FETCH_ASSOC);
    }
}

$entreeValue = $pointage ? formatDatetimeLocal($pointage['DateHeureEntree'] ?? '') : '';
$sortieValue = $pointage ? formatDatetimeLocal($pointage['DateHeureSortie'] ?? '') : '';

$employeeFullName = '';
if (is_array($employe)) {
    $nom = isset($employe['Nom']) ? (string)$employe['Nom'] : '';
    $prenom = isset($employe['Prenom']) ? (string)$employe['Prenom'] : '';
    $employeeFullName = trim($nom . ' ' . $prenom);
}
?>
<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">

<title>Modification pointage</title>

<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">

<link rel="stylesheet" href="css/bootstrap.min.css">

<link rel="stylesheet" href="css/bootstrap-theme.min.css">

<link rel="stylesheet" href="css/fontAwesome.css">

<link rel="stylesheet" href="css/hero-slider.css">

<link rel="stylesheet" href="css/tooplate-style.css">

<?php
$cssPath = __DIR__ . '/css/style.css';
$cssVersion = file_exists($cssPath) ? filemtime($cssPath) : time();
$modifCssPath = __DIR__ . '/css/modif-pointage.css';
$modifCssVersion = file_exists($modifCssPath) ? filemtime($modifCssPath) : time();
?>
<link rel="stylesheet" href="css/style.css?v=<?php echo (int)$cssVersion; ?>">
<link rel="stylesheet" href="css/modif-pointage.css?v=<?php echo (int)$modifCssVersion; ?>">

<link href="https://fonts.googleapis.com/css?family=Open+Sans:300,400,600,700,800" rel="stylesheet">

<script src="js/vendor/modernizr-2.8.3-respond-1.4.2.min.js"></script>

</head>

<body class="page-modif-pointage theme-dark">

<main class="ptg-page" role="main">
  <div class="ptg-card">
    <header class="ptg-header">
      <h1>Modification d’un pointage</h1>
      <?php if ($employeeFullName !== ''): ?>
        <p class="ptg-subtitle"><?php echo htmlspecialchars($employeeFullName, ENT_QUOTES, 'UTF-8'); ?></p>
      <?php endif; ?>
    </header>

    <?php if (!$pointage): ?>
      <div class="ptg-alert" role="status">
        Pointage introuvable (ou identifiant manquant).
      </div>
      <div class="ptg-actions">
        <a class="ptg-btn secondary" href="index.php">Retour accueil</a>
      </div>
    <?php else: ?>
      <form class="ptg-form" action="traitement_modif_pointage.php" method="post" autocomplete="off">
        <input type="hidden" name="pointageId" value="<?php echo (int)$pointage['PointageID']; ?>">

        <div class="ptg-field">
          <label for="DateHeureEntree">Date / heure d’entrée</label>
          <input
            id="DateHeureEntree"
            type="datetime-local"
            name="DateHeureEntree"
            value="<?php echo htmlspecialchars($entreeValue, ENT_QUOTES, 'UTF-8'); ?>"
            required
          >
          <p class="ptg-help">Format local (Europe/Paris).</p>
        </div>

        <div class="ptg-field">
          <label for="DateHeureSortie">Date / heure de sortie</label>
          <input
            id="DateHeureSortie"
            type="datetime-local"
            name="DateHeureSortie"
            value="<?php echo htmlspecialchars($sortieValue, ENT_QUOTES, 'UTF-8'); ?>"
          >
          <p class="ptg-help">Laissez vide si le pointage doit rester “en cours”.</p>
        </div>

        <div class="ptg-actions">
          <?php if (is_int($userId) || is_numeric($userId)): ?>
            <a class="ptg-btn secondary" href="pointage.php?employe=<?php echo (int)$userId; ?>&view=1">Annuler</a>
          <?php else: ?>
            <a class="ptg-btn secondary" href="index.php">Annuler</a>
          <?php endif; ?>
          <button type="submit" class="ptg-btn primary">Valider</button>
        </div>
      </form>
    <?php endif; ?>
  </div>
</main>

<script src="//ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
<script>window.jQuery || document.write('<script src="js/vendor/jquery-1.11.2.min.js"><\/script>')</script>
<script src="js/vendor/bootstrap.min.js"></script>
<script src="js/plugins.js"></script>
<script src="js/main.js"></script>

</body>
</html>
            
            </body>
            
            </html>"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 6260,
                "sha1": "7079f445a548d5fc0c6bece1b753c0074bf03540",
                "content_b64": "<?php
session_start();
/* doc-project | pointages/modif_pointage.php | Gère l’accès sécurisé par appareil et l’affichage du formulaire de modification d’un pointage existant avec assets locaux versionnés sans Modernizr/Respond ni Bootstrap JS. | Expose: aucun | Dépend de: config.php, includes/device_auth.php, includes/asset_version.php, traitement_modif_pointage.php, index.php, connexion.php, pointage.php, js/vendor/jquery-1.11.2.min.js, js/plugins.js, js/main.js | Impacte: session PHP, cookie d’appareil, affichage UI, redirection d’accès, toasts internes | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_pointages(PointageID, UserID, DateHeureEntree, DateHeureSortie), z_ptg_aqp_utilisateurs(UserID, Nom, Prenom) */

require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_once __DIR__ . "/includes/asset_version.php";
require_device_authorized($pdo);

date_default_timezone_set('Europe/Paris');

$pointage = null;
$employe = null;
$userId = null;

function formatDatetimeLocal($value) {
    $raw = is_string($value) ? trim($value) : '';
    if ($raw === '') return '';
    try {
        $dt = new DateTime($raw);
        $dt->setTimezone(new DateTimeZone('Europe/Paris'));
        return $dt->format('Y-m-d\TH:i');
    } catch (Exception $e) {
        return '';
    }
}

if (isset($_GET['pointageId'])) {
    $pointageId = (int)$_GET['pointageId'];

    // Récupération des informations de pointage
    $stmt = $pdo->prepare("SELECT * FROM z_ptg_aqp_pointages WHERE PointageID = :pointageId");
    $stmt->execute([':pointageId' => $pointageId]);
    $pointage = $stmt->fetch(PDO::FETCH_ASSOC);

    if ($pointage && isset($pointage['UserID'])) {
        $userId = (int)$pointage['UserID'];
        $stmt = $pdo->prepare("SELECT Nom, Prenom FROM z_ptg_aqp_utilisateurs WHERE UserID = :userID LIMIT 1");
        $stmt->execute([':userID' => $userId]);
        $employe = $stmt->fetch(PDO::FETCH_ASSOC);
    }
}

$entreeValue = $pointage ? formatDatetimeLocal($pointage['DateHeureEntree'] ?? '') : '';
$sortieValue = $pointage ? formatDatetimeLocal($pointage['DateHeureSortie'] ?? '') : '';

$employeeFullName = '';
if (is_array($employe)) {
    $nom = isset($employe['Nom']) ? (string)$employe['Nom'] : '';
    $prenom = isset($employe['Prenom']) ? (string)$employe['Prenom'] : '';
    $employeeFullName = trim($nom . ' ' . $prenom);
}
?>
<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">

<title>Modification pointage</title>

<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">

<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap-theme.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/fontAwesome.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/hero-slider.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/tooplate-style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/modif-pointage.css'), ENT_QUOTES, 'UTF-8'); ?>">

</head>

<body class="page-modif-pointage theme-dark">

<main class="ptg-page" role="main">
  <div class="ptg-card">
    <header class="ptg-header">
      <h1>Modification d’un pointage</h1>
      <?php if ($employeeFullName !== ''): ?>
        <p class="ptg-subtitle"><?php echo htmlspecialchars($employeeFullName, ENT_QUOTES, 'UTF-8'); ?></p>
      <?php endif; ?>
    </header>

    <?php if (!$pointage): ?>
      <div class="ptg-alert" role="status">
        Pointage introuvable (ou identifiant manquant).
      </div>
      <div class="ptg-actions">
        <a class="ptg-btn secondary" href="index.php">Retour accueil</a>
      </div>
    <?php else: ?>
      <form class="ptg-form" action="traitement_modif_pointage.php" method="post" autocomplete="off">
        <input type="hidden" name="pointageId" value="<?php echo (int)$pointage['PointageID']; ?>">

        <div class="ptg-field">
          <label for="DateHeureEntree">Date / heure d’entrée</label>
          <input
            id="DateHeureEntree"
            type="datetime-local"
            name="DateHeureEntree"
            value="<?php echo htmlspecialchars($entreeValue, ENT_QUOTES, 'UTF-8'); ?>"
            required
          >
          <p class="ptg-help">Format local (Europe/Paris).</p>
        </div>

        <div class="ptg-field">
          <label for="DateHeureSortie">Date / heure de sortie</label>
          <input
            id="DateHeureSortie"
            type="datetime-local"
            name="DateHeureSortie"
            value="<?php echo htmlspecialchars($sortieValue, ENT_QUOTES, 'UTF-8'); ?>"
          >
          <p class="ptg-help">Laissez vide si le pointage doit rester “en cours”.</p>
        </div>

        <div class="ptg-actions">
          <?php if (is_int($userId) || is_numeric($userId)): ?>
            <a class="ptg-btn secondary" href="pointage.php?employe=<?php echo (int)$userId; ?>&view=1">Annuler</a>
          <?php else: ?>
            <a class="ptg-btn secondary" href="index.php">Annuler</a>
          <?php endif; ?>
          <button type="submit" class="ptg-btn primary">Valider</button>
        </div>
      </form>
    <?php endif; ?>
  </div>
</main>

<script src="<?php echo htmlspecialchars(asset_version_url('js/vendor/jquery-1.11.2.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/plugins.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
<script src="<?php echo htmlspecialchars(asset_version_url('js/main.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>

</body>
</html>"
            }
        },
        {
            "path": "pointages/pointage.php",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 21002,
                "sha1": "fb4cc13eb2c401f4ccf913d3c8e7888be6b2e42e",
                "content_b64": "<?php

/* doc-project | pointages/pointage.php | Affiche, sécurise et pilote l’interface de pointage avec autorisation permanente par appareil. | Expose: aucun | Dépend de: config.php, includes/device_auth.php, connexion.php, enregistrement_arrivee.php, enregistrement_depart.php, sortir_employe.php, modif_pointage.php | Impacte: session PHP, cookie d’appareil, redirections, affichage UI, AJAX, pointages en base | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_utilisateurs(UserID, Nom, Prenom), z_ptg_aqp_pointages(PointageID, UserID, DateHeureEntree, DateHeureSortie) */


// Démarrage ou reprise de la session
session_start();
require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_device_authorized($pdo);

date_default_timezone_set('Europe/Paris');

// Récupération de l'ID de l'employé
$employeId = isset($_GET['employe']) ? intval($_GET['employe']) : null;
$viewOnly = (isset($_GET['view']) && $_GET['view'] === '1');

// Récupération des informations de l'employé depuis la BDD
$stmt = $pdo->prepare("SELECT Nom, Prenom FROM z_ptg_aqp_utilisateurs WHERE UserID = :userID");
$stmt->execute([':userID' => $employeId]);
$employe = $stmt->fetch(PDO::FETCH_ASSOC);

// Récupération du dernier pointage de l'employé
$stmt = $pdo->prepare("SELECT * FROM z_ptg_aqp_pointages WHERE UserID = :userID ORDER BY DateHeureEntree DESC, PointageID DESC LIMIT 1");
$stmt->execute([':userID' => $employeId]);
$lastPointage = $stmt->fetch(PDO::FETCH_ASSOC);

$isWorking = $lastPointage && $lastPointage['DateHeureSortie'] === null;

?>
<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">





<title>Pointages salariés</title>

<!--



Template 2089 Meteor



http://www.tooplate.com/view/2089-meteor



-->

<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">



<link rel="stylesheet" href="css/bootstrap.min.css">

<link rel="stylesheet" href="css/bootstrap-theme.min.css">

<link rel="stylesheet" href="css/fontAwesome.css">

<link rel="stylesheet" href="css/fontAwesome.css">
<link rel="stylesheet" href="css/hero-slider.css">
<link rel="stylesheet" href="css/tooplate-style.css">
<?php
$cssPath = __DIR__ . '/css/style.css';
$cssVersion = file_exists($cssPath) ? filemtime($cssPath) : time();
?>
<link rel="stylesheet" href="css/style.css?v=<?php echo (int)$cssVersion; ?>">



<link href="https://fonts.googleapis.com/css?family=Open+Sans:300,400,600,700,800" rel="stylesheet">



<script src="js/vendor/modernizr-2.8.3-respond-1.4.2.min.js"></script>

</head>



<body class="page-pointage theme-dark">




<div id="about" class="page-section">
<div class="container">
<div class="row">
<div class="col-md-12">
<div class="section-heading">
<h1>Pointages <?php echo $employe['Nom'] . " " . $employe['Prenom']; ?></h1>
<div class="line-dec"></div>
<center>
<font color="white">
<br>
<button type="button" id="backHomeButton" class="btn" onclick="window.location.href='index.php';">Retour accueil</button>
<br>
<?php if (!$viewOnly): ?>
    <?php if (!$isWorking): ?>
        <form id="contact">
        <button data-employe="<?php echo $employeId; ?>" id="arrivalButton" class="btn">Arrivée</button>
        </form>
    <?php else: ?>
        <form id="contact2">
        <button data-employe="<?php echo $employeId; ?>" id="departureButton" class="btn">Départ</button>
        </form>
    <?php endif; ?>
<?php endif; ?>
        <!-- Autre contenu... -->
        </font>
        </center>
        </div>
        </div>
        </div>
        </div>
        
        
        <br><br> 
        
        
        
        
        
        <div class="line-dec"></div>
        
        <br><br>
        <center>
        <?php
        // Récupération de l'ID de l'employé
        $employeId = isset($_GET['employe']) ? intval($_GET['employe']) : null;
        
        // Initialisation des variables pour le tableau
        $total_hours = 0;
        $table_rows = "";
        
        $current_month = date('m');
        $current_year = date('Y');
        date_default_timezone_set('Europe/Paris');
        
        
        // Récupération des pointages de l'employé depuis la BDD
        $stmt = $pdo->prepare("SELECT PointageID, UNIX_TIMESTAMP(DateHeureEntree) AS EntreeTimestamp, UNIX_TIMESTAMP(DateHeureSortie) AS SortieTimestamp FROM z_ptg_aqp_pointages WHERE UserID = :userID AND MONTH(DateHeureEntree) = :currentMonth AND YEAR(DateHeureEntree) = :currentYear");
        $stmt->execute([':userID' => $employeId, ':currentMonth' => $current_month, ':currentYear' => $current_year]);
        $pointages = $stmt->fetchAll(PDO::FETCH_ASSOC);
        
        foreach ($pointages as $pointage) {
            // Création de l'objet DateTime pour l'heure d'arrivée
            $arrival_time = DateTime::createFromFormat('U', $pointage['EntreeTimestamp']);
            $arrival_time->setTimezone(new DateTimeZone('Europe/Paris'));
            $date_display = $arrival_time->format('Y-m-d');
            $arrival_time_display = $arrival_time->format('H:i');
            
            // Gérer le cas où le pointage de sortie n'est pas encore défini
            if (empty($pointage['SortieTimestamp'])) {
                $departure_time_display = "En cours";
                $work_hours_display = "En cours";
            } else {
                // Création de l'objet DateTime pour l'heure de départ
                $departure_time = DateTime::createFromFormat('U', $pointage['SortieTimestamp']);
                $departure_time->setTimezone(new DateTimeZone('Europe/Paris'));
                $departure_time_display = $departure_time->format('H:i');
                
                // Calcul des heures travaillées
                $work_seconds = $pointage['SortieTimestamp'] - $pointage['EntreeTimestamp'];
                $work_hours = $work_seconds / 3600;
                $work_hours_display = number_format($work_hours, 2);
                $total_hours += $work_hours;
            }
            
            // Ajout de la ligne dans le tableau
            $table_rows .= "<tr data-pointage-id='".$pointage['PointageID']."' data-entree-timestamp='".$pointage['EntreeTimestamp']."' data-sortie-timestamp='".$pointage['SortieTimestamp']."' onclick='afficherModal(this)'><td>{$date_display}</td><td>{$arrival_time_display}</td><td>{$departure_time_display}</td><td>$work_hours_display</td></tr>";
        }
        
        
        
        
        // Affichage du tableau des pointages
        echo "<table>";
        echo "<tr><th>Date</th><th>Heure d'arrivée</th><th>Heure de départ</th><th>Nombre d'heures</th></tr>";
        echo $table_rows;
        echo "</table>";
        
        ?>
        
        
        <font color="white">
        <?php
        // Affichage du total d'heures travaillées
        echo "Total d'heures travaillées : " . number_format($total_hours, 2);
        ?>
        </font>
        
        
        
        
        
        <br><br><br>
        <font color="white">
        <h4>Affichage sur dates sélectionnées</h4>
        </font>
        
        <br>
        <div class="d1">
        
        <div class="map">
        
        <form id="contact" method="post" action="">
        
        <font color="white"> Date de début : </font><font color="black"><input type="date" name="start_date" required></font><br><br>
        
        <font color="white">Date de fin : </font><font color="black"><input type="date" name="end_date" required></font><br><br>
        
        <font color="black"><input class="btn" id="form-submit" type="submit" value="Afficher la sélection"></font>
        
        </form>
        
        </div></div><br><br><br><br><br><br>
        
        <font color="white">
        
        
        
        <?php
        // Vérification de la soumission du formulaire
        if ($_SERVER['REQUEST_METHOD'] == 'POST' && isset($_GET['employe'])) {
            $employeId = intval($_GET['employe']);
            $start_date = $_POST['start_date'];
            $end_date = $_POST['end_date'];
            
            // Conversion des dates en format Y-m-d
            $start_date_formatted = date('Y-m-d', strtotime($start_date));
            $end_date_formatted = date('Y-m-d', strtotime($end_date . ' +1 day'));
            
            $total_hours = 0;
            $table_rows = "";
            date_default_timezone_set('Europe/Paris');
            
            // Requête pour obtenir les pointages entre les dates spécifiées
            $stmt = $pdo->prepare("SELECT PointageID, UNIX_TIMESTAMP(DateHeureEntree) AS EntreeTimestamp, UNIX_TIMESTAMP(DateHeureSortie) AS SortieTimestamp FROM z_ptg_aqp_pointages WHERE UserID = :userID AND DateHeureEntree BETWEEN :startDate AND :endDate");
            
            $stmt->execute([':userID' => $employeId, ':startDate' => $start_date_formatted, ':endDate' => $end_date_formatted]);
            $pointages = $stmt->fetchAll(PDO::FETCH_ASSOC);
            if (!$stmt->execute()) {
                print_r($stmt->errorInfo());
            }
            
            foreach ($pointages as $pointage) {
                // Création de l'objet DateTime pour l'heure d'arrivée
                $arrival_time = DateTime::createFromFormat('U', $pointage['EntreeTimestamp']);
                $arrival_time->setTimezone(new DateTimeZone('Europe/Paris'));
                $date_display = $arrival_time->format('Y-m-d');
                $arrival_time_display = $arrival_time->format('H:i');
                
                // Gérer le cas où le pointage de sortie n'est pas encore défini
                if (empty($pointage['SortieTimestamp'])) {
                    $departure_time_display = "En cours";
                    $work_hours_display = "En cours";
                } else {
                    // Création de l'objet DateTime pour l'heure de départ
                    $departure_time = DateTime::createFromFormat('U', $pointage['SortieTimestamp']);
                    $departure_time->setTimezone(new DateTimeZone('Europe/Paris'));
                    $departure_time_display = $departure_time->format('H:i');
                    
                    // Calcul des heures travaillées
                    $work_seconds = $pointage['SortieTimestamp'] - $pointage['EntreeTimestamp'];
                    $work_hours = $work_seconds / 3600;
                    $work_hours_display = number_format($work_hours, 2);
                    $total_hours += $work_hours;
                }
                
                // Ajout de la ligne dans le tableau
                $table_rows .= "<tr data-pointage-id='".$pointage['PointageID']."' data-entree-timestamp='".$pointage['EntreeTimestamp']."' data-sortie-timestamp='".$pointage['SortieTimestamp']."' onclick='afficherModal(this)'><td>{$date_display}</td><td>{$arrival_time_display}</td><td>{$departure_time_display}</td><td>$work_hours_display</td></tr>";
            }
            
            
            
            // Affichage du tableau
            echo "<table><tr><th>Date</th><th>Heure d'arrivée</th><th>Heure de départ</th><th>Nombre d'heures</th></tr>$table_rows</table>";
            echo "Total d'heures travaillées : " . number_format($total_hours, 2);
            
            
        }
        ?>
        
        
        <br><br><br>
        
        <p><a href="#" onclick="promptExitPin()">Sortir Employé</a></p>
        
        </div>
        
        
        </div></div>
        
        </div>
        
        </div>
        
        
        
        </center>
        
        
        <!-- Modal PIN (design refait) -->
        <div id="modalBackground" onclick="closePinModal()" aria-hidden="true"></div>
        <div id="pinModal" role="dialog" aria-modal="true" aria-labelledby="pinModalTitle">
          <form id="pinForm" onsubmit="verifierPin(); return false;">
            <h2 id="pinModalTitle" class="pin-modal-title">Code PIN</h2>
            <p class="pin-modal-subtitle">Saisissez votre code à 4 chiffres</p>

            <div class="pin-input-wrapper" onclick="focusPinInput()">
              <label for="pinInput" class="sr-only">Code PIN (4 chiffres)</label>
              <div class="pin-boxes" aria-hidden="true">
                <div class="pin-box" data-idx="0"></div>
                <div class="pin-box" data-idx="1"></div>
                <div class="pin-box" data-idx="2"></div>
                <div class="pin-box" data-idx="3"></div>
              </div>
              <input
                type="tel"
                id="pinInput"
                class="pin-hidden-input"
                inputmode="numeric"
                pattern="[0-9]*"
                maxlength="4"
                autocomplete="off"
                autocapitalize="off"
                autocorrect="off"
                spellcheck="false"
                title="Seulement des chiffres."
                value=""
              >
            </div>

            <div class="pin-modal-actions">
              <button type="button" class="pin-modal-btn cancel" onclick="closePinModal()">Annuler</button>
              <button type="submit" class="pin-modal-btn confirm">Valider</button>
            </div>
          </form>
        </div>
        
        
        
        <script>
        
        
        
        
        
        
        
        </script>
        
        
        
        
        
        
        <script src="//ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
        
        <script src="js/vendor/bootstrap.min.js"></script>
        
        <script src="js/plugins.js"></script>
        
        <script src="js/main.js"></script>
        
        


        <script type="text/javascript">

        var employeeFirstName = <?php echo json_encode(($employe && isset($employe['Prenom'])) ? (string)$employe['Prenom'] : '', JSON_UNESCAPED_UNICODE | JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT); ?>;
        function buildPunchToast(action) { var p = employeeFirstName ? (" " + employeeFirstName) : ""; return (action === "depart") ? ("Départ" + p + " enregistré") : ("Arrivée" + p + " enregistrée"); }

        $("#arrivalButton").click(function(e) {
            e.preventDefault();
            let employe = $(this).data('employe');
            $.ajax({
                type: "POST",
                url: "enregistrement_arrivee.php",
                data: {employe: employe},
                success: function(response) {
                    var msg = buildPunchToast("arrivee");
                    window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
                },
                error: function() {
                    if (typeof window.showToast === "function") {
                      window.showToast("Erreur lors de l'enregistrement du pointage.", "error");
                    } else {
                      alert("Erreur lors de l'enregistrement du pointage.");
                    }
                }
            });
        });
        
        
        $("#departureButton").click(function(e) {
            e.preventDefault();
            let employe = $(this).data('employe');
            $.ajax({
                type: "POST",
                url: "enregistrement_depart.php",
                data: {employe: employe},
                success: function(response) {
                    var msg = buildPunchToast("depart");
                    window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
                },
                error: function() {
                    if (typeof window.showToast === "function") {
                      window.showToast("Erreur lors de l'enregistrement du pointage.", "error");
                    } else {
                      alert("Erreur lors de l'enregistrement du pointage.");
                    }
                }
            });
        });
        
        
        var actionContext = ""; // Variable globale pour stocker le contexte de l'action

function sanitizePinValue(val) {
  return String(val || '').replace(/\D/g, '').slice(0, 4);
}

function updatePinBoxes(pin) {
  var boxes = document.querySelectorAll('#pinModal .pin-box');
  for (var i = 0; i < boxes.length; i++) {
    boxes[i].textContent = (i < pin.length) ? '★' : '';
  }
}

function focusPinInput() {
  var input = document.getElementById('pinInput');
  if (!input) return;
  try { input.focus(); } catch (e) {}
}

function resetPinModal() {
  var input = document.getElementById('pinInput');
  if (input) input.value = '';
  updatePinBoxes('');
}

function openPinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (!modal || !bg) return;
  modal.style.display = 'block';
  bg.style.display = 'block';
  resetPinModal();
  window.setTimeout(function() { focusPinInput(); }, 10);
}

function closePinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (modal) modal.style.display = 'none';
  if (bg) bg.style.display = 'none';
  resetPinModal();
}

// Bind input once (digits only + sync UI boxes)
(function bindPinOnce() {
  var input = document.getElementById('pinInput');
  if (!input || input.getAttribute('data-bound') === '1') return;
  input.setAttribute('data-bound', '1');
  input.addEventListener('input', function() {
    var clean = sanitizePinValue(input.value);
    if (input.value !== clean) input.value = clean;
    updatePinBoxes(clean);
  });
  input.addEventListener('keydown', function(e) {
    var key = e.key || '';
    if (key === 'Escape') {
      closePinModal();
    }
  });
})();

function afficherModal(element) {
    var pointageId = element.getAttribute('data-pointage-id');
    window.pointageIdPourModification = pointageId;
    window.location.href = "modif_pointage.php?pointageId=" + encodeURIComponent(pointageId);
}

function verifierPin() {
    var input = document.getElementById('pinInput');
    var pin = sanitizePinValue(input ? input.value : '');
    if (input && input.value !== pin) input.value = pin;
    updatePinBoxes(pin);

    if (pin.length !== 4) {
        if (typeof window.showToast === "function") {
          window.showToast("Entrez un code PIN à 4 chiffres.", "error");
        } else {
          alert("Entrez un code PIN à 4 chiffres.");
        }
        try { focusPinInput(); } catch (e) {}
        return;
    }
    if (pin) {
        // Envoi du code PIN à un script PHP pour vérification
        var xhr = new XMLHttpRequest();
        xhr.onreadystatechange = function() {
            if (this.readyState == 4 && this.status == 200) {
                if (this.responseText == "OK") {
                    // Exécuter l'action en fonction du contexte
                    if (actionContext === "modificationPointage") {
                        window.location.href = "modif_pointage.php?pointageId=" + window.pointageIdPourModification;
                    } else if (actionContext === "sortieEmploye") {
                        sortirEmploye(window.pointageIdPourSortie);
                    }
                } else {
                    if (typeof window.showToast === "function") {
                      window.showToast("Code PIN incorrect.", "error");
                    } else {
                      alert("Code PIN incorrect.");
                    }
                    resetPinModal();
                    try { focusPinInput(); } catch (e) {}
                }
            }
        };
        xhr.open("POST", "verifier_pin.php", true);
        xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
        xhr.send("pin=" + pin);
    }
}

function promptExitPin() {
    window.pointageIdPourSortie = <?php echo $employeId; ?>;
    sortirEmploye(window.pointageIdPourSortie);
}

function sortirEmploye(employeId) {
    $.ajax({
        type: "POST",
        url: "sortir_employe.php", // Script PHP pour sortir l'employé
        data: {employe: employeId},
        success: function(response) {
            var msg = "Statut de l'employé mis à jour.";
            window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
        },
        error: function() {
            if (typeof window.showToast === "function") {
              window.showToast("Erreur lors de la mise à jour du statut employé.", "error");
            } else {
              alert("Erreur lors de la mise à jour du statut employé.");
            }
        }
    });
}

            
            
            </script>
            
            
            
            </body>
            
            </html>"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 21480,
                "sha1": "8494f3c5d174b38d0f4ff29871209011ca5b26fd",
                "content_b64": "<?php

/* doc-project | pointages/pointage.php | Affiche, sécurise et pilote l’interface de pointage avec autorisation permanente par appareil et assets locaux versionnés sans Modernizr/Respond ni Bootstrap JS. | Expose: aucun | Dépend de: config.php, includes/device_auth.php, includes/asset_version.php, connexion.php, enregistrement_arrivee.php, enregistrement_depart.php, sortir_employe.php, modif_pointage.php, js/vendor/jquery-1.11.2.min.js, js/plugins.js, js/main.js | Impacte: session PHP, cookie d’appareil, redirections, affichage UI, AJAX, pointages en base, toasts internes | Tables: pos_device_authorizations(token_hash, authorized_at, last_used_at), z_ptg_aqp_utilisateurs(UserID, Nom, Prenom), z_ptg_aqp_pointages(PointageID, UserID, DateHeureEntree, DateHeureSortie) */


// Démarrage ou reprise de la session
session_start();
require_once "config.php";
require_once __DIR__ . "/includes/device_auth.php";
require_once __DIR__ . "/includes/asset_version.php";
require_device_authorized($pdo);

date_default_timezone_set('Europe/Paris');

// Récupération de l'ID de l'employé
$employeId = isset($_GET['employe']) ? intval($_GET['employe']) : null;
$viewOnly = (isset($_GET['view']) && $_GET['view'] === '1');

// Récupération des informations de l'employé depuis la BDD
$stmt = $pdo->prepare("SELECT Nom, Prenom FROM z_ptg_aqp_utilisateurs WHERE UserID = :userID");
$stmt->execute([':userID' => $employeId]);
$employe = $stmt->fetch(PDO::FETCH_ASSOC);

// Récupération du dernier pointage de l'employé
$stmt = $pdo->prepare("SELECT * FROM z_ptg_aqp_pointages WHERE UserID = :userID ORDER BY DateHeureEntree DESC, PointageID DESC LIMIT 1");
$stmt->execute([':userID' => $employeId]);
$lastPointage = $stmt->fetch(PDO::FETCH_ASSOC);

$isWorking = $lastPointage && $lastPointage['DateHeureSortie'] === null;

?>
<!DOCTYPE html>

<html>

<head>

<meta http-equiv="content-type" content="text/html; charset=utf-8" />

<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">





<title>Pointages salariés</title>

<!--



Template 2089 Meteor



http://www.tooplate.com/view/2089-meteor



-->

<meta name="description" content="">

<meta name="viewport" content="width=device-width, initial-scale=1">

<link rel="apple-touch-icon" href="apple-touch-icon.png">



<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/bootstrap-theme.min.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/fontAwesome.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/hero-slider.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/tooplate-style.css'), ENT_QUOTES, 'UTF-8'); ?>">
<link rel="stylesheet" href="<?php echo htmlspecialchars(asset_version_url('css/style.css'), ENT_QUOTES, 'UTF-8'); ?>">

</head>



<body class="page-pointage theme-dark">




<div id="about" class="page-section">
<div class="container">
<div class="row">
<div class="col-md-12">
<div class="section-heading">
<h1>Pointages <?php echo htmlspecialchars(($employe['Nom'] ?? '') . " " . ($employe['Prenom'] ?? ''), ENT_QUOTES, 'UTF-8'); ?></h1>
<div class="line-dec"></div>
<center>
<font color="white">
<br>
<button type="button" id="backHomeButton" class="btn" onclick="window.location.href='index.php';">Retour accueil</button>
<br>
<?php if (!$viewOnly): ?>
    <?php if (!$isWorking): ?>
        <form id="contact">
        <button data-employe="<?php echo $employeId; ?>" id="arrivalButton" class="btn">Arrivée</button>
        </form>
    <?php else: ?>
        <form id="contact2">
        <button data-employe="<?php echo $employeId; ?>" id="departureButton" class="btn">Départ</button>
        </form>
    <?php endif; ?>
<?php endif; ?>
        <!-- Autre contenu... -->
        </font>
        </center>
        </div>
        </div>
        </div>
        </div>
        
        
        <br><br> 
        
        
        
        
        
        <div class="line-dec"></div>
        
        <br><br>
        <center>
        <?php
        // Récupération de l'ID de l'employé
        $employeId = isset($_GET['employe']) ? intval($_GET['employe']) : null;
        
        // Initialisation des variables pour le tableau
        $total_hours = 0;
        $table_rows = "";
        
        $current_month = date('m');
        $current_year = date('Y');
        date_default_timezone_set('Europe/Paris');
        
        
        // Récupération des pointages de l'employé depuis la BDD
        $stmt = $pdo->prepare("SELECT PointageID, UNIX_TIMESTAMP(DateHeureEntree) AS EntreeTimestamp, UNIX_TIMESTAMP(DateHeureSortie) AS SortieTimestamp FROM z_ptg_aqp_pointages WHERE UserID = :userID AND MONTH(DateHeureEntree) = :currentMonth AND YEAR(DateHeureEntree) = :currentYear");
        $stmt->execute([':userID' => $employeId, ':currentMonth' => $current_month, ':currentYear' => $current_year]);
        $pointages = $stmt->fetchAll(PDO::FETCH_ASSOC);
        
        foreach ($pointages as $pointage) {
            // Création de l'objet DateTime pour l'heure d'arrivée
            $arrival_time = DateTime::createFromFormat('U', $pointage['EntreeTimestamp']);
            $arrival_time->setTimezone(new DateTimeZone('Europe/Paris'));
            $date_display = $arrival_time->format('Y-m-d');
            $arrival_time_display = $arrival_time->format('H:i');
            
            // Gérer le cas où le pointage de sortie n'est pas encore défini
            if (empty($pointage['SortieTimestamp'])) {
                $departure_time_display = "En cours";
                $work_hours_display = "En cours";
            } else {
                // Création de l'objet DateTime pour l'heure de départ
                $departure_time = DateTime::createFromFormat('U', $pointage['SortieTimestamp']);
                $departure_time->setTimezone(new DateTimeZone('Europe/Paris'));
                $departure_time_display = $departure_time->format('H:i');
                
                // Calcul des heures travaillées
                $work_seconds = $pointage['SortieTimestamp'] - $pointage['EntreeTimestamp'];
                $work_hours = $work_seconds / 3600;
                $work_hours_display = number_format($work_hours, 2);
                $total_hours += $work_hours;
            }
            
            // Ajout de la ligne dans le tableau
            $table_rows .= "<tr data-pointage-id='".$pointage['PointageID']."' data-entree-timestamp='".$pointage['EntreeTimestamp']."' data-sortie-timestamp='".$pointage['SortieTimestamp']."' onclick='afficherModal(this)'><td>{$date_display}</td><td>{$arrival_time_display}</td><td>{$departure_time_display}</td><td>$work_hours_display</td></tr>";
        }
        
        
        
        
        // Affichage du tableau des pointages
        echo "<table>";
        echo "<tr><th>Date</th><th>Heure d'arrivée</th><th>Heure de départ</th><th>Nombre d'heures</th></tr>";
        echo $table_rows;
        echo "</table>";
        
        ?>
        
        
        <font color="white">
        <?php
        // Affichage du total d'heures travaillées
        echo "Total d'heures travaillées : " . number_format($total_hours, 2);
        ?>
        </font>
        
        
        
        
        
        <br><br><br>
        <font color="white">
        <h4>Affichage sur dates sélectionnées</h4>
        </font>
        
        <br>
        <div class="d1">
        
        <div class="map">
        
        <form id="contact" method="post" action="">
        
        <font color="white"> Date de début : </font><font color="black"><input type="date" name="start_date" required></font><br><br>
        
        <font color="white">Date de fin : </font><font color="black"><input type="date" name="end_date" required></font><br><br>
        
        <font color="black"><input class="btn" id="form-submit" type="submit" value="Afficher la sélection"></font>
        
        </form>
        
        </div></div><br><br><br><br><br><br>
        
        <font color="white">
        
        
        
        <?php
        // Vérification de la soumission du formulaire
        if ($_SERVER['REQUEST_METHOD'] == 'POST' && isset($_GET['employe'])) {
            $employeId = intval($_GET['employe']);
            $start_date = $_POST['start_date'];
            $end_date = $_POST['end_date'];
            
            // Conversion des dates en format Y-m-d
            $start_date_formatted = date('Y-m-d', strtotime($start_date));
            $end_date_formatted = date('Y-m-d', strtotime($end_date . ' +1 day'));
            
            $total_hours = 0;
            $table_rows = "";
            date_default_timezone_set('Europe/Paris');
            
            // Requête pour obtenir les pointages entre les dates spécifiées
            $stmt = $pdo->prepare("SELECT PointageID, UNIX_TIMESTAMP(DateHeureEntree) AS EntreeTimestamp, UNIX_TIMESTAMP(DateHeureSortie) AS SortieTimestamp FROM z_ptg_aqp_pointages WHERE UserID = :userID AND DateHeureEntree BETWEEN :startDate AND :endDate");
            
            $stmt->execute([':userID' => $employeId, ':startDate' => $start_date_formatted, ':endDate' => $end_date_formatted]);
            $pointages = $stmt->fetchAll(PDO::FETCH_ASSOC);
            if (!$stmt->execute()) {
                print_r($stmt->errorInfo());
            }
            
            foreach ($pointages as $pointage) {
                // Création de l'objet DateTime pour l'heure d'arrivée
                $arrival_time = DateTime::createFromFormat('U', $pointage['EntreeTimestamp']);
                $arrival_time->setTimezone(new DateTimeZone('Europe/Paris'));
                $date_display = $arrival_time->format('Y-m-d');
                $arrival_time_display = $arrival_time->format('H:i');
                
                // Gérer le cas où le pointage de sortie n'est pas encore défini
                if (empty($pointage['SortieTimestamp'])) {
                    $departure_time_display = "En cours";
                    $work_hours_display = "En cours";
                } else {
                    // Création de l'objet DateTime pour l'heure de départ
                    $departure_time = DateTime::createFromFormat('U', $pointage['SortieTimestamp']);
                    $departure_time->setTimezone(new DateTimeZone('Europe/Paris'));
                    $departure_time_display = $departure_time->format('H:i');
                    
                    // Calcul des heures travaillées
                    $work_seconds = $pointage['SortieTimestamp'] - $pointage['EntreeTimestamp'];
                    $work_hours = $work_seconds / 3600;
                    $work_hours_display = number_format($work_hours, 2);
                    $total_hours += $work_hours;
                }
                
                // Ajout de la ligne dans le tableau
                $table_rows .= "<tr data-pointage-id='".$pointage['PointageID']."' data-entree-timestamp='".$pointage['EntreeTimestamp']."' data-sortie-timestamp='".$pointage['SortieTimestamp']."' onclick='afficherModal(this)'><td>{$date_display}</td><td>{$arrival_time_display}</td><td>{$departure_time_display}</td><td>$work_hours_display</td></tr>";
            }
            
            
            
            // Affichage du tableau
            echo "<table><tr><th>Date</th><th>Heure d'arrivée</th><th>Heure de départ</th><th>Nombre d'heures</th></tr>$table_rows</table>";
            echo "Total d'heures travaillées : " . number_format($total_hours, 2);
            
            
        }
        ?>
        
        
        <br><br><br>
        
        <p><a href="#" onclick="promptExitPin()">Sortir Employé</a></p>
        
        </div>
        
        
        </div></div>
        
        </div>
        
        </div>
        
        
        
        </center>
        
        
        <!-- Modal PIN (design refait) -->
        <div id="modalBackground" onclick="closePinModal()" aria-hidden="true"></div>
        <div id="pinModal" role="dialog" aria-modal="true" aria-labelledby="pinModalTitle">
          <form id="pinForm" onsubmit="verifierPin(); return false;">
            <h2 id="pinModalTitle" class="pin-modal-title">Code PIN</h2>
            <p class="pin-modal-subtitle">Saisissez votre code à 4 chiffres</p>

            <div class="pin-input-wrapper" onclick="focusPinInput()">
              <label for="pinInput" class="sr-only">Code PIN (4 chiffres)</label>
              <div class="pin-boxes" aria-hidden="true">
                <div class="pin-box" data-idx="0"></div>
                <div class="pin-box" data-idx="1"></div>
                <div class="pin-box" data-idx="2"></div>
                <div class="pin-box" data-idx="3"></div>
              </div>
              <input
                type="tel"
                id="pinInput"
                class="pin-hidden-input"
                inputmode="numeric"
                pattern="[0-9]*"
                maxlength="4"
                autocomplete="off"
                autocapitalize="off"
                autocorrect="off"
                spellcheck="false"
                title="Seulement des chiffres."
                value=""
              >
            </div>

            <div class="pin-modal-actions">
              <button type="button" class="pin-modal-btn cancel" onclick="closePinModal()">Annuler</button>
              <button type="submit" class="pin-modal-btn confirm">Valider</button>
            </div>
          </form>
        </div>
        
        
        
        <script>
        
        
        
        
        
        
        
        </script>
        
        
        
        
        
        
        <script src="<?php echo htmlspecialchars(asset_version_url('js/vendor/jquery-1.11.2.min.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
        
        <script src="<?php echo htmlspecialchars(asset_version_url('js/plugins.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
        
        <script src="<?php echo htmlspecialchars(asset_version_url('js/main.js'), ENT_QUOTES, 'UTF-8'); ?>"></script>
        
        


        <script type="text/javascript">

        var employeeFirstName = <?php echo json_encode(($employe && isset($employe['Prenom'])) ? (string)$employe['Prenom'] : '', JSON_UNESCAPED_UNICODE | JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT); ?>;
        function buildPunchToast(action) { var p = employeeFirstName ? (" " + employeeFirstName) : ""; return (action === "depart") ? ("Départ" + p + " enregistré") : ("Arrivée" + p + " enregistrée"); }

        $("#arrivalButton").click(function(e) {
            e.preventDefault();
            let employe = $(this).data('employe');
            $.ajax({
                type: "POST",
                url: "enregistrement_arrivee.php",
                data: {employe: employe},
                success: function(response) {
                    var msg = buildPunchToast("arrivee");
                    window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
                },
                error: function() {
                    if (typeof window.showToast === "function") {
                      window.showToast("Erreur lors de l'enregistrement du pointage.", "error");
                    } else {
                      alert("Erreur lors de l'enregistrement du pointage.");
                    }
                }
            });
        });
        
        
        $("#departureButton").click(function(e) {
            e.preventDefault();
            let employe = $(this).data('employe');
            $.ajax({
                type: "POST",
                url: "enregistrement_depart.php",
                data: {employe: employe},
                success: function(response) {
                    var msg = buildPunchToast("depart");
                    window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
                },
                error: function() {
                    if (typeof window.showToast === "function") {
                      window.showToast("Erreur lors de l'enregistrement du pointage.", "error");
                    } else {
                      alert("Erreur lors de l'enregistrement du pointage.");
                    }
                }
            });
        });
        
        
        var actionContext = ""; // Variable globale pour stocker le contexte de l'action

function sanitizePinValue(val) {
  return String(val || '').replace(/\D/g, '').slice(0, 4);
}

function updatePinBoxes(pin) {
  var boxes = document.querySelectorAll('#pinModal .pin-box');
  for (var i = 0; i < boxes.length; i++) {
    boxes[i].textContent = (i < pin.length) ? '★' : '';
  }
}

function focusPinInput() {
  var input = document.getElementById('pinInput');
  if (!input) return;
  try { input.focus(); } catch (e) {}
}

function resetPinModal() {
  var input = document.getElementById('pinInput');
  if (input) input.value = '';
  updatePinBoxes('');
}

function openPinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (!modal || !bg) return;
  modal.style.display = 'block';
  bg.style.display = 'block';
  resetPinModal();
  window.setTimeout(function() { focusPinInput(); }, 10);
}

function closePinModal() {
  var modal = document.getElementById('pinModal');
  var bg = document.getElementById('modalBackground');
  if (modal) modal.style.display = 'none';
  if (bg) bg.style.display = 'none';
  resetPinModal();
}

// Bind input once (digits only + sync UI boxes)
(function bindPinOnce() {
  var input = document.getElementById('pinInput');
  if (!input || input.getAttribute('data-bound') === '1') return;
  input.setAttribute('data-bound', '1');
  input.addEventListener('input', function() {
    var clean = sanitizePinValue(input.value);
    if (input.value !== clean) input.value = clean;
    updatePinBoxes(clean);
  });
  input.addEventListener('keydown', function(e) {
    var key = e.key || '';
    if (key === 'Escape') {
      closePinModal();
    }
  });
})();

function afficherModal(element) {
    var pointageId = element.getAttribute('data-pointage-id');
    window.pointageIdPourModification = pointageId;
    window.location.href = "modif_pointage.php?pointageId=" + encodeURIComponent(pointageId);
}

function verifierPin() {
    var input = document.getElementById('pinInput');
    var pin = sanitizePinValue(input ? input.value : '');
    if (input && input.value !== pin) input.value = pin;
    updatePinBoxes(pin);

    if (pin.length !== 4) {
        if (typeof window.showToast === "function") {
          window.showToast("Entrez un code PIN à 4 chiffres.", "error");
        } else {
          alert("Entrez un code PIN à 4 chiffres.");
        }
        try { focusPinInput(); } catch (e) {}
        return;
    }
    if (pin) {
        // Envoi du code PIN à un script PHP pour vérification
        var xhr = new XMLHttpRequest();
        xhr.onreadystatechange = function() {
            if (this.readyState == 4 && this.status == 200) {
                if (this.responseText == "OK") {
                    // Exécuter l'action en fonction du contexte
                    if (actionContext === "modificationPointage") {
                        window.location.href = "modif_pointage.php?pointageId=" + window.pointageIdPourModification;
                    } else if (actionContext === "sortieEmploye") {
                        sortirEmploye(window.pointageIdPourSortie);
                    }
                } else {
                    if (typeof window.showToast === "function") {
                      window.showToast("Code PIN incorrect.", "error");
                    } else {
                      alert("Code PIN incorrect.");
                    }
                    resetPinModal();
                    try { focusPinInput(); } catch (e) {}
                }
            }
        };
        xhr.open("POST", "verifier_pin.php", true);
        xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
        xhr.send("pin=" + pin);
    }
}

function promptExitPin() {
    window.pointageIdPourSortie = <?php echo $employeId; ?>;
    sortirEmploye(window.pointageIdPourSortie);
}

function sortirEmploye(employeId) {
    $.ajax({
        type: "POST",
        url: "sortir_employe.php", // Script PHP pour sortir l'employé
        data: {employe: employeId},
        success: function(response) {
            var msg = "Statut de l'employé mis à jour.";
            window.location.href = "index.php?toast=" + encodeURIComponent(msg) + "&toastType=success";
        },
        error: function() {
            if (typeof window.showToast === "function") {
              window.showToast("Erreur lors de la mise à jour du statut employé.", "error");
            } else {
              alert("Erreur lors de la mise à jour du statut employé.");
            }
        }
    });
}

            
            
            </script>
            
            
            
            </body>
            
            </html>"
            }
        }
    ],
    "meta": {
        "summary": {
            "changed": 7,
            "created": 0,
            "deleted": 0,
            "errors": 0
        },
        "patch_sha1": "439746c439bbb2a78b5d5ecd829291d83999aa9e",
        "created_files": [],
        "branching": {
            "auto_branch_created": false,
            "auto_branch_id": null,
            "source_branch_id": "main",
            "base_event_id": "event_1d594ac189671de7"
        },
        "transition_store": {
            "dir": "event-assets/event_a4c22d3ca872c3ae",
            "manifest": "event-assets/event_a4c22d3ca872c3ae/manifest.json",
            "files_count": 7
        }
    },
    "impacted_paths": [
        "pointages/admin.php",
        "pointages/css/tooplate-style.css",
        "pointages/index.php",
        "pointages/js/driver-modal.js",
        "pointages/js/main.js",
        "pointages/modif_pointage.php",
        "pointages/pointage.php"
    ]
}