This commit is contained in:
KONE SOREL 2026-03-18 09:00:31 +00:00
parent eebe13a698
commit 3efa88d108
5 changed files with 116 additions and 32 deletions

View File

@ -69,4 +69,14 @@ class ControleurAjaxprestatairesreseau extends Controleurrequete
$this->executerAction("index"); $this->executerAction("index");
} }
public function rechercher() {
$query = $this->requete->getParametreFormulaire("query");
$codeReseau = $_SESSION['codeReseau'];
$resultats = $this->reseau->rechercherPrestatairesDispo($codeReseau, $query);
// On génère une vue partielle qui ne contient que les <tr>
$this->genererVueAjax(array('resultats' => $resultats));
}
} }

View File

@ -23,13 +23,11 @@ class ControleurPrestatairesreseau extends Controleur {
$prestatairesdispo = $this->reseau->getPrestatairesReseauNonAffectes($codeReseau); $prestatairesdispo = $this->reseau->getPrestatairesReseauNonAffectes($codeReseau);
$prestatairesreseau = $this->reseau->getPrestatairesReseauAffectes($codeReseau); $prestatairesreseau = $this->reseau->getPrestatairesReseauAffectes($codeReseau);
$chemin = $this->menuvue->getChemin('Prestatairesreseau');
$this->genererVue(array( $this->genererVue(array(
'reseau' => $reseau, 'reseau' => $reseau,
'prestatairesdispo' => $prestatairesdispo, 'prestatairesdispo' => $prestatairesdispo,
'prestatairesreseau' => $prestatairesreseau, 'prestatairesreseau' => $prestatairesreseau
'chemin' => $chemin
) )
); );
} }

View File

@ -46866,7 +46866,7 @@ function afficher_prestataires_reseau()
data: donnees, data: donnees,
success: function(data) { success: function(data) {
$('#div_prestataires_reseau').html(data); $('#div_prestataires_reseau').html(data);
appliquerDataTable(); //appliquerDataTable();
}, },
error: function(data) { error: function(data) {
}, },
@ -92357,4 +92357,5 @@ function actualiser_garantie_produit() {
`); `);
} }
}); });
} }

View File

@ -373,4 +373,19 @@ class Reseausoins extends Modele
return $resultat->fetchAll(PDO::FETCH_ASSOC); return $resultat->fetchAll(PDO::FETCH_ASSOC);
} }
/**
* Recherche les prestataires non affectés avec un filtre textuel (Serveur-side)
*/
public function rechercherPrestatairesDispo($codeReseau, $query) {
$query = contruireParamLike($query);
$codeSociete = $_SESSION['codeSociete'];
$sql = "CALL sp_rechercher_prestataires_dispo(?, ?, ?)"; // Sécurité pour ne pas saturer le DOM
$resultat = $this->executerRequete($sql, array($codeSociete, $codeReseau, $query));
return $resultat->fetchAll(PDO::FETCH_ASSOC);
}
} }

View File

