1,20 → 1,20 |
<?php |
// declare(encoding='UTF-8'); |
/** |
* Classe recherchant des infos sur une zone géo ou bien des coordonnées. |
* |
* @internal Mininum PHP version : 5.2 |
* @category CEL |
* @package Services |
* @subpackage Bibliothèques |
* @version 0.1 |
* @author Mathias CHOUET <mathias@tela-botanica.org> |
* @author Jean-Pascal MILCENT <jpm@tela-botanica.org> |
* @author Aurelien PERONNET <aurelien@tela-botanica.org> |
* @license GPL v3 <http://www.gnu.org/licenses/gpl.txt> |
* @license CECILL v2 <http://www.cecill.info/licences/Licence_CeCILL_V2-en.txt> |
* @copyright 1999-2014 Tela Botanica <accueil@tela-botanica.org> |
*/ |
// declare(encoding='UTF-8'); |
/** |
* Classe recherchant des infos sur une zone géo ou bien des coordonnées. |
* |
* @internal Mininum PHP version : 5.2 |
* @category CEL |
* @package Services |
* @subpackage Bibliothèques |
* @version 0.1 |
* @author Mathias CHOUET <mathias@tela-botanica.org> |
* @author Jean-Pascal MILCENT <jpm@tela-botanica.org> |
* @author Aurelien PERONNET <aurelien@tela-botanica.org> |
* @license GPL v3 <http://www.gnu.org/licenses/gpl.txt> |
* @license CECILL v2 <http://www.cecill.info/licences/Licence_CeCILL_V2-en.txt> |
* @copyright 1999-2014 Tela Botanica <accueil@tela-botanica.org> |
*/ |
class RechercheInfosZoneGeo extends Cel { |
|
public $maxTimeoutRequeteMondiale = 0; |
57,21 → 57,21 |
*/ |
|
//TODO: migrer tout ce qui concerne la zone géo dans cette classe |
public function obtenirInfosPourCoordonnees($coordonnees) { |
$infos_coord = null; |
// Test facile qui permet d'abord de tenter une localisation bien plus rapide si les coordonnées |
// sont dans la bounding box approximative de la France |
if($this->testerCoordonneesWgsFrance($coordonnees['latitude'], $coordonnees['longitude'])) { |
$infos_coord = $this->chercherInfosCommune('osm', $coordonnees['latitude'], $coordonnees['longitude']); |
if ($infos_coord == null) { |
// Sinon recherche par pays |
$infos_coord = $this->chercherInfosPays($coordonnees['latitude'], $coordonnees['longitude']); |
} |
} else { |
// Recherche par pays immédiate si en dehors de la bouding box française |
$infos_coord = $this->chercherInfosPays($coordonnees['latitude'], $coordonnees['longitude']); |
} |
return $infos_coord; |
public function obtenirInfosPourCoordonnees($coordonnees) { |
$infos_coord = null; |
// Test facile qui permet d'abord de tenter une localisation bien plus rapide si les coordonnées |
// sont dans la bounding box approximative de la France |
if($this->testerCoordonneesWgsFrance($coordonnees['latitude'], $coordonnees['longitude'])) { |
$infos_coord = $this->chercherInfosCommune('osm', $coordonnees['latitude'], $coordonnees['longitude']); |
if ($infos_coord == null) { |
// Sinon recherche par pays |
$infos_coord = $this->chercherInfosPays($coordonnees['latitude'], $coordonnees['longitude']); |
} |
} else { |
// Recherche par pays immédiate si en dehors de la bouding box française |
$infos_coord = $this->chercherInfosPays($coordonnees['latitude'], $coordonnees['longitude']); |
} |
return $infos_coord; |
} |
|
public function obtenirInfosPourNoms($nom, $pays, $code) { |
93,132 → 93,132 |
} |
|
// Si on ne trouve rien on teste dans le monde entier |
if($retour == false) { |
if($retour == false) { |
$retour = $this->effectuerRequeteGeocodingMondiale($nom, '2,3,4,5,6,7,8', 1, $pays); |
} |
} |
|
return $retour; |
} |
|
public function chercherInfosCommune($projet, $latitude, $longitude) { |
|
$valeurs_params = '?lat='.$latitude.'&lon='.$longitude; |
$url_service = $this->config['cel']['url_service_geo_local'].$valeurs_params; |
$url_service = str_replace(',', '.', $url_service); |
|
$ch = curl_init($url_service); |
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); |
$reponse = curl_exec($ch); |
|
$infos_coord = $this->traiterReponseServiceCommune($reponse); |
curl_close($ch); |
|
return $infos_coord; |
} |
|
} |
|
public function chercherInfosCommune($projet, $latitude, $longitude) { |
|
$valeurs_params = '?lat='.$latitude.'&lon='.$longitude; |
$url_service = $this->config['cel']['url_service_geo_local'].$valeurs_params; |
$url_service = str_replace(',', '.', $url_service); |
|
$ch = curl_init($url_service); |
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); |
$reponse = curl_exec($ch); |
|
$infos_coord = $this->traiterReponseServiceCommune($reponse); |
curl_close($ch); |
|
return $infos_coord; |
} |
|
public function chercherInfosPays($latitude, $longitude) { |
|
|
$valeurs_params = '?lat='.$latitude.'&lon='.$longitude; |
$url_service = $this->config['cel']['url_service_geo_mondial'].$valeurs_params; |
$url_service = str_replace(',', '.', $url_service); |
|
$ch = curl_init($url_service); |
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); |
$reponse = curl_exec($ch); |
|
$res = json_decode($reponse, true); |
$url_service = str_replace(',', '.', $url_service); |
|
$ch = curl_init($url_service); |
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); |
$reponse = curl_exec($ch); |
|
$res = json_decode($reponse, true); |
$retour = null; |
|
$retour = array( |
'code_zone' => '', |
'code_pays' => '', |
'nom' => '', |
$retour = array( |
'code_zone' => '', |
'code_pays' => '', |
'nom' => '', |
'type' => '' |
); |
|
if(!empty($res)) { |
// Cas où l'on a recherché des coordonnées en France |
// On en profite pour remplir plus de données |
// que le simple pays si elles sont présentes |
if(isset($res[8]) && isset($res[8]['codeInsee'])) { |
$retour = array( |
|
if(!empty($res)) { |
// Cas où l'on a recherché des coordonnées en France |
// On en profite pour remplir plus de données |
// que le simple pays si elles sont présentes |
if(isset($res[8]) && isset($res[8]['codeInsee'])) { |
$retour = array( |
'code_zone' => $res[8]['codeInsee'], |
'code_pays' => 'FR', |
'nom' => $res[8]['nom'], |
'type' => 'france' |
); |
} else { |
// Cas de la recherche en dehors de France |
// La zone de plus haut niveau est toujours un pays |
// (car le niveau de zone est limité à 2) |
$infos_pays = $res[min(array_keys($res))]; |
// La zone de niveau le plus bas est la "localité" |
// la plus précise qu'on a pu trouver |
$infos_localite = $res[max(array_keys($res))]; |
|
// Cas où l'on a trouvé un code pays |
if(!empty($infos_pays['codeIso31661'])) { |
$retour = array( |
'code_pays' => $infos_pays['codeIso31661'], |
'nom' => '' |
); |
} elseif(!empty($infos_pays['codeIso31662'])) { |
// Quelquefois la zone de plus haut niveau est une région ou une province |
// heureusement son code est de forme XX-YY ou XX est le code iso du pays ! |
$retour = array( |
'code_pays' => substr($infos_pays['codeIso31662'], 0, 2), |
'nom' => '' |
); |
} |
|
// Pas de sens de prendre un nom de localité si on a pas de pays |
if(!empty($retour['code_pays'])) { |
// Type sert à savoir quelle fonction de mise à jour on appellera |
$retour['type'] = 'monde'; |
'code_pays' => 'FR', |
'nom' => $res[8]['nom'], |
'type' => 'france' |
); |
} else { |
// Cas de la recherche en dehors de France |
// La zone de plus haut niveau est toujours un pays |
// (car le niveau de zone est limité à 2) |
$infos_pays = $res[min(array_keys($res))]; |
// La zone de niveau le plus bas est la "localité" |
// la plus précise qu'on a pu trouver |
$infos_localite = $res[max(array_keys($res))]; |
|
// Cas où l'on a trouvé un code pays |
if(!empty($infos_pays['codeIso31661'])) { |
$retour = array( |
'code_pays' => $infos_pays['codeIso31661'], |
'nom' => '' |
); |
} elseif(!empty($infos_pays['codeIso31662'])) { |
// Quelquefois la zone de plus haut niveau est une région ou une province |
// heureusement son code est de forme XX-YY ou XX est le code iso du pays ! |
$retour = array( |
'code_pays' => substr($infos_pays['codeIso31662'], 0, 2), |
'nom' => '' |
); |
} |
|
// Pas de sens de prendre un nom de localité si on a pas de pays |
if(!empty($retour['code_pays'])) { |
// Type sert à savoir quelle fonction de mise à jour on appellera |
$retour['type'] = 'monde'; |
$retour['nom'] = $infos_localite['intitule']; |
$retour['code_zone'] = $infos_localite['codeZoneGeo']; |
} |
} |
$retour['code_zone'] = $infos_localite['codeZoneGeo']; |
} |
} |
} |
|
return $retour; |
} |
|
public function traiterReponseServiceCommune($reponse) { |
$retour = null; |
$reponse = json_decode($reponse); |
// cas du service lion 1906 qui peut renvoyer plusieurs communes (on prend la première) |
if (is_array($reponse)) { |
$reponse = $reponse[0]; |
} |
if (isset($reponse->codeINSEE)) { |
// Type sert à savoir quelle fonction de mise à jour on appellera |
|
return $retour; |
} |
|
public function traiterReponseServiceCommune($reponse) { |
$retour = null; |
$reponse = json_decode($reponse); |
// cas du service lion 1906 qui peut renvoyer plusieurs communes (on prend la première) |
if (is_array($reponse)) { |
$reponse = $reponse[0]; |
} |
if (isset($reponse->codeINSEE)) { |
// Type sert à savoir quelle fonction de mise à jour on appellera |
$retour = array( |
'code_zone' => $reponse->codeINSEE, |
'code_pays' => 'FR', |
'nom' => $reponse->nom, |
'code_pays' => 'FR', |
'nom' => $reponse->nom, |
'type' => 'france' |
); |
} |
return $retour; |
); |
} |
return $retour; |
} |
|
private function effectuerRequeteGeocodingMondiale($nom, $niveau = '2,3,4,5,6,7,8', $limite = 1, $pays = null) { |
$url_sans_pays = $this->config['cel']['url_service_geo_mondial'].'?masque='.urlencode($nom).'&niveau='.$niveau.'&limite='.$limite; |
private function effectuerRequeteGeocodingMondiale($nom, $niveau = '2,3,4,5,6,7,8', $limite = 1, $pays = null) { |
$url_sans_pays = $this->config['cel']['url_service_geo_mondial'].'?masque='.urlencode($nom).'&niveau='.$niveau.'&limite='.$limite; |
$url = $url_sans_pays.(!empty($pays) ? '&pays='.urlencode($pays) : ''); |
|
$ch = curl_init($url); |
$ch = curl_init($url); |
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); |
curl_setopt($ch, CURLOPT_TIMEOUT, $this->maxTimeoutRequeteMondiale); |
curl_setopt($ch, CURLOPT_TIMEOUT, $this->maxTimeoutRequeteMondiale); |
$reponse = curl_exec($ch); |
|
$res = @json_decode($reponse, true); |
|
$retour = array(); |
$retour = array(); |
|
if(!empty($res)) { |
// cas de la localisation directe d'une commune |
// cas de la localisation directe d'une commune |
if($limite == 1) { |
$retour = $this->traiterLigneResultatRequeteGeocodingMondiale($res); |
} else { |
235,42 → 235,42 |
// préciser de pays |
$res = @json_decode(file_get_contents($url_sans_pays), true); |
|
if(!empty($res)) { |
if(!empty($res)) { |
$retour = $this->traiterLigneResultatRequeteGeocodingMondiale($res); |
} |
} |
|
return $retour; |
} |
|
return $retour; |
} |
|
private function chercherCentroideCommuneBdd($commune, $departement) { |
private function chercherCentroideCommuneBdd($commune, $departement) { |
$commune_formatee = str_replace(array(' ', '-'), '_', trim($commune)); |
|
$departement = trim($departement, "*"); |
if (strlen($departement) > 2) { |
$departement = substr($departement, 0, 2); |
} |
$requete = 'SELECT utm_x, utm_y, utm_secteur, code FROM cel_zones_geo '. |
'WHERE nom LIKE '.Cel::db()->proteger($commune_formatee).' '. |
'AND code LIKE '.Cel::db()->proteger($departement.'%').' '. |
$departement = trim($departement, "*"); |
if (strlen($departement) > 2) { |
$departement = substr($departement, 0, 2); |
} |
$requete = 'SELECT utm_x, utm_y, utm_secteur, code FROM cel_zones_geo '. |
'WHERE nom LIKE '.Cel::db()->proteger($commune_formatee).' '. |
'AND code LIKE '.Cel::db()->proteger($departement.'%').' '. |
' -- '.__FILE__.':'.__LINE__; |
|
$commune_coordonnees = Cel::db()->requeter($requete); |
|
$retour = false; |
if ($commune_coordonnees && is_array($commune_coordonnees) && count($commune_coordonnees) > 0) { |
$lat_lon = $this->convertirUtmVersLatLong($commune_coordonnees[0]['utm_x'],$commune_coordonnees[0]['utm_y'],$commune_coordonnees[0]['utm_secteur']); |
|
$retour = array( |
'lat' => (float) $lat_lon['lat'], |
'lng' => (float) $lat_lon['lng'], |
'nom' => $commune, |
'code_zone' => $commune_coordonnees[0]['code'], |
|
$commune_coordonnees = Cel::db()->requeter($requete); |
|
$retour = false; |
if ($commune_coordonnees && is_array($commune_coordonnees) && count($commune_coordonnees) > 0) { |
$lat_lon = $this->convertirUtmVersLatLong($commune_coordonnees[0]['utm_x'],$commune_coordonnees[0]['utm_y'],$commune_coordonnees[0]['utm_secteur']); |
|
$retour = array( |
'lat' => (float) $lat_lon['lat'], |
'lng' => (float) $lat_lon['lng'], |
'nom' => $commune, |
'code_zone' => $commune_coordonnees[0]['code'], |
'code_pays' => 'FR', |
'type' => 'france' |
); |
} |
return $retour; |
'type' => 'france' |
); |
} |
return $retour; |
} |
|
public function obtenirListeInfosPourNom($lieu) { |
284,42 → 284,44 |
$lieux_retour += $lieux_mondiaux; |
} |
|
// Suppression de possibles doublons |
$lieux_retour = $this->reduireListeLieux($lieux_retour); |
// Suppression de possibles doublons @WARNING marche pas !!! |
//$lieux_retour = $this->reduireListeLieux($lieux_retour); |
|
return $lieux_retour; |
} |
|
public function obtenirListeCommunes($lieu) { |
$lieu_formate = $this->formaterChaineLieuPourRequete($lieu); |
$retour = array(); |
|
if ($this->estUneChaineRequeteValide($lieu_formate)) { |
$requete = 'SELECT DISTINCT nom, code '. |
'FROM cel_zones_geo '. |
'WHERE nom LIKE '.Cel::db()->proteger($lieu_formate.'%').' '. |
'ORDER BY nom '. |
'LIMIT 50 '. |
' -- '.__FILE__.':'.__LINE__; |
|
$liste_lieux = Cel::db()->requeter($requete); |
if ($liste_lieux) { |
foreach ($liste_lieux as $lieu_trouve) { |
$retour[] = $this->formaterLigneResultat($lieu_trouve); |
} |
} |
} |
$lieu_formate = $this->formaterChaineLieuPourRequete($lieu); |
$retour = array(); |
|
if ($this->estUneChaineRequeteValide($lieu_formate)) { |
$requete = 'SELECT DISTINCT nom, code '. |
'FROM cel_zones_geo '. |
'WHERE nom LIKE '.Cel::db()->proteger($lieu_formate.'%').' '. |
'ORDER BY nom '. |
'LIMIT 50 '. |
' -- '.__FILE__.':'.__LINE__; |
|
$liste_lieux = Cel::db()->requeter($requete); |
if ($liste_lieux) { |
foreach ($liste_lieux as $lieu_trouve) { |
$retour[] = $this->formaterLigneResultat($lieu_trouve); |
} |
} |
} |
return $retour; |
} |
|
/** |
* Fonctions utilitaires |
* |
*/ |
|
*/ |
|
// qui a écrit cette m*rde ? |
private function reduireListeLieux($tableau) { |
$index = array(); |
foreach($tableau as $ligne) { |
// merci de mettre des commentaires !!! |
// ça enlève les 2 premières lettres ou chiffres du nom de lieu => wtf ? |
$index_loc = trim(preg_replace("/\([a-zA-Z0-9]{2}\)$/", "", $ligne[0])); |
if(!isset($index[$index_loc])) { |
$index[$index_loc] = $ligne; |
340,96 → 342,96 |
private function trierListeLieux($a, $b) { |
return strcmp($a[0], $b[0]); |
} |
|
private function testerCoordonneesWgsFrance($latitude, $longitude) { |
$coord_france = false; |
if ($latitude != '' && $longitude != '') { |
if ($latitude < 51.071667 && $latitude > 41.316667) { |
if ($longitude < 9.513333 && $longitude > -5.140278) { |
$coord_france = true; |
} |
} |
} |
return $coord_france; |
|
private function testerCoordonneesWgsFrance($latitude, $longitude) { |
$coord_france = false; |
if ($latitude != '' && $longitude != '') { |
if ($latitude < 51.071667 && $latitude > 41.316667) { |
if ($longitude < 9.513333 && $longitude > -5.140278) { |
$coord_france = true; |
} |
} |
} |
return $coord_france; |
} |
|
private function traiterLigneResultatRequeteGeocodingMondiale($res) { |
|
$lat = ""; |
$lon = ""; |
$nom = ""; |
$code =""; |
$code_pays=""; |
$type=""; |
|
if(!empty($res)) { |
$code_pays = $this->extraireCodePays($res); |
|
$nom = $res['intitule']; |
$lat = $res['centre_lat']; |
$lon = $res['centre_lng']; |
$type = $code_pays == 'FR' || $code_pays == 'FX' ? 'france' : 'monde'; |
|
$retour = array( |
'lat' => (float)$lat, |
'lng' => (float)$lon, |
'nom' => $nom, |
'code_zone' => $code, |
'code_pays' => $code_pays, |
'type' => $type |
); |
} |
|
return $retour; |
private function traiterLigneResultatRequeteGeocodingMondiale($res) { |
|
$lat = ""; |
$lon = ""; |
$nom = ""; |
$code =""; |
$code_pays=""; |
$type=""; |
|
if(!empty($res)) { |
$code_pays = $this->extraireCodePays($res); |
|
$nom = $res['intitule']; |
$lat = $res['centre_lat']; |
$lon = $res['centre_lng']; |
$type = $code_pays == 'FR' || $code_pays == 'FX' ? 'france' : 'monde'; |
|
$retour = array( |
'lat' => (float)$lat, |
'lng' => (float)$lon, |
'nom' => $nom, |
'code_zone' => $code, |
'code_pays' => $code_pays, |
'type' => $type |
); |
} |
|
return $retour; |
} |
|
private function convertirUtmVersLatLong($x, $y, $sector) { |
$convertisseur = new gPoint(); |
$convertisseur->setUTM($x, $y, $sector); |
$convertisseur->convertTMtoLL(); |
|
$lat_long = array(); |
$lat_long['lat'] = str_replace(',','.',$convertisseur->Lat()); |
$lat_long['lng'] = str_replace(',','.',$convertisseur->Long()); |
return $lat_long; |
private function convertirUtmVersLatLong($x, $y, $sector) { |
$convertisseur = new gPoint(); |
$convertisseur->setUTM($x, $y, $sector); |
$convertisseur->convertTMtoLL(); |
|
$lat_long = array(); |
$lat_long['lat'] = str_replace(',','.',$convertisseur->Lat()); |
$lat_long['lng'] = str_replace(',','.',$convertisseur->Long()); |
return $lat_long; |
} |
|
private function extraireCodePays($infos_pays) { |
$code = ""; |
if(!empty($infos_pays['codeIso31661'])) { |
$code = $infos_pays['codeIso31661']; |
} elseif(!empty($infos_pays['codeIso31662'])) { |
// Si pas de code pays, le sous code peut exister et content le code pays |
// sous la forme codepays-souscode |
$code = substr($infos_pays['codeIso31662'], 0, 2); |
} |
return $code; |
private function extraireCodePays($infos_pays) { |
$code = ""; |
if(!empty($infos_pays['codeIso31661'])) { |
$code = $infos_pays['codeIso31661']; |
} elseif(!empty($infos_pays['codeIso31662'])) { |
// Si pas de code pays, le sous code peut exister et content le code pays |
// sous la forme codepays-souscode |
$code = substr($infos_pays['codeIso31662'], 0, 2); |
} |
return $code; |
} |
|
private function estUneChaineRequeteValide($lieu) { |
return (strlen($lieu) > 0) && ($lieu != '%'); |
|
private function estUneChaineRequeteValide($lieu) { |
return (strlen($lieu) > 0) && ($lieu != '%'); |
} |
|
|
private function formaterLigneResultat($ligne) { |
if(empty($ligne['code']) || $ligne['code'] == null) { |
$res = array($ligne['nom'], ''); |
} else { |
$res = array($ligne['nom'].' ('.substr(sprintf('%02s', $ligne['code']),0,2).')', $ligne['code']); |
} |
return $res; |
} |
return $res; |
} |
|
/** |
* Remplace les * par % pour faire des recherches floues |
* Remplace les + par _ (nginx envoie des "+" dans l'URL à la place des espaces) |
* Remplace les espaces et les - par _ car les noms de communes peuvent avoir des espaces ou des tirets |
* @param string $lieu |
* @return string le lieu formaté pour la recherche |
*/ |
private function formaterChaineLieuPourRequete($lieu) { |
$lieu = ltrim($lieu); |
$lieu = preg_replace('/\*+/', '%', $lieu); |
$lieu = str_replace(['+', ' ', '-'], '_', $lieu); |
return $lieu; |
/** |
* Remplace les * par % pour faire des recherches floues |
* Remplace les + par _ (nginx envoie des "+" dans l'URL à la place des espaces) |
* Remplace les espaces et les - par _ car les noms de communes peuvent avoir des espaces ou des tirets |
* @param string $lieu |
* @return string le lieu formaté pour la recherche |
*/ |
private function formaterChaineLieuPourRequete($lieu) { |
$lieu = ltrim($lieu); |
$lieu = preg_replace('/\*+/', '%', $lieu); |
$lieu = str_replace(['+', ' ', '-'], '_', $lieu); |
return $lieu; |
} |
} |