Rev 1356 | Blame | Last modification | View Log | RSS feed
<?php/*** Le web service observations récupère toutes les observations et, pour chacune d'elle, les* images qui lui sont associées.** @category php 5.2* @package del* @subpackage images* @author Grégoire Duché <gregoire@tela-botanica.org>* @author Aurélien Peronnet <aurelien@tela-botanica.org>* @author Raphaël Droz <raphael@tela-botanica.org>* @copyright Copyright (c) 2012, 2013 Tela Botanica (accueil@tela-botanica.org)* @license http://www.cecill.info/licences/Licence_CeCILL_V2-fr.txt Licence CECILL* @license http://www.gnu.org/licenses/gpl.html Licence GNU-GPL* @see http://www.tela-botanica.org/wikini/eflore/wakka.php?wiki=ApiIdentiplante01Observations*/class ListeObservations {private $conteneur;private $navigation;private $masque;private $gestionBdd;private $bdd;private $parametres = array();private $ressources = array();private $tri = 'date_transmission';private $directionTri = 'desc';static $tris_possibles = array('date_observation');public function __construct(Conteneur $conteneur = null) {$this->conteneur = $conteneur == null ? new Conteneur() : $conteneur;$this->conteneur->chargerConfiguration('config_departements_bruts.ini');$this->conteneur->chargerConfiguration('config_observations.ini');$this->conteneur->chargerConfiguration('config_mapping_votes.ini');$this->conteneur->chargerConfiguration('config_mapping_commentaires.ini');$this->navigation = $conteneur->getNavigation();$this->masque = $conteneur->getMasque();$this->gestionBdd = $conteneur->getGestionBdd();$this->bdd = $this->gestionBdd->getBdd();}/*** RequeteSansParametres** permet de vérifier qu'il n'y a aucun paramètre dans la requete, excepté les informations de start et limite,* pour ajuster la requête pour un gain de temps* @param array $ressources les ressources telles qu'elles sont passées au script* @param array $parametres les paramètres tels qu'il sont passés au script* */public function requeteSansParametres($ressources, $parametres) {$estSansParametres = true;$parametresDeTri = array('navigation.depart', 'navigation.limite', 'tri', 'ordre');if (sizeof($ressources) > 0) {$estSansParametres = false;}if (sizeof($parametres) > 0) {foreach ($parametres as $cle => $valeur) {if (!in_array($cle, $parametresDeTri)) {$estSansParametres = false;break;}}}return $estSansParametres;}/*** Méthode principale de la classe.* Lance la récupération des images dans la base et les place dans un objet ResultatService* pour l'afficher.* @param array $ressources les ressources situées après l'url de base (ex : http://url/ressource1/ressource2)* @param array $parametres les paramètres situés après le ? dans l'url* */public function consulter($ressources, $parametres) {$this->initialiserRessourcesEtParametres($ressources, $parametres);$this->configurer();$this->verifierConfiguration();$this->verifierParametresTri();$this->initialiserTri();$resultat = new ResultatService();if ($this->requeteSansParametres($ressources, $parametres)) {$liaisons = $this->chargerLiaisonsSimple($total);$observations = $this->chargerObservations($liaisons);$this->navigation->setTotal($total);// modifie $observations$this->chargerImages($observations);// modifie $observations$this->chargerDeterminations($observations);//Compute$resultats = array();foreach ($observations as $id => $observation) {$idObs = $observation['id_observation'];$resultats['"'.$idObs.'"'] = $observation;}// Mettre en forme le résultat et l'envoyer pour affichage$resultat->corps = array('entete' => $this->conteneur->getEntete(), 'resultats' => $resultats);} else {$liaisons = $this->chargerLiaisons($total);$this->navigation->setTotal($total);$observations = $this->chargerObservations($liaisons);// modifie $observations$this->chargerImages($observations);// modifie $observations$this->chargerDeterminations($observations);// Mettre en forme le résultat et l'envoyer pour affichage$resultat->corps = array('entete' => $this->conteneur->getEntete(), 'resultats' => $observations);}return $resultat;}private function initialiserRessourcesEtParametres($ressources, $parametres) {$this->ressources = $ressources;$this->parametres = $parametres;}private function verifierParametresTri() {$erreurs = array();if(isset($this->parametres['tri']) && !in_array($this->parametres['tri'], self::$tris_possibles)) {$erreurs[] = '- le type de tri demandé est incorrect, les valeurs possibles sont '.$self::tris_possibles.' ;';}$directions_tri = array('asc', 'desc');if(isset($this->parametres['ordre']) && !in_array($this->parametres['ordre'], $directions_tri)) {$erreurs[] = '- la direction du tri demandé est incorrecte, les valeurs supportées sont asc ou desc ;';}if (!empty($erreurs)) {$e = 'Erreur lors de l\'analyse des parametres du tri : '."\n";$e .= implode("\n", $erreurs);throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);}}private function initialiserTri() {$this->tri = isset($this->parametres['tri']) ? $this->parametres['tri'] : $this->tri;$this->directionTri = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : $this->directionTri;}/*-------------------------------------------------------------------------------CONFIGURATION DU SERVICE--------------------------------------------------------------------------------*//*** Configuration du service en fonction du fichier de config config_del.ini* */private function configurer() {$this->mappingFiltre = $this->conteneur->getParametre('mapping_masque');$this->mappingObservation = $this->conteneur->getParametre('mapping_observation');$this->mappingVotes = $this->conteneur->getParametre('mapping_votes');$this->mappingCommentaire = $this->conteneur->getParametre('mapping_commentaire');}/*** Vérifier que le service est bien configuré* */private function verifierConfiguration() {$erreurs = array();$tableauObservations = $this->conteneur->getParametre('observations');if (empty($tableauObservations)) {$erreurs[] = '- le fichier de configuration ne contient pas le tableau [images] ou celui-ci est vide ;';} else {if ($this->conteneur->getParametre('url_service') == null) {$erreurs[] = '- paramètre "url_service" manquant ;';}if ($this->conteneur->getParametre('url_images') == null) {$erreurs[] = '- paramètre "url_images" manquant ;';}}if (empty($this->mappingObservation)) {$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_observation] ou celui-ci est vide ;';} else {$champsMappingObs = array('id_observation', 'date_observation', 'date_transmission', 'famille','nom_sel', 'nom_sel_nn', 'nt', 'ce_zone_geo', 'zone_geo', 'lieudit', 'station', 'courriel','ce_utilisateur', 'nom', 'prenom');foreach ($champsMappingObs as $champ) {if (!isset($this->mappingObservation[$champ])) {$erreurs[] = '- le mapping du champ "'.$champ.'" pour l\'observation est manquant ;';}}}if (empty($this->mappingCommentaire)) {$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_commentaire] ou celui-ci est vide ;';} else {$champsMappingCom = array('id_commentaire', 'texte', 'ce_utilisateur', 'utilisateur_nom','utilisateur_prenom', 'utilisateur_courriel', 'date');foreach ($champsMappingCom as $champ) {if (!isset($this->mappingCommentaire[$champ])) {$erreurs[] = '- le mapping du champ "'.$champ.'" pour le commentaire est manquant ;';}}}if (empty($this->mappingFiltre)) {$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_masque] ou celui-ci est vide ;';} else {$champsMappingFiltre = array('famille', 'ns', 'nn', 'date', 'tag', 'commune');foreach ($champsMappingFiltre as $champ) {if (!isset($this->mappingFiltre[$champ])) {$erreurs[] = '- le mapping du champ "'.$champ.'" pour l\'observation est manquant ;';}}}if (!empty($erreurs)) {$e = 'Erreur lors de la configuration : '."\n";$e .= implode("\n", $erreurs);throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);}}/*** Obtenir une chaine de caractère concaténant nom et prénom séparé par une virgule* @param String $auteurId l'identifiant de l'auteur* @return String la chaine de concaténation* */private function getChaineNomPrenom($auteurId) {$nomPrenom = explode(' ', $auteurId);$nomPrenom = $this->proteger($nomPrenom);$chaineNomPrenom = implode(', ', $nomPrenom);return $chaineNomPrenom;}/*** Charger la clause WHERE en fonction des paramètres de masque* */private function chargerClauseWhere() {$where = array();$tableauMasque = $this->masque->getMasque();if (!empty($tableauMasque)) {foreach ($tableauMasque as $idMasque => $valeurMasque) {$idMasque = str_replace('masque.', '', $idMasque);switch ($idMasque) {// nom du masque => nom BDDcase 'auteur' :$whereAuteur = ' '.$this->creerFiltreAuteur($this->masque->getMasque('auteur'));if ($whereAuteur != '') {$where[] = $whereAuteur;}break;case 'date' :$whereDate = ' '.$this->creerFiltreDate($valeurMasque);if ($whereDate != '') {$where[] = $whereDate;}break;case 'departement' :$where[] = ' '.$this->creerFiltreIdZoneGeo($valeurMasque);break;case 'genre' :$where[] = ' '.$this->mappingFiltre['ns'].' LIKE '.$this->proteger('%'.$valeurMasque.'% %');break;case 'tag' :$where[] = ' '.$this->creerFiltreMotsCles($valeurMasque);break;case 'ns' :$where[] = ' nom_sel LIKE '.$this->proteger($valeurMasque.'%');break;case 'commune' :$where[] = ' '.$this->mappingFiltre[$idMasque].' LIKE '.$this->proteger(str_replace(array('-',' '), '_', $valeurMasque).'%');break;case 'masque' :$where[] = ' '.$this->creerFiltreMasqueGeneral($valeurMasque);break;case 'type' ://TODO : en discussion doit faire ressortir les observations qui ont plusieurs propositions de nom OU celle qui ont plus de N commentaires$tableauTypes = explode(';', $valeurMasque);$requeteType = array();if (in_array('adeterminer', $tableauTypes)) {//On récupère toutes les observations qui on le tag "aDeterminer" ou qui n'ont pas de nom d'espèce$requeteType[] = '(dob.mots_cles_texte LIKE '.$this->proteger("%".$this->conteneur->getParametre('tag_adeterminer')."%").' OR nom_sel_nn IS NULL '.' OR dob.certitude = "aDeterminer" '.') ';}if (in_array('aconfirmer', $tableauTypes)) {//On récupère toutes les observations qui ne sont pas "aDeterminer" et qui ont un nom d'espèce$requeteType[] = '(nom_sel IS NOT NULL AND '.'dob.mots_cles_texte NOT LIKE '.$this->proteger("%".$this->conteneur->getParametre('tag_adeterminer')."%").' AND dob.certitude != "aDeterminer" '.') ';}if (in_array('endiscussion', $tableauTypes)) {//Si on veut les observations en discussion, on va récupérer les ids des observations dont le nombre de commentaire est supérieur à N$idsObservations = $this->chargerIdsObservationsParNbCommentaires();if(!empty($idsObservations)) {$requeteType[] = 'dob.id_observation IN ('.$idsObservations.') ';} else {$requeteType[] = 'dob.id_observation IN (NULL) ';}}if ($valeurMasque != '') {$where[] = implode(' OR ', $requeteType);}break;default:$where[] = ' '.$this->mappingFiltre[$idMasque].' = '.$this->proteger($valeurMasque);break;}}}if (!empty($where)) {return ' WHERE '.implode('AND', $where);} else {return;}}public function chargerIdsObservationsParNbCommentaires() {$requeteIdsObservations = 'SELECT ce_observation '.'FROM del_commentaire '.'GROUP BY ce_observation '.'HAVING count(id_commentaire) > '.$this->conteneur->getParametre('nb_commentaires_discussion');$listeIdsObservations = $this->bdd->recupererTous($requeteIdsObservations);$idsObservations = '';foreach ($listeIdsObservations as $idObservation) {$idsObservations .= $idObservation['ce_observation'].',';}$idsObservations = rtrim($idsObservations, ',');return $idsObservations;}private function creerFiltreMasqueGeneral($valeurMasque) {//TODO: affecter d'aborder les variables, puis les tester pour les// ajouter à la chaine$whereAuteur = $this->creerFiltreAuteur($valeurMasque);$whereIdZoneGeo = $this->creerFiltreIdZoneGeo($valeurMasque);$masqueGeneral = '( '.(($whereAuteur != '') ? $whereAuteur.' OR ' : '' ).(($whereIdZoneGeo != '') ? $whereIdZoneGeo.' OR ' : '' ).'zone_geo LIKE '.$this->proteger($this->remplacerParJokerCaractere($valeurMasque).'%').' OR '.$this->creerFiltreMotsCles($valeurMasque).' OR '.'nom_sel LIKE '.$this->proteger($valeurMasque.'%').' OR '.'famille LIKE '.$this->proteger($valeurMasque.'%').' OR '.'milieu LIKE '.$this->proteger($valeurMasque).' OR '.$this->mappingFiltre['ns'].' LIKE '.$this->proteger('%'.$valeurMasque.'% %').' OR '.$this->creerFiltreDate($valeurMasque).') ';return $masqueGeneral;}private function creerFiltreAuteur($valeurMasque) {$masque = '';$auteurId = $valeurMasque;if (is_numeric($auteurId)) {$masque = ' ce_utilisateur = '.$auteurId;} else {if (strpos($auteurId, '@') === false) {$tableauNomPrenom = explode(' ',$auteurId, 2);if(count($tableauNomPrenom) == 2) {// on teste potentiellement un nom prenom ou bien un prénom nom$masque = '('.'(nom LIKE '.$this->proteger($tableauNomPrenom[0].'%').' AND '.'prenom LIKE '.$this->proteger($tableauNomPrenom[1].'%').') OR '.'(nom LIKE '.$this->proteger($tableauNomPrenom[1].'%').' AND '.'prenom LIKE '.$this->proteger($tableauNomPrenom[0].'%').') OR '.'(dob.nom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[0].'%').' AND '.'dob.prenom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[1].'%').') OR '.'(dob.nom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[1].'%').' AND '.'dob.prenom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[0].'%').') '.')';} else {$masque = '((nom LIKE '.$this->proteger($auteurId.'%').' OR '.'prenom LIKE '.$this->proteger($auteurId.'%').' OR '.'dob.nom_utilisateur LIKE '.$this->proteger($auteurId.'%').' OR '.'dob.prenom_utilisateur LIKE '.$this->proteger($auteurId.'%').')'.')';}} else {$masque = ' courriel LIKE '.$this->proteger($valeurMasque.'%').' OR dob.courriel_utilisateur LIKE '.$this->proteger($valeurMasque.'%').' ';}}return $masque;}private function remplacerParJokerCaractere($valeurMasque) {return str_replace(array('-',' '), '_', $valeurMasque);}//TODO: déplacer les fonctions ci dessus et dessous dans une classe// utilitairefunction supprimerAccents($str, $charset='utf-8') {$str = htmlentities($str, ENT_NOQUOTES, $charset);$str = preg_replace('#&([A-za-z])(?:acute|cedil|circ|grave|orn|ring|slash|th|tilde|uml);#', '\1', $str);$str = preg_replace('#&([A-za-z]{2})(?:lig);#', '\1', $str); // pour les ligatures e.g. 'œ'$str = preg_replace('#&[^;]+;#', '', $str); // supprime les autres caractèresreturn $str;}private function obtenirIdDepartement($nomDpt) {$nomDpt = $this->supprimerAccents($nomDpt);$nomDpt = strtolower(str_replace(' ','-',$nomDpt));$idDpt = $this->conteneur->getParametre($nomDpt);if ($idDpt == null || $idDpt == ' ') {$idDpt = ' ';}return $idDpt;}private function creerFiltreIdZoneGeo($valeurMasque) {$masque = '';$dept = $valeurMasque;if (is_numeric($dept)) {$dept = sprintf('%02s', $dept);$dept = sprintf("%-'_5s", $dept);$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$dept);} else {$deptId = $this->conteneur->getParametre($dept);if ($deptId != null) {$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$deptId.'%');} else {$id = $this->obtenirIdDepartement($valeurMasque);$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$id.'%');}}return $masque;}private function creerFiltreDate($valeurMasque) {//TODO: définir dans le fichier de config un tableau contenant plusieurs format de date// autorisés pour la recherche, qui seraient ajoutés au OR$masque = '(';$masque .= (is_numeric($valeurMasque)) ? ' YEAR(date_observation) = '.$this->proteger($valeurMasque).' OR ' : '';$masque .= " DATE_FORMAT(date_observation, '%d/%m/%Y') = ".$this->proteger($valeurMasque).' )';return $masque;}private function creerFiltreMotsCles($valeurMasque) {$mots_cles = explode(' ', $valeurMasque);$requeteMotsClesImg = '';$requeteMotsClesObs = '';//TODO voir s'il existe un moyen plus simple que le foreach + rtrim// comme avec implode (attention au fait que l'on concatène des % au début et à la fin)foreach($mots_cles as $mot_cle) {$requeteMotsCles = $this->proteger('%'.$mot_cle.'%');$requeteMotsClesImg .= "di.mots_cles_texte LIKE $requeteMotsCles AND ";$requeteMotsClesObs .= "dob.mots_cles_texte LIKE $requeteMotsCles AND ";}$requeteMotsClesImg = rtrim($requeteMotsClesImg, ' AND ');$requeteMotsClesObs = rtrim($requeteMotsClesObs, ' AND ');$masque = "( ($requeteMotsClesImg) OR ($requeteMotsClesObs) ) ";return $masque;}/*-------------------------------------------------------------------------------CHARGEMENT DES OBSERVATIONS--------------------------------------------------------------------------------*//*** Chargement depuis la bdd de toutes les liaisons entre images et observations** Cette triple jointure pose divers problèmes d'efficacité.* - le SQL_CALC_FOUND_ROWS en premier lieu est (était) inefficace* (cf http://www.mysqlperformanceblog.com/2007/08/28/to-sql_calc_found_rows-or-not-to-sql_calc_found_rows/)* - les INDEX ne sont peut-être pas non-plus bien réglés (certains manquent cruellement, notamment* pour les champs présents dans le ORDER BY)* - enfin id_utilisateur à un typage très aléatoire selon les tables/vues d'origine*/private function chargerLiaisons(&$count) {$where = $this->chargerClauseWhere();// on effectue d'abord la sélection des observations et leur comptage// sur une table simple et restreinte (en fait, une vue) sans jointures ...$r = sprintf('SELECT id_observation FROM del_observation %s ORDER BY %s %s %s',$where,$this->tri,$this->directionTri,$this->gestionBdd->getLimitSql());$idObs = array_values(array_map(create_function('$a', 'return $a["id_observation"];'),$this->bdd->recupererTous($r)));$c = $this->bdd->recupererTous(sprintf('SELECT count(1) AS c FROM del_observation AS dob %s', $where));$count = $c[0]['c'];// puis seulement après l'obtention des données complètes, mais cette fois-ci avec// l'ensemble prédéterminé d'id d'observations obtenus ci-dessus$requeteLiaisons = sprintf('SELECT '.' dob.id_observation, nom_sel, famille, ce_zone_geo, zone_geo, lieudit, '.' station, milieu, date_observation, dob.mots_cles_texte, date_transmission, di.id_image, '.' dob.ce_utilisateur, prenom, nom, courriel, '.' dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, '.' nom_original, dob.commentaire AS dob_commentaire, '.' dob.nt, dob.nom_sel_nn '.'FROM del_observation AS dob '.' LEFT JOIN del_utilisateur AS du '.' ON du.id_utilisateur = dob.ce_utilisateur '.' LEFT JOIN del_obs_image AS doi '.' ON doi.id_observation = dob.id_observation '.' LEFT JOIN del_image AS di '.' ON di.id_image = doi.id_image '.' WHERE dob.id_observation IN (%s) -- %s',implode(',',$idObs),__FILE__ . ':' . __LINE__);return $this->bdd->recupererTous($requeteLiaisons);}private function chargerLiaisonsSimple(&$count) {$r = sprintf('SELECT id_observation FROM del_observation ORDER BY %s %s %s',$this->tri,$this->directionTri,$this->gestionBdd->getLimitSql());$idObs = array_values(array_map(create_function('$a', 'return $a["id_observation"];'),$this->bdd->recupererTous($r)));$c = $this->bdd->recupererTous('SELECT count(1) AS c FROM del_observation');$count = $c[0]['c'];$requeteLiaisons = sprintf('SELECT '.' dob.id_observation, nom_sel, famille, ce_zone_geo, zone_geo, lieudit, '.' station, milieu, date_observation, dob.mots_cles_texte, date_transmission, '.' dob.ce_utilisateur, prenom, nom, courriel, '.' dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, '.' dob.commentaire as dob_commentaire, '.' dob.nt, dob.nom_sel_nn '.'FROM del_observation AS dob '.' LEFT JOIN del_utilisateur AS du ON dob.ce_utilisateur = du.id_utilisateur '.' WHERE dob.id_observation IN (%s) -- %s',implode(',',$idObs),__FILE__ . ':' . __LINE__);return $this->bdd->recupererTous($requeteLiaisons);}/*** Compter le nombre total d'images dans la base pour affichage dans entete.* */private function compterObservations() {$requete = 'SELECT FOUND_ROWS() AS nbre ';$resultats = $this->bdd->recuperer($requete);return (int) $resultats['nbre'];}/*** Retourner un tableau d'images formaté en fonction des liaisons trouvées* @param $liaisons les liaisons de la table del_obs_images* */private function chargerObservations($liaisons) {$observations = array();foreach ($liaisons as $liaison) {$idObs = $liaison[$this->mappingObservation['id_observation']];if($liaison['ce_utilisateur'] == 0) {$liaison['prenom'] = $liaison['prenom_utilisateur'];$liaison['nom'] = $liaison['nom_utilisateur'];}$observation = $this->formaterObservation($liaison);// attention, il est important que les index du tableau soient des chaines// de caractères pour que l'ordre d'insertion soit respecté lors de la lecture// du json par les navigateur (voir bug du moteur javascript v8 #164)$observations['"'.$idObs.'"'] = $observation;}return $observations;}/*** Sélectionner toutes les images de chaque observation* @param array $observations la liste des observations* */private function chargerImages(&$observations) {$idObs = array_values(array_map(create_function('$a', 'return $a["id_observation"];'),$observations));$r = sprintf('SELECT id_observation, doi.id_image, date_prise_de_vue AS date, hauteur , largeur ' .' FROM del_obs_image AS doi ' .' INNER JOIN del_image AS di ON doi.id_image = di.id_image ' .' WHERE doi.id_observation IN (%s) -- %s',implode(',',$idObs),__FILE__ . ':' . __LINE__);$images = $this->bdd->recupererTous($r);if(!$images) return;foreach ($images as $image) {$idObs = $image['id_observation'];// rapide formatage de l'image puisque le SELECT à déjà fait le gros du travailunset($image['id_observation']);$image['binaire.href'] = $this->formaterLienImage($image['id_image']);$observations['"' . $idObs . '"']['images'][] = $image;}}/*** Récupérer toutes les déterminations et le nombre de commentaire au total* @param array $observations la liste des observations à mettre à jour* */private function chargerDeterminations(&$observations) {$idObs = array_values(array_map(create_function('$a', 'return $a["id_observation"];'),$observations));$r = sprintf('SELECT * FROM del_commentaire AS dc WHERE dc.nom_sel IS NOT NULL AND ce_observation IN (%s) -- %s',implode(',',$idObs),__FILE__ . ':' . __LINE__);$propositions = $this->bdd->recupererTous($r);if(!$propositions) return;foreach ($propositions as $proposition) {$idObs = $proposition['ce_observation'];$observations['"' . $idObs . '"']['commentaires'][] = $this->formaterDeterminations($proposition);}}/*** Charger les votes sur les déterminations* @param Array $observations le tableau des observations à mettre à jour* */private function chargerVotes($ids_proposition, &$propositions) {$resultatsVotes = $this->bdd->recupererTous(sprintf('SELECT * FROM del_commentaire_vote WHERE ce_proposition IN (%s)',implode(',', $ids_proposition)));foreach ($resultatsVotes as $vote) {$propositions[$vote['ce_proposition']]['votes'][$vote['id_vote']] = $this->formaterVote($vote);}}/*** Charger le nombre de commentaires (sans détermination) associé à l'observation* @param Array $observations le tableau des observatins à mettre à jour* */private function chargerNombreCommentaire($ids_proposition, &$propositions) {$listeCommentaires = $this->bdd->recupererTous(sprintf('SELECT ce_commentaire_parent, ce_proposition, COUNT( id_commentaire ) AS nb '.'FROM del_commentaire WHERE ce_proposition IN (%s) GROUP BY ce_proposition',implode(',', $ids_proposition)));foreach ($listeCommentaires as $ligneProposition) {// ce test sert à exclure les proposition de 1er niveau qui sont elles aussi des commentairesif($ligneProposition['ce_commentaire_parent'] != null && $ligneProposition['ce_commentaire_parent'] != 0) {$propositions[$ligneProposition['ce_commentaire_parent']]['nb_commentaires'] = $ligneProposition['nb'];} else {$propositions[$ligneProposition['ce_proposition']]['observation']['nb_commentaires'] = $ligneProposition['nb'];}}}/*-------------------------------------------------------------------------------FORMATER ET METTRE EN FORME--------------------------------------------------------------------------------*//*** Formater une observation depuis une ligne liaison* @param $liaison liaison issue de la recherche* @return $observation l'observation mise en forme* */private function formaterObservation($liaison) {$observation = array();foreach ($this->mappingObservation as $nomOriginal => $nomFinal) {$observation[$nomFinal] = $liaison[$nomOriginal];}$observation['images'] = array();return $observation;}/*** Formater le lien de l'image en fonction du fichier de config et de l'identifiant de l'image* */private function formaterLienImage($idImage) {$idImage = sprintf('%09s', $idImage);$url = $this->conteneur->getParametre('url_images');$urlImage = str_replace('%s', $idImage, $url);return $urlImage;}/**** Formate une liste de proposition en fonction du fichier de configuration*/private function formaterDeterminations($listePropositions) {if(!$listePropositions) return array();$propositions_format = array();foreach ($listePropositions as $id => $proposition) {$ids_proposition[] = $proposition['id_commentaire'];$proposition_formatee = array('nb_commentaires' => '0');foreach ($this->mappingCommentaire as $nomOriginal => $nomFinal) {if (isset($proposition[$nomOriginal])) {$proposition_formatee[$nomFinal] = $proposition[$nomOriginal];}}$propositions_format[$proposition['id_commentaire']] = $proposition_formatee;}// modifie $propositions_format$this->chargerVotes($ids_proposition, $propositions_format);// modifie $propositions_format$this->chargerNombreCommentaire($ids_proposition, $propositions_format);return $propositions_format;}/*** Formater un vote en fonction du fichier de configuration config_votes.ini* @param $votes array()* */private function formaterVote($vote) {$retour = array();foreach ($vote as $param=>$valeur) {$retour[$this->mappingVotes[$param]] = $valeur;}return $retour;}private function proteger($valeur) {if (is_array($valeur)) {return $this->bdd->protegerTableau($valeur);} else {return $this->bdd->proteger($valeur);}}}?>