@ -1,7 +1,14 @@
<?php <?php
$this->titre = "INTER SANTE - Affectation Prestataires";
$codeReseau = $this->nettoyer($reseau['codeReseau']); $codeReseau = $this->nettoyer($reseau['codeReseau']);
$idReseau = $this->nettoyer($reseau['id']); $idReseau = $this->nettoyer($reseau['id']);
$nomReseau = $this->nettoyer($reseau['libelle']); $nomReseau = $this->nettoyer($reseau['libelle']);
// Extraction unique des types pour les selects (Optionnel)
$typesDispo = array_unique(array_column($prestatairesdispo, 'typeprestataire'));
sort($typesDispo);
$typesAffect = array_unique(array_column($prestatairesreseau, 'typeprestataire'));
sort($typesAffect);
?> ?>
<div class="page-content animate__animated animate__fadeIn"> <div class="page-content animate__animated animate__fadeIn">
@ -30,8 +37,8 @@
<input type="hidden" id="codeReseau" name="codeReseau" value="<?= $codeReseau ?>"> <input type="hidden" id="codeReseau" name="codeReseau" value="<?= $codeReseau ?>">
<input type="hidden" id="idReseau" name="idReseau" value="<?= $idReseau ?>"> <input type="hidden" id="idReseau" name="idReseau" value="<?= $idReseau ?>">
<div class="row g-4"> <div id="div_prestataires_reseau" class="row g-4">
<div class="col-lg-5"> <div class="col-lg-5">
<div class="card border-0 shadow-sm h-100"> <div class="card border-0 shadow-sm h-100">
<div class="card-header bg-white py-3 border-bottom"> <div class="card-header bg-white py-3 border-bottom">
@ -43,14 +50,29 @@
<?= count($prestatairesdispo) ?> <?= count($prestatairesdispo) ?>
</span> </span>
</div> </div>
<div class="input-group input-group-sm mb-2">
<span class="input-group-text bg-white border-2 border-end-0 text-muted"><i class="fas fa-search"></i></span> <div class="bg-primary-ghost-light p-2 rounded mb-2 border border-primary-subtle">
<input type="text" id="search_dispo" class="form-control border-2 border-start-0 shadow-none" placeholder="<?= _("Filtrer par nom, ville...") ?>" onkeyup="filterTable('tab_dispo', this.value, 'count_dispo')"> <div class="row g-1">
<div class="col-12 mb-1">
<input type="text" id="search_nom_dispo" class="form-control form-control-sm border-2 shadow-none" placeholder="<?= _("Nom du prestataire...") ?>" onkeyup="multiFilter('tab_dispo', 'count_dispo')">
</div>
<div class="col-6">
<input type="text" id="search_ville_dispo" class="form-control form-control-sm border-2 shadow-none" placeholder="<?= _("Ville...") ?>" onkeyup="multiFilter('tab_dispo', 'count_dispo')">
</div>
<div class="col-6">
<select id="search_type_dispo" class="form-select form-select-sm border-2 shadow-none" onchange="multiFilter('tab_dispo', 'count_dispo')">
<option value=""><?= _("Tous les types") ?></option>
<?php foreach($typesDispo as $t): ?> <option value="<?= $t ?>"><?= $t ?></option> <?php endforeach; ?>
</select>
</div>
</div>
</div> </div>
<button class="btn btn-primary-ghost btn-sm w-100 mt-2 rounded-pill fw-bold" onclick="ajouter_tous_prestataires_reseau();">
<button class="btn btn-primary-ghost btn-sm w-100 rounded-pill fw-bold" onclick="ajouter_tous_prestataires_reseau();">
<?= _("Tout ajouter") ?> <i class="fas fa-angle-double-right ms-1"></i> <?= _("Tout ajouter") ?> <i class="fas fa-angle-double-right ms-1"></i>
</button> </button>
</div> </div>
<div class="card-body p-0"> <div class="card-body p-0">
<div class="table-responsive" style="max-height: 500px; overflow-y: auto;"> <div class="table-responsive" style="max-height: 500px; overflow-y: auto;">
<table class="table table-hover align-middle mb-0" id="tab_dispo" style="font-size: 8.5pt;"> <table class="table table-hover align-middle mb-0" id="tab_dispo" style="font-size: 8.5pt;">
@ -65,8 +87,9 @@
<tr> <tr>
<td class="ps-3"> <td class="ps-3">
<div class="fw-bold text-dark search-target"><?= $this->nettoyer($v['libelle']) ?></div> <div class="fw-bold text-dark search-target"><?= $this->nettoyer($v['libelle']) ?></div>
<div class="text-muted smaller search-target"> <div class="text-muted smaller">
<?= $this->nettoyer($v['localite']) ?> | <?= $this->nettoyer($v['typeprestataire']) ?> <span class="search-target"><?= $this->nettoyer($v['localite']) ?></span> |
<span class="text-primary search-target"><?= $this->nettoyer($v['typeprestataire']) ?></span>
</div> </div>
</td> </td>
<td class="text-center pe-3"> <td class="text-center pe-3">
@ -94,14 +117,29 @@
<?= count($prestatairesreseau) ?> <?= count($prestatairesreseau) ?>
</span> </span>
</div> </div>
<div class="input-group input-group-sm mb-2">
<span class="input-group-text bg-white border-2 border-end-0 text-muted"><i class="fas fa-search"></i></span> <div class="bg-success-ghost p-2 rounded mb-2 border border-success-subtle">
<input type="text" id="search_affect" class="form-control border-2 border-start-0 shadow-none" placeholder="<?= _("Rechercher dans la liste...") ?>" onkeyup="filterTable('tab_affect', this.value, 'count_affect')"> <div class="row g-1">
<div class="col-12 mb-1">
<input type="text" id="search_nom_affect" class="form-control form-control-sm border-2 shadow-none" placeholder="<?= _("Rechercher dans la liste...") ?>" onkeyup="multiFilter('tab_affect', 'count_affect')">
</div>
<div class="col-6">
<input type="text" id="search_ville_affect" class="form-control form-control-sm border-2 shadow-none" placeholder="<?= _("Filtrer par ville...") ?>" onkeyup="multiFilter('tab_affect', 'count_affect')">
</div>
<div class="col-6">
<select id="search_type_affect" class="form-select form-select-sm border-2 shadow-none" onchange="multiFilter('tab_affect', 'count_affect')">
<option value=""><?= _("Tous les types") ?></option>
<?php foreach($typesAffect as $t): ?> <option value="<?= $t ?>"><?= $t ?></option> <?php endforeach; ?>
</select>
</div>
</div>
</div> </div>
<button class="btn btn-danger-light btn-sm w-100 mt-2 rounded-pill fw-bold" onclick="retirer_tous_prestataires_du_site();">
<i class="fas fa-angle-double-left me-1"></i> <?= _("Retirer tous") ?> <button class="btn btn-danger-light btn-sm w-100 rounded-pill fw-bold" onclick="retirer_tous_prestataires_du_site();">
<i class="fas fa-angle-double-left me-1"></i> <?= _("Retirer tous les prestataires") ?>
</button> </button>
</div> </div>
<div class="card-body p-0"> <div class="card-body p-0">
<div class="table-responsive" style="max-height: 500px; overflow-y: auto;"> <div class="table-responsive" style="max-height: 500px; overflow-y: auto;">
<table class="table table-hover align-middle mb-0" id="tab_affect" style="font-size: 8.5pt;"> <table class="table table-hover align-middle mb-0" id="tab_affect" style="font-size: 8.5pt;">
@ -109,7 +147,7 @@
<tr> <tr>
<th class="ps-3 py-2" width="50px"></th> <th class="ps-3 py-2" width="50px"></th>
<th class="py-2"><?= _("Nom du Prestataire") ?></th> <th class="py-2"><?= _("Nom du Prestataire") ?></th>
<th class="py-2 pe-3"><?= _("Ville / Localité") ?></th> <th class="py-2 pe-3"><?= _("Ville & Type") ?></th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@ -120,8 +158,11 @@
<i class="fas fa-chevron-left"></i> <i class="fas fa-chevron-left"></i>
</button> </button>
</td> </td>
<td class="fw-bold search-target"><?= $this->nettoyer($v['libelle']) ?></td> <td class="fw-bold text-dark text-uppercase search-target"><?= $this->nettoyer($v['libelle']) ?></td>
<td class="pe-3 search-target"><?= $this->nettoyer($v['localite']) ?></td> <td class="pe-3 search-target">
<span class="text-muted"><i class="fas fa-map-marker-alt me-1"></i><?= $this->nettoyer($v['localite']) ?></span>
<span class="badge bg-light text-muted fw-normal ms-2 search-target"><?= $this->nettoyer($v['typeprestataire']) ?></span>
</td>
</tr> </tr>
<?php endforeach; ?> <?php endforeach; ?>
</tbody> </tbody>
@ -135,19 +176,27 @@
<script> <script>
/** /**
* Fonction de filtrage en temps réel * Filtrage multicritères universel pour INTER-SANTE
* @param {string} tableId - ID de la table à filtrer
* @param {string} query - Texte recherché
* @param {string} counterId - ID du badge de décompte
*/ */
function filterTable(tableId, query, counterId) { function multiFilter(tableId, counterId) {
const filter = query.toLowerCase(); const suffix = tableId === 'tab_dispo' ? '_dispo' : '_affect';
const valNom = document.getElementById('search_nom' + suffix).value.toLowerCase();
const valVille = document.getElementById('search_ville' + suffix).value.toLowerCase();
const valType = document.getElementById('search_type' + suffix).value.toLowerCase();
const rows = document.querySelectorAll(`#${tableId} tbody tr`); const rows = document.querySelectorAll(`#${tableId} tbody tr`);
let visibleCount = 0; let visibleCount = 0;
rows.forEach(row => { rows.forEach(row => {
const text = row.innerText.toLowerCase(); // On récupère le texte complet de la ligne pour le filtrage
if (text.includes(filter)) { const rowText = row.innerText.toLowerCase();
const matchNom = rowText.includes(valNom);
const matchVille = rowText.includes(valVille);
const matchType = valType === "" || rowText.includes(valType);
if (matchNom && matchVille && matchType) {
row.style.display = ""; row.style.display = "";
visibleCount++; visibleCount++;
} else { } else {
@ -155,16 +204,27 @@ function filterTable(tableId, query, counterId) {
} }
}); });
// Mise à jour du compteur
document.getElementById(counterId).innerText = visibleCount; document.getElementById(counterId).innerText = visibleCount;
} }
</script> </script>
<style> <style>
.bg-primary-ghost { background-color: rgba(33, 46, 83, 0.08) !important; } .bg-primary-ghost { background-color: rgba(33, 46, 83, 0.08) !important; }
.bg-primary-ghost-light { background-color: rgba(33, 46, 83, 0.03) !important; }
.btn-primary-ghost { color: var(--bs-primary); background-color: rgba(33, 46, 83, 0.1); border: none; } .btn-primary-ghost { color: var(--bs-primary); background-color: rgba(33, 46, 83, 0.1); border: none; }
.btn-primary-ghost:hover { background-color: rgba(33, 46, 83, 0.2); }
.bg-success-light { background-color: rgba(25, 135, 84, 0.08) !important; } .bg-success-light { background-color: rgba(25, 135, 84, 0.08) !important; }
.bg-success-ghost { background-color: rgba(25, 135, 84, 0.03) !important; }
.btn-danger-light { color: var(--bs-danger); background-color: rgba(220, 53, 69, 0.1); border: none; } .btn-danger-light { color: var(--bs-danger); background-color: rgba(220, 53, 69, 0.1); border: none; }
.action-icon { width: 30px; height: 30px; display: inline-flex; align-items: center; justify-content: center; } .btn-danger-light:hover { background-color: rgba(220, 53, 69, 0.2); }
.input-group-sm > .form-control:focus { border-color: #dee2e6; }
.action-icon { width: 30px; height: 30px; padding: 0; display: inline-flex; align-items: center; justify-content: center; transition: all 0.2s; }
.action-icon:hover { transform: scale(1.15); }
.smaller { font-size: 0.85em; }
/* Scrollbar */
.table-responsive::-webkit-scrollbar { width: 6px; }
.table-responsive::-webkit-scrollbar-thumb { background: #cbd5e0; border-radius: 10px; }
</style> </style>