Subversion Repositories eFlore/Applications.eflore-consultation

Rev

Rev 792 | Go to most recent revision | Blame | Last modification | View Log | RSS feed

<?php
class DeterminationFormateur implements Formateur {

        const TPL_VUE = 'determination';

        private $parametres = null;
        private $surligneur = null;
        private $trieur = null;
        private $urls = null;
        private $fusioneur = null;
        private $manipulateurDeChaine = null;
        private $imagesService = null;

        private $motsASurligner = array();
        private $noms = array();
        private $infosPourTpl = array();
        

        public function __construct(ParametresResultats $parametres, Array $resultats,
                Surligneur $surligneur = null, Trieur $trieur = null, AppUrls $urls = null,
                ChaineManipulateur $manipulateurDeChaine = null, Images $imagesService = null) {

                $this->parametres = $parametres;
                $this->noms = $resultats['resultat'];
                $this->surligneur = (is_null($surligneur)) ? new Surligneur() : $surligneur;
                $this->trieur = (is_null($trieur)) ? new Trieur() : $trieur;
                $this->urls = (is_null($urls)) ? new AppUrls() : $urls;
                $this->manipulateurDeChaine = is_null($manipulateurDeChaine) ? new ChaineManipulateur() : $manipulateurDeChaine;
                $this->imagesService = is_null($imagesService) ? new Images($this->parametres->projetImg) : $imagesService;
        }

        public function getTplInfos() {
                return $this->infosPourTpl;
        }

        public function getTplNom() {
                return self::TPL_VUE;
        }

        public function formater() {
                $this->obtenirUrlsImagesCoste();
                $this->obtenirUrlsPhotos();
                $this->chargerRepartition();
                $this->extraireInfosNomsPourTplDetermination();
                        
        }

        private function obtenirUrlsPhotos() {
                $this->imagesService->setProjet('cel');
                $ids = $this->extraireIdDesNoms();
                $urls = $this->imagesService->getUrlsImagesParIdsNoms($ids);
                $this->infosPourTpl['imagesUrls'] = $this->supprimerCodeReftaxDesIds($urls);
        }
        
        private function obtenirUrlsImagesCoste() {
                $this->extraireInfosTaxons();
                $this->imagesService->setProjet('coste');
                $tax = implode(',', $this->infosPourTpl['taxons']);
                $this->imagesService->setNnTaxon($tax);
                $costeImg = $this->imagesService->getInfosImagesTaxons();
                if (!empty($costeImg)) {
                        foreach ($costeImg as  $infos) {
                                $num_taxon = $infos['num_taxonomique'];
                                $images[$num_taxon][] = $infos['binaire.href'];
                                $this->infosPourTpl['imagesCoste'] = $images;
                        }
                }
        }

        private function extraireInfosTaxons() {
                foreach ($this->noms as $id => $nom ) {
                        if (array_key_exists('num_taxonomique', $nom)) {
                                $this->infosPourTpl['taxons'][$id] = $nom['num_taxonomique'];
                        }
                }
        }
        
        // TODO : utiliser le conteneur pour charger tous les objets de cette classe
        private function chargerRepartition() {
                $conteneur = new Conteneur();
                $cartesWs = $conteneur->getApiCartes();
                $cartesWs->setProjet('chorodep');
                $cartesWs->setLargeur('108x101');
                $urls = array();
                foreach ($this->noms as $id => $nom) {
                        if (array_key_exists('nom_retenu.id', $nom)) {
                                $id = $nom['nom_retenu.id'];
                                $cartesWs->setId("nn:$id");
                                if (array_key_exists($id, $urls) == false) {
                                        $urls[$id] = $cartesWs->getUrlPng();
                                }
                        }
                }
                $this->infosPourTpl['repartition']['urls'] = $urls;
        }



        
        private function extraireIdDesNoms() {
                $ids = array();
                foreach ($this->noms as $id => $nom) {
                        $idAAjouter = $id;
                        if (is_numeric($idAAjouter)) {
                                $ids[] = $idAAjouter;
                        }
                        if (array_key_exists('nom_retenu.id', $nom)) {
                                if (in_array($nom['nom_retenu.id'], $ids) == false) {
                                        $idAAjouter = $nom['nom_retenu.id'];
                                        if (is_numeric($idAAjouter)) {
                                                $ids[] = $idAAjouter;
                                        }
                                }
                        }
                }
                return $ids;
        }

        private function supprimerCodeReftaxDesIds($urls) {
                $urlsNettoyees = array();
                foreach ($urls as $id => $url) {
                        $id = $this->supprimerCodeReftax($id);
                        $urlsNettoyees[$id] = $url;
                }
                return $urlsNettoyees;
        }

        private function supprimerCodeReftax($chaine) {
                $codeReftax = $this->parametres->reftaxCourant.'.';
                $chaine = str_replace($codeReftax, '', $chaine);
                return $chaine;
        }

        /**
         * division ordonnée par  
         * 1 - noms retenus qui commencent par la requete  
         * 2 - requete contenue dans un synonyme dont le nom retenu ne contient pas la requete
         * 3 - requete contenue dans un hybride  retenu
         * 4 - requete contenue dans un nom retenu mais pas au début
         * 5 - requete contenue dans un nom sans correspondance
         * 
         */
        private function diviserResultats() {
                $tri = array();
                $sansCorres = array();
                $synonymes = array();
                $retenus = array();
                foreach ($this->noms as $cle => $valeurs) {
                        if ($valeurs['nom_retenu.libelle'] == null) {//sans correspondances
                                $sansCorres[$cle] = $valeurs['nom_sci'];
                        } elseif ($valeurs['retenu'] == 'true') { // retenus
                                if (preg_match('/ x |x /',$valeurs['nom_sci'] )  ) {
                                        //hybrides
                                        $retenus[2][$cle] = $this->retournerInfosNomRetenu($cle,$valeurs['nom_sci'], $valeurs['rang.libelle']);
                                } elseif (preg_match('/^'.strtolower($this->parametres->masqueRecherche).' |^'.strtolower($this->parametres->masqueRecherche).'$/', strtolower($valeurs['nom_sci']) ) ) {
                                        //commence par
                                        $retenus[0][$cle] = $this->retournerInfosNomRetenu($cle,$valeurs['nom_sci'], $valeurs['rang.libelle']);
                                } else {
                                        //contient
                                        $retenus[3][$cle] = $this->retournerInfosNomRetenu($cle,$valeurs['nom_sci'], $valeurs['rang.libelle']);
                                }
                        } else {//synonymes
                                $idNomRetenu = $valeurs['nom_retenu.id'];
                                if (preg_match('/'.strtolower($this->parametres->masqueRecherche).'/', strtolower($valeurs['nom_retenu.libelle'])) == 0) {
                                        //synonymes dont le nom retenu ne contient pas la requête
                                        $retenus[1][$idNomRetenu] = $this->retournerInfosNomRetenu($cle,$valeurs['nom_retenu.libelle'], $valeurs['rang.libelle']);
                                        $this->infosPourTpl['taxons'][$idNomRetenu] = $valeurs['num_taxonomique']; //num taxon pour images coste
                                }
                                $synonymes[$idNomRetenu][] = $this->retournerInfosSynonyme($cle, $valeurs);
                        }
                }
                ksort($retenus);
                $tri['retenus'] = $retenus;
                $tri['synonymes'] = $synonymes;
                $tri['sansCorres'] = $sansCorres;
                return $tri;
        }
        
        private  function retournerInfosNomRetenu($cle, $nom_sci, $rang) {
                $infos = array();
                $infos['nomSciRetenu'] = $nom_sci;
                $infos['urlFiche'] = $this->urls->obtenirUrlFiche($cle, $this->parametres->typeNom,  strtolower($this->parametres->masqueRecherche), $nom_sci);
                $infos['rang'] = $rang;
                return $infos;
        }       
        
        
        private  function retournerInfosSynonyme($cle, $valeurs) {
                $infos = array();
                $infos['nn'] = $cle;
                $infos['nomSci'] = $valeurs['nom_sci'];
                $infos['urlFiche'] = $this->urls->obtenirUrlFiche($cle, $this->parametres->typeNom, $this->parametres->masqueRecherche, $valeurs['nom_retenu.libelle']);
                return $infos;
        }
        
        private function classerAlphabetiquement($champs, $valeurs) {
                $this->trieur->setTableau($valeurs);
                $this->trieur->setChampsEtOrdres(array($champs => SORT_NATURAL));
                return $this->trieur->trier();
        }
        
        
        private function extraireInfosNomsPourTplDetermination() {      
                $tri = $this->diviserResultats();
                $this->infosPourTpl['nomsSansCorrespondance'] = isset($tri['sansCorres']) ? $tri['sansCorres'] : false;
                $this->infosPourTpl['noms'] = isset($tri['retenus'])  ? $tri['retenus'] : false;
                $this->infosPourTpl['synonymes'] = isset($tri['synonymes'] ) ?  $tri['synonymes']  : false;
        
        }
        
        private function donnerNombreResultatRetenus() {
                $nbre = 0;
                foreach ($this->infosPourTpl['noms'] as $lignes) {
                        $nbre += count($lignes);
                        if ($nbre >= 3 ) {
                                break;
                        }
                }
                return $nbre;
        }

        public function trier() {               
                if ($this->donnerNombreResultatRetenus() <= 3) { // classement par score
                        $nomsRetenus = array();
                        foreach ($this->infosPourTpl['noms'] as $categorie) { //ote la division des retenus
                                $nomsRetenus += $categorie;
                        }
                        $this->ajouterAuxNomsScoreSimilariteAvec($this->parametres->masqueRecherche);
                        $this->trieur->setTableau($nomsRetenus);
                        $this->trieur->setChampsEtOrdres(array('score' => SORT_DESC));
                        $this->infosPourTpl['noms'] = $this->trieur->trier();
                        $this->infosPourTpl['nomsSansCorrespondance'] =
                                                $this->classerAlphabetiquement('nomSciRetenu', $this->infosPourTpl['nomsSansCorrespondance'] );
                } else { // classement alphabétique
                        $nomsRetenus = array();
                        foreach ($this->infosPourTpl['noms'] as $categorie => $valeurs ) { //classement alpha par groupes 
                                $nomsRetenus += $this->classerAlphabetiquement('nomSciRetenu', $valeurs);
                        }
                        $this->infosPourTpl['noms'] = $nomsRetenus;
                        $this->infosPourTpl['nomsSansCorrespondance'] = 
                        $this->classerAlphabetiquement('nomSciRetenu', $this->infosPourTpl['nomsSansCorrespondance'] );
                }
        }
        
        private function ajouterAuxNomsScoreSimilariteAvec($masque) {
                $nom_demande_ss = strtolower($this->manipulateurDeChaine->supprimerAccents($masque));
                foreach ($this->infosPourTpl['noms'] as $id => $nom) {
                        $nom_flou_ss = strtolower($this->manipulateurDeChaine->supprimerAccents($nom['nomSciRetenu']));
                        $stat = array();
                        // Prime pour la ressemblance globale :
                        $score = 500 - levenshtein($nom_flou_ss, $nom_demande_ss);
                        // On affine
                        $score = $score + (similar_text($nom_demande_ss, $nom_flou_ss) * 3);
                        $nom['score'] = $score;
                        $this->infosPourTpl['noms'][$id] = $nom;
                }
        }

        public function surligner() {
                $this->definirMotsASurligner();
                foreach ($this->infosPourTpl['noms'] as $idNom => $nom) {
                        $this->infosPourTpl['noms'][$idNom]['nomSciRetenu'] = $this->surlignerMotsMasqueRecherche($nom['nomSciRetenu']);
                        if (isset($this->infosPourTpl['synonymes'][$idNom])) {
                                foreach ($this->infosPourTpl['synonymes'][$idNom] as $idSyn => $synonyme) {
                                        $this->infosPourTpl['synonymes'][$idNom][$idSyn]['nomSci'] = $this->surlignerMotsMasqueRecherche($synonyme['nomSci']);
                                }
                        }
                }
        }

        private function definirMotsASurligner() {
                $this->motsASurligner = explode(' ', $this->parametres->masqueRecherche);
        }

        private function surlignerMotsMasqueRecherche($nom) {
                $this->surligneur->setTexte($nom);
                $nom = $this->surligneur->surlignerMots($this->motsASurligner);
                return $nom;
        }
}
?>