Subversion Repositories eFlore/Applications.del

Rev

Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
709 gduche 1
<?php
2
// declare(encoding='UTF-8');
3
/**
4
	* Le web service image récupère toutes les données de la table del_obs_images
5
 * pour retourner une liste d'images associée à une observation
6
 *
7
 * @category	php 5.2
8
 * @package	del
9
 * @subpackage images
10
 * @author		Jean-Pascal MILCENT <jpm@tela-botanica.org>
11
 * @copyright	Copyright (c) 2012, Tela Botanica (accueil@tela-botanica.org)
12
 * @license	http://www.cecill.info/licences/Licence_CeCILL_V2-fr.txt Licence CECILL
13
 * @license	http://www.gnu.org/licenses/gpl.html Licence GNU-GPL
14
 * @version	$Id: Bdd.php 403 2012-02-22 14:35:20Z gduche $
15
 * @see http://www.tela-botanica.org/wikini/eflore/wakka.php?wiki=ApiIdentiplante01Images
16
 */
17
 
1326 gduche 18
 
19
/**
20
 * FONCTION TEMPORAIRE de debug pour afficher le contenu d'une variable en format lisible
21
 * @param $r la variable à afficher
22
 * */
23
function debug($r) {
24
	echo '<pre>'.print_r($r, true).'</pre>';
25
}
26
 
27
 
28
/**
29
 * Le service ListeImages récupère les données des tables observation et images
30
 * et les mets au format JSON pour identiplante / pictoflora
31
 * */
751 delphine 32
class ListeImages {
709 gduche 33
 
1326 gduche 34
 
35
	// Variables :
36
	// Configuration générale du service
719 gduche 37
	private $conteneur;
725 gduche 38
	private $navigation;
1326 gduche 39
	private $bdd;
725 gduche 40
	private $gestionBdd;
1326 gduche 41
 
42
	// Parametres
43
	private $ressources = array();
759 delphine 44
	private $parametres = array();
1326 gduche 45
	private $masque;
46
 
800 aurelien 47
	private $tri = 'date_transmission';
48
	private $directionTri = 'desc';
1345 aurelien 49
	private $formatRetour = 'XL';
709 gduche 50
 
1326 gduche 51
	private $imageIds = array();
52
 
53
 
54
	/**
55
	 * Constructeur de l'application
56
	 * Initialisation des variables générale de l'application
57
	 * @param Conteneu $conteneur le conteneur de classes de l'application
58
	 * */
719 gduche 59
	public function __construct(Conteneur $conteneur = null) {
60
		$this->conteneur = $conteneur == null ? new Conteneur() : $conteneur;
791 aurelien 61
		$this->conteneur->chargerConfiguration('config_departements_bruts.ini');
798 aurelien 62
		$this->conteneur->chargerConfiguration('config_mapping_votes.ini');
775 aurelien 63
		$this->conteneur->chargerConfiguration('config_images.ini');
725 gduche 64
		$this->navigation = $conteneur->getNavigation();
65
		$this->masque = $conteneur->getMasque();
66
		$this->gestionBdd = $conteneur->getGestionBdd();
775 aurelien 67
		$this->bdd = $this->gestionBdd->getBdd();
719 gduche 68
	}
69
 
1326 gduche 70
 
709 gduche 71
	/**
72
	 * Méthode principale de la classe.
1326 gduche 73
	 * Lance la récupération des images dans la base et les place dans un objet ResultatService
709 gduche 74
	 * pour l'afficher.
75
	 * @param array $ressources les ressources situées après l'url de base (ex : http://url/ressource1/ressource2)
76
	 * @param array $parametres les paramètres situés après le ? dans l'url
77
	 * */
78
	public function consulter($ressources, $parametres) {
1326 gduche 79
 
759 delphine 80
		$this->initialiserRessourcesEtParametres($ressources, $parametres);
709 gduche 81
 
82
		$this->configurer();
711 gduche 83
		$this->verifierConfiguration();
1326 gduche 84
 
800 aurelien 85
		$this->verifierParametresTri();
86
		$this->initialiserTri();
1345 aurelien 87
 
88
		$this->verifierParametreFormatRetour();
89
		$this->initialiserFormatRetour();
1326 gduche 90
 
91
		$parametres = $this->nettoyerParametres($parametres);
92
 
93
		// En fonction des paramètres de recherche, on n'effectue
94
		// pas la même requête, pour optimiser les jointures et les
95
		// rapidités d'éxécution.
1443 mathias 96
		$type = $this->getTypeRequete($ressources, $parametres);
1326 gduche 97
 
98
		switch ($type) {
99
			case 'obs' :
100
				$liaisons = $this->chargerLiaisonsObs();
101
			break;
102
			case 'images' :
103
				$liaisons = $this->chargerLiaisonsImages();
104
			break;
105
			case 'obs-images' :
106
				$liaisons = $this->chargerLiaisons();
107
			break;
1443 mathias 108
			case 'id':
109
				$liaisons = $this->chargerLiaisonsParId();
110
			break;
1326 gduche 111
			default : //case simple !
1443 mathias 112
				$liaisons = $this->chargerLiaisonsSimple();
1326 gduche 113
		}
114
 
1439 raphael 115
		$images = array();
116
		if($liaisons) {
117
			// Partie commune à tous les cas : on complète les liaisons avec les informations des votes
118
			// et des images, puis on affiche sous forme de JSON
119
			$images = $this->chargerImage($liaisons);
120
			$images = $this->chargerVotes($images);
121
		}
1326 gduche 122
 
1321 gduche 123
		$resultat = new ResultatService();
124
		$resultat->corps = array('entete' => $this->conteneur->getEntete(), 'resultats' => $images);
1326 gduche 125
 
1321 gduche 126
		return $resultat;
709 gduche 127
	}
128
 
800 aurelien 129
 
130
 
1326 gduche 131
	/**************************************************************************************
132
	 *								FONCTION LIEES AUX REQUETES 						  *
133
	 **************************************************************************************/
709 gduche 134
 
135
	/**
1326 gduche 136
	 * Charger la clause WHERE en fonction des paramètres de masque
710 gduche 137
	 * */
1326 gduche 138
	private function chargerClauseWhere() {
1443 mathias 139
 
709 gduche 140
		$where = array();
725 gduche 141
		$tableauMasque = $this->masque->getMasque();
142
		if (!empty($tableauMasque)) {
143
			foreach($tableauMasque as $idMasque => $valeurMasque) {
1326 gduche 144
				//TODO: scinder ceci en fonctions réutilisables ?
145
				// voir si c'est interessant par rapport à la recherche générale
725 gduche 146
				$idMasque = str_replace('masque.', '', $idMasque);
147
				switch ($idMasque) {
1326 gduche 148
				// nom du masque => nom BDD
725 gduche 149
					case 'auteur' :
1326 gduche 150
							$whereAuteur = ' '.$this->creerFiltreAuteur($this->masque->getMasque('auteur'));
151
									if($whereAuteur != '') {
152
										$where[] = $whereAuteur;
153
									}
154
									break;
791 aurelien 155
					case 'date' :
1326 gduche 156
					$whereDate = ' '.$this->creerFiltreDate($valeurMasque);
157
					if($whereDate != '') {
158
						$where[] = $whereDate;
159
					}
160
					break;
791 aurelien 161
					case 'departement' :
162
						$where[] = ' '.$this->creerFiltreIdZoneGeo($valeurMasque);
163
						break;
725 gduche 164
					case 'genre' :
719 gduche 165
						$where[] = ' '.$this->mappingFiltre['ns'].' LIKE '.$this->proteger($valeurMasque.' %');
725 gduche 166
						break;
167
					case 'tag' :
791 aurelien 168
						$where[] = ' '.$this->creerFiltreMotsCles($valeurMasque);
725 gduche 169
						break;
1365 aurelien 170
					case 'referentiel' :
1413 aurelien 171
						$where[] = ' dob.nom_referentiel LIKE '.$this->proteger($valeurMasque.'%');
1365 aurelien 172
						break;
775 aurelien 173
					case 'ns' :
791 aurelien 174
						$where[] = ' nom_sel LIKE '.$this->proteger($valeurMasque.'%');
1326 gduche 175
						break;
1332 aurelien 176
					case 'nn' :
1441 mathias 177
						$where[] = ' (nom_sel_nn = '.intval($valeurMasque).' OR '.
178
								   ' nom_ret_nn = '.intval($valeurMasque).') ';
1332 aurelien 179
						break;
791 aurelien 180
					case 'commune' :
181
						$where[] = ' '.$this->mappingFiltre[$idMasque].' LIKE '.$this->proteger(str_replace(array('-',' '), '_', $valeurMasque).'%');
1326 gduche 182
						break;
791 aurelien 183
					case 'masque' :
184
						$where[] = ' '.$this->creerFiltreMasqueGeneral($valeurMasque);
775 aurelien 185
						break;
725 gduche 186
					default:
187
						$where[] = ' '.$this->mappingFiltre[$idMasque].' = '.$this->proteger($valeurMasque);
1326 gduche 188
						break;
725 gduche 189
				}
1326 gduche 190
			}
709 gduche 191
		}
1326 gduche 192
 
709 gduche 193
		if (!empty($where)) {
194
			return ' WHERE '.implode('AND', $where);
195
		} else {
196
			return;
197
		}
198
	}
199
 
1326 gduche 200
	/**
201
	 * Créer un masque général lorsque l'on souhaite utiliser le passe partout
202
	 * @param la valeur du passe partout
203
	 * */
791 aurelien 204
	private function creerFiltreMasqueGeneral($valeurMasque) {
1326 gduche 205
		//TODO: affecter d'aborder les variables, puis les tester pour les
791 aurelien 206
		// ajouter à la chaine
207
		$whereAuteur = $this->creerFiltreAuteur($valeurMasque);
1326 gduche 208
		$whereIdZoneGeo = $this->creerFiltreIdZoneGeo($valeurMasque);
209
 
791 aurelien 210
		$masqueGeneral = '( '.
1326 gduche 211
				(($whereAuteur != '') ? $whereAuteur.' OR ' : '' ).
212
				(($whereIdZoneGeo != '') ? $whereIdZoneGeo.' OR ' : '' ).
213
				'zone_geo LIKE '.$this->proteger($this->remplacerParJokerCaractere($valeurMasque).'%').' OR '.
214
				$this->creerFiltreMotsCles($valeurMasque).' OR '.
215
				'nom_sel LIKE '.$this->proteger($valeurMasque.'%').' OR '.
216
				'famille LIKE '.$this->proteger($valeurMasque.'%').' OR '.
217
				'milieu LIKE '.$this->proteger($valeurMasque).' OR '.
218
				$this->mappingFiltre['ns'].' LIKE '.$this->proteger('%'.$valeurMasque.'% %').' OR '.
219
				$this->creerFiltreDate($valeurMasque).
220
				') ';
791 aurelien 221
 
222
		return $masqueGeneral;
223
	}
224
 
1326 gduche 225
	/**
226
	 * Créer le filtre auteur en recherchant dans son nom, prénom, adresse email en fonction
227
	 * de la chaine donnée
228
	 * @param la valeur recherchée
229
	 * */
791 aurelien 230
	private function creerFiltreAuteur($valeurMasque) {
231
		$masque = '';
232
		$auteurId = $valeurMasque;
233
		if (is_numeric($auteurId)) {
234
			$masque = ' ce_utilisateur = '.$auteurId;
235
		} else {
236
			if (strpos($auteurId, '@') === false) {
237
				$tableauNomPrenom = explode(' ',$auteurId, 2);
238
				if(count($tableauNomPrenom) == 2) {
239
					// on teste potentiellement un nom prenom ou bien un prénom nom
240
					$masque = '('.
1326 gduche 241
							'(nom LIKE '.$this->proteger($tableauNomPrenom[0].'%').' AND '.
242
							'prenom LIKE '.$this->proteger($tableauNomPrenom[1].'%').') OR '.
243
							'(nom LIKE '.$this->proteger($tableauNomPrenom[1].'%').' AND '.
244
							'prenom LIKE '.$this->proteger($tableauNomPrenom[0].'%').') OR '.
245
							'(dob.nom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[0].'%').' AND '.
246
							'dob.prenom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[1].'%').') OR '.
247
							'(dob.nom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[1].'%').' AND '.
248
							'dob.prenom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[0].'%').') OR '.
249
							'(nom LIKE '.$this->proteger($valeurMasque.'%').') OR '.
250
							'(prenom LIKE '.$this->proteger($valeurMasque.'%').') OR '.
251
							'(dob.nom_utilisateur LIKE '.$this->proteger($valeurMasque.'%').') OR '.
252
							'(dob.prenom_utilisateur LIKE '.$this->proteger($valeurMasque.'%').') '.
253
							')';
791 aurelien 254
				} else {
255
					$masque = '(
256
						            (nom LIKE '.$this->proteger($auteurId.'%').' OR '.
1326 gduche 257
							            'prenom LIKE '.$this->proteger($auteurId.'%').' OR '.
258
							            'dob.nom_utilisateur LIKE '.$this->proteger($auteurId.'%').' OR '.
259
							            'dob.prenom_utilisateur LIKE '.$this->proteger($auteurId.'%').')'.
260
							            ')';
791 aurelien 261
				}
262
			} else {
1333 gduche 263
				$masque = " ( courriel LIKE ".$this->proteger($valeurMasque.'%').
264
				" OR dob.courriel_utilisateur LIKE ".$this->proteger($valeurMasque.'%').")  ";
791 aurelien 265
			}
266
		}
267
		return $masque;
268
	}
269
 
270
 
1326 gduche 271
	/**
272
	 * Obtenir une chaine de caractère concaténant nom et prénom séparé par une virgule
273
	 * @param String $auteurId l'identifiant de l'auteur
274
	 * @return String la chaine de concaténation
275
	 * */
276
	private function getChaineNomPrenom($auteurId) {
277
		$nomPrenom = explode(' ', $auteurId);
278
		$nomPrenom = $this->proteger($nomPrenom);
279
		$chaineNomPrenom = implode(', ', $nomPrenom);
280
		return $chaineNomPrenom;
791 aurelien 281
	}
282
 
1326 gduche 283
	/**
284
	 * Créer le filtre de recherche par zone géographique en fonction du masque
285
	 * @param $valeurMasque le terme de la recherche
286
	 * */
791 aurelien 287
	private function creerFiltreIdZoneGeo($valeurMasque) {
288
		$masque = '';
289
		$dept = $valeurMasque;
290
		if (is_numeric($dept)) {
291
			$dept = sprintf('%02s', $dept);
292
			$dept = sprintf("%-'_5s", $dept);
293
			$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$dept);
294
		} else {
295
			$deptId = $this->conteneur->getParametre($dept);
296
			if ($deptId != null) {
297
				$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$deptId.'%');
298
			} else {
299
				$id = $this->obtenirIdDepartement($valeurMasque);
300
				$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$id.'%');
301
			}
302
		}
303
		return $masque;
304
	}
305
 
1326 gduche 306
	/**
307
	 * Générer la chaine de recherche pour la date en  fonction du masque
308
	 * @param $valeurMasque la date recherchée (AAAA ou JJ/MM/AAAA)
309
	 * */
791 aurelien 310
	private function creerFiltreDate($valeurMasque) {
311
		//TODO: définir dans le fichier de config un tableau contenant plusieurs format de date
312
		// autorisés pour la recherche, qui seraient ajoutés au OR
313
		$masque = '(';
314
		$masque .= (is_numeric($valeurMasque)) ? ' YEAR(date_observation) = '.$this->proteger($valeurMasque).' OR ' : '';
315
		$masque .= " DATE_FORMAT(date_observation, '%d/%m/%Y') = ".$this->proteger($valeurMasque).' '.
1326 gduche 316
				')';
791 aurelien 317
		return $masque;
318
	}
801 aurelien 319
 
1326 gduche 320
	/**
321
	 * Générer la chaine de recherche dans les mots clés en fonction du masque
322
	 * @param $valeurMasque le mot clé recherché
323
	 * */
791 aurelien 324
	private function creerFiltreMotsCles($valeurMasque) {
1326 gduche 325
 
800 aurelien 326
		$mots_cles = explode(' ', $valeurMasque);
812 gduche 327
		$requeteMotsClesImg = array();
328
		$requeteMotsClesObs = array();
1180 aurelien 329
		$requeteMotsClesImgPublic = array();
1326 gduche 330
 
800 aurelien 331
		foreach($mots_cles as $mot_cle) {
1180 aurelien 332
			//TODO: rechercher sur les mots clés normalisés dans tous les cas ?
857 gduche 333
			$requeteMotsCles = $this->proteger('%'.$mot_cle.'%');
1180 aurelien 334
			$motsCleProtege = $this->proteger($this->normaliserMotCle('%'.$mot_cle.'%'));
1326 gduche 335
			$requeteMotsClesImgPublic[] = 'di.id_image IN (SELECT ce_image FROM del_image_tag WHERE tag_normalise LIKE '.$motsCleProtege.' AND actif = 1)';
857 gduche 336
			$requeteMotsClesImg[] = 'di.mots_cles_texte LIKE '.$requeteMotsCles;
337
			$requeteMotsClesObs[] = 'dob.mots_cles_texte LIKE '.$requeteMotsCles;
800 aurelien 338
		}
1326 gduche 339
 
1180 aurelien 340
		$requeteMotsClesImgPublic = implode(' AND ', $requeteMotsClesImgPublic);
857 gduche 341
		$requeteMotsClesImg = implode(' AND ', $requeteMotsClesImg);
342
		$requeteMotsClesObs = implode(' AND ', $requeteMotsClesObs);
1326 gduche 343
 
800 aurelien 344
		$masque = '('.
1326 gduche 345
				'('.$requeteMotsClesImgPublic.') OR '.
346
				'('.$requeteMotsClesImg.') OR '.
347
				'('.$requeteMotsClesObs.') '.
348
				')';
791 aurelien 349
 
1326 gduche 350
		return $masque;
709 gduche 351
	}
352
 
1326 gduche 353
	// ??
1168 aurelien 354
	private function assemblercomptageOccurencesMotsClesCel() {
355
		$chaineMotsClesAffiches = $this->conteneur->getParametre('mots_cles_cel_affiches');
356
		$tabMotsClesAffiches = explode(',',$chaineMotsClesAffiches);
357
		$chaineSql = '';
1326 gduche 358
 
1168 aurelien 359
		// Comptage du nombre de mots clés officiels présents dans la chaine mots clés texte
360
		foreach ($tabMotsClesAffiches as $motCle) {
361
			if($chaineSql != '') {
362
				$chaineSql .= ' + ';
363
			}
364
			$chaineSql .= 'IF(FIND_IN_SET('.$this->proteger($motCle).',di.mots_cles_texte) != 0, 1, 0)';
365
		}
1326 gduche 366
 
1168 aurelien 367
		return '('.$chaineSql.')';
368
	}
369
 
1326 gduche 370
	private function getTri() {
800 aurelien 371
		$order = '';
951 aurelien 372
		if($this->doitJoindreTableVotes()) {
373
			$order = ' GROUP BY dvote.ce_image, dob.id_observation ORDER BY total_votes '.$this->directionTri.', date_transmission desc ';
1168 aurelien 374
		} else if($this->doitJoindreTableTags()) {
375
			$order = ' GROUP BY doi.id_image ORDER BY total_tags '.$this->directionTri.', date_transmission desc ';
951 aurelien 376
		} else {
800 aurelien 377
			$order = ' ORDER BY '.$this->tri.' '.$this->directionTri.' ';
378
		}
379
		return $order;
380
	}
381
 
709 gduche 382
	/**
1326 gduche 383
	 * Compter le nombre total d'images dans la base pour affichage dans entete.
384
	 * */
385
	private function getFoundRows() {
709 gduche 386
		$requete = 'SELECT FOUND_ROWS() AS nbre ';
719 gduche 387
		$resultats = $this->bdd->recuperer($requete);
709 gduche 388
		return (int) $resultats['nbre'];
389
	}
390
 
1326 gduche 391
 
709 gduche 392
	/**
1326 gduche 393
	 * En fonction des paramètres, générer les conditions de recherche
394
	 * des observations
395
	 * */
396
	private function getConditionsObs() {
1327 gduche 397
		$conditionsObs = array();
1326 gduche 398
		$masques = $this->masque->getMasque();
399
		if (isset($masques['masque'])) {
400
			$passe = $masques['masque'];
401
 
402
			// Si on a saisi le masque passe partout, alors on doit chercher dans tous les champs
403
			// de la table observation (OR)
404
			$conditionLibre = array();
405
 
406
			if (!isset($masques['masque.ns'])) {
407
				$conditionsLibre[] = "nom_sel LIKE '$passe%'";
408
			}
409
 
410
			if (!isset($masques['masque.famille'])) {
411
				$conditionsLibre[] = "famille LIKE '$passe%'";
412
			}
413
 
414
			if (!isset($masques['masque.milieu'])) {
415
				$conditionsLibre[] = "nom_sel LIKE '$passe%'";
416
			}
417
 
418
			if (!isset($masques['masque.date'])) {
419
				$conditionsLibre[] = $this->creerFiltreDate($passe);
420
			}
421
 
422
			if (!isset($masques['masque.auteur'])) {
423
				$conditionsLibre[] = $this->creerFiltreAuteur($passe);
424
			}
425
 
426
			/*
427
			 * FIXME : remplacer par motcle projet !
428
			* if (!isset($masques['masque.tag'])) {
429
			$conditionsLibre[] = "mots_cles_texte LIKE '%$passe%'";
430
			}*/
431
 
432
			$conditionsObs[] = implode(' OR ', $conditionsLibre);
433
		}
1443 mathias 434
 
1366 aurelien 435
		// referentiel
436
		if (isset($masques['masque.referentiel'])) {
437
			$ref = $masques['masque.referentiel'];
1413 aurelien 438
			$conditionsObs[] = "dob.nom_referentiel LIKE '$ref%'";
1366 aurelien 439
		}
440
 
1326 gduche 441
		// nom sel
442
		if (isset($masques['masque.ns'])) {
443
			$nom_sel = $masques['masque.ns'];
444
			$conditionsObs[] = "nom_sel LIKE '$nom_sel%'";
445
		}
446
 
1332 aurelien 447
		// nom sel
448
		if (isset($masques['masque.nn'])) {
449
			$num_nom = $masques['masque.nn'];
450
			$conditionsObs[] = "(nom_sel_nn = '$num_nom' OR nom_ret_nn = '$num_nom') ";
451
		}
452
 
1326 gduche 453
		// famille
454
		if (isset($masques['masque.famille'])) {
455
			$famille = $masques['masque.famille'];
456
			$conditionsObs[] = "famille LIKE '$famille%'";
457
		}
458
 
459
		// genre
460
		if (isset($masques['masque.genre'])) {
461
			$genre = $masques['masque.genre'];
462
			$conditionsObs[] = "nom_sel LIKE '$genre%'";
463
		}
464
 
465
		// milieu
466
		if (isset($masques['masque.milieu'])) {
467
			$milieu = $masques['masque.milieu'];
468
			$conditionsObs[] = "nom_sel LIKE '$milieu%'";
469
		}
470
 
471
		// date
472
		if (isset($masques['masque.date'])) {
473
			$date = $masques['masque.date'];
474
			$conditionsObs[] = $this->creerFiltreDate($date);
475
		}
476
 
477
		// utilisateur
478
		if (isset($masques['masque.auteur'])) {
479
			$auteur = $masques['masque.auteur'];
480
			$conditionsObs[] = $this->creerFiltreAuteur($auteur);
481
		}
482
 
483
		// commune
484
		if (isset($masques['masque.commune'])) {
485
			$commune = $masques['masque.commune'];
486
			$conditionsObs[] = " zone_geo LIKE ".$this->proteger(str_replace(array('-',' '), '_', $commune).'%');
487
		}
488
 
489
		// commune
490
		if (isset($masques['masque.departement'])) {
491
			$dept = $masques['masque.departement'];
492
			$conditionsObs[] = $this->creerFiltreIdZoneGeo($dept);
493
		}
1332 aurelien 494
 
1326 gduche 495
		return $conditionsObs;
496
	}
497
 
498
	/**
499
	 * Obtenir le tableu de chaines de condition de requete images en fonction des masques
500
	 * */
501
	private function getConditionsImages() {
502
		$conditionsImg = array();
503
		$masques = $this->masque->getMasque();
504
		if (isset($masques['masque.tag'])) {
505
			$tag = $masques['masque.tag'];
506
			$conditionsImg[] = " dit.tag_normalise LIKE '$tag%' ";
507
			$conditionsImg[] = " di.mots_cles_texte LIKE '%$tag%' ";
508
		}
509
 
510
		return $conditionsImg;
511
	}
512
 
513
 
514
	/*-------------------------------------------------------------------------------
515
	 CHARGEMENT DES IMAGES
516
	--------------------------------------------------------------------------------*/
517
	/**
518
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
519
	 * Méthode appelée uniquement lorsque les paramètres sont vides
520
	 * */
521
	private function chargerLiaisonsSimple() {
522
 
523
		// On récupère d'abord les N images de del_obs_image, éventuellement triées,
524
		// Et on complète avec les informations des observations associées
525
		$requeteImages = ' SELECT *, di.mots_cles_texte as mots_cles_texte_image '.
526
						 ' FROM del_obs_image doi '.
527
						 ' INNER JOIN del_image di ON doi.id_image = di.id_image ';
528
 
529
		// Si le tri se fait par date d'observation, on récupère les identifiants de N observations triées
530
		if (isset($this->parametres['tri']) && $this->parametres['tri'] == 'date_observation') {
531
 
532
			$ordre = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : 'DESC';
533
 
534
			$requeteIdObs = ' SELECT doi.id_image  as id_image '.
535
							' FROM del_obs_image doi '.
536
							' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation '.
537
							' INNER JOIN del_image di ON doi.id_image = di.id_image '.
538
							' ORDER BY date_observation '.$ordre.', dob.id_observation '.$ordre;
539
			$requeteIdObs .= $this->gestionBdd->getLimitSql();
540
 
541
			// Récupérer les N observations triées par date
1438 raphael 542
			$observations = $this->bdd->recupererTous($requeteIdObs . ' -- ' . __FILE__ . ':' . __LINE__);
1326 gduche 543
 
544
			$idsImages = array();
545
			foreach ($observations as $observation) {
546
				$idsImages[] = $observation['id_image'];
547
			}
548
 
549
			$chaineIdImages = implode(',', $idsImages);
550
			$requeteImages .= ' WHERE doi.id_image IN ('.$chaineIdImages.') '.
1328 aurelien 551
							  ' GROUP BY doi.id_image, doi.id_observation '.
1326 gduche 552
							  ' ORDER BY FIELD(doi.id_image,  '.$chaineIdImages.')'.
553
							  ' LIMIT '.$this->navigation->getLimite(); // On limite sur le nombre car les obs peuvent avoir plusieurs images
554
 
555
		} else {
1328 aurelien 556
			$requeteImages .= ' GROUP BY doi.id_image, doi.id_observation ';
1392 raphael 557
			$requeteImages .= ' ORDER BY doi.id_observation DESC';
1326 gduche 558
			$requeteImages .= $this->gestionBdd->getLimitSql();
559
		}
560
 
1438 raphael 561
		$liaisons = $this->bdd->recupererTous($requeteImages . ' -- ' . __FILE__ . ':' . __LINE__);
1326 gduche 562
 
563
		// Ce n'est pas la peine de continuer s'il n'y a pas eu de résultats dans la table del_obs_images
564
		if (!empty($liaisons)) {
565
 
566
			$idsObservations = array();
567
			foreach ($liaisons as $image) {
568
				$idObs = $image['id_observation'];
569
				$idsObservations[$idObs] = $idObs;
570
			}
571
 
572
			$chaineIdObs = implode(',', $idsObservations);
573
 
574
 
575
			// On récupère les observations qui complètent la requête précédente
576
			$requeteObservations = ' SELECT * '.
577
								   ' FROM del_observation dob '.
578
								   ' LEFT JOIN del_utilisateur du ON dob.ce_utilisateur = du.id_utilisateur '.
579
								   ' WHERE id_observation IN ('.$chaineIdObs.')';
580
 
1438 raphael 581
			$resultatsObservations = $this->bdd->recupererTous($requeteObservations . ' -- ' . __FILE__ . ':' . __LINE__);
1326 gduche 582
 
583
			// FIXME : Ca ne doit pas arriver, mais que se passe-t-il s'il n'y a pas d'observation pour l'image ?!
584
 
585
			// On range les observations dans un tableau pour pouvoir les retrouver par leur id :
586
			$observations = array();
587
			foreach ($resultatsObservations as $id => $observation) {
588
				$idObs = $observation['id_observation'];
589
				$observations[$idObs] = $observation;
590
			}
591
 
592
			// Enfin, pour chaque image préalablement récupérées, on complète avec les informations de l'observation
593
			// FIXME : peut-être peut-on utiliser un array_merge ici ?
594
			foreach ($liaisons as $id => $liaison) {
595
				$idObs = $liaison['id_observation'];
596
 
597
				$observation = $observations[$idObs];
598
				foreach ($observation as $cle => $valeur) {
599
					$liaisons[$id][$cle] = $valeur;
600
				}
601
			}
602
 
603
 
604
			// On compte à part les images issues de la jointure de del_obs_image et del_image car la fonction
605
			// SQL_CALC_FOUND_ROWS dans la fonction requete image fait passer le temps d'éxécution de 0.0011 à 15s !
1328 aurelien 606
			$requeteNbImages = 'SELECT SUM(t.nb) as nb FROM (SELECT count(DISTINCT doi.id_image) as nb '.
607
							   'FROM del_obs_image doi '.
608
							   'INNER JOIN del_image di '.
609
							   'ON di.id_image = doi.id_image '.
610
							   'GROUP BY doi.id_image, doi.id_observation) t ';
1438 raphael 611
			$resultatNbImages = $this->bdd->recupererTous($requeteNbImages . ' -- ' . __FILE__ . ':' . __LINE__);
1328 aurelien 612
 
1326 gduche 613
			$total = (int) $resultatNbImages[0]['nb'];
614
			$this->navigation->setTotal($total);
615
		}
1328 aurelien 616
 
1326 gduche 617
		return $liaisons;
618
	}
1443 mathias 619
 
1326 gduche 620
	/**
1443 mathias 621
	 * Charge les liaisons pour une seule image dont l'id est spécifié dans l'URL
622
	 * Copie de chargerLiaisonsObs avec critère sur l'id_image uniquement
623
	 * Supporte seulement le masque sur referentiel
624
	 */
625
	private function chargerLiaisonsParId() {
626
 
627
		$idImage = $this->ressources[0];
628
 
629
		// Récupérer les liaisons
630
		$requeteObs = ' SELECT SQL_CALC_FOUND_ROWS dob.id_observation as id_observation, dob.nom_referentiel, nom_sel, nom_sel_nn, nt, famille, ce_zone_geo, zone_geo, lieudit, station, milieu, '.
631
				' date_observation, dob.mots_cles_texte as mots_cles_texte, dob.commentaire as commentaire, di.mots_cles_texte as mots_cles_texte_image , date_transmission, '.
632
				' doi.id_image as id_image, di.ce_utilisateur as ce_utilisateur, prenom, nom, courriel, dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, nom_original '.
633
				'FROM del_observation dob '.
634
				' INNER JOIN del_obs_image doi ON dob.id_observation = doi.id_observation '.
635
				' INNER JOIN del_image di ON doi.id_image = di.id_image '.
636
				' LEFT JOIN del_utilisateur du ON dob.ce_utilisateur = du.id_utilisateur '.
637
				' WHERE doi.id_image = '.intval($idImage);
638
 
639
		if (isset($this->parametres['masque.referentiel'])) {
640
			$requeteObs .= " AND dob.nom_referentiel LIKE '" . $this->parametres['masque.referentiel'] . "%'";
641
		}
642
 
643
		$observations = $this->bdd->recupererTous($requeteObs . ' -- ' . __FILE__ . ':' . __LINE__);
644
 
645
		$total = $this->getFoundRows();
646
		$this->navigation->setTotal($total);
647
 
648
		return $observations;
649
	}
650
 
651
	/**
1326 gduche 652
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
653
	 * Méthode appelée uniquement lorsque les paramètres concernent une observation
654
	 * */
655
	private function chargerLiaisonsObs() {
656
 
657
		// Récupérer les liaisons
1413 aurelien 658
		$requeteObs = ' SELECT SQL_CALC_FOUND_ROWS dob.id_observation as id_observation, dob.nom_referentiel, nom_sel, nom_sel_nn, nt, famille, ce_zone_geo, zone_geo, lieudit, station, milieu, '.
1326 gduche 659
				 	  ' date_observation, dob.mots_cles_texte as mots_cles_texte, dob.commentaire as commentaire, di.mots_cles_texte as mots_cles_texte_image , date_transmission, '.
660
					  ' doi.id_image as id_image, di.ce_utilisateur as ce_utilisateur, prenom, nom, courriel, dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, nom_original '.
661
					  'FROM del_observation dob '.
662
					  ' INNER JOIN del_obs_image doi ON dob.id_observation = doi.id_observation '.
663
					  ' INNER JOIN del_image di ON doi.id_image = di.id_image '.
664
					  ' LEFT JOIN del_utilisateur du ON dob.ce_utilisateur = du.id_utilisateur ';
665
 
666
		// Récupérer les conditions sous forme de tableau
667
		$conditionsObs = $this->getConditionsObs();
668
 
669
		if (!empty($conditionsObs)) {
670
			$where = ' WHERE '.implode(' AND ', $conditionsObs);
671
			$requeteObs .= $where;
672
		}
673
 
674
		// Gérer le tri (uniquement si c'est date_observation)
675
		if (isset($this->parametres['tri']) && $this->parametres['tri'] == 'date_observation') {
676
			$ordre = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : 'DESC';
677
			$tri = ' ORDER BY '.$this->parametres['tri'].' '.$ordre.', doi.id_observation '.$ordre.' ';
678
			$requeteObs .= $tri;
679
		}
1332 aurelien 680
 
1326 gduche 681
		$requeteObs .= $this->gestionBdd->getLimitSql();
1438 raphael 682
		$observations = $this->bdd->recupererTous($requeteObs . ' -- ' . __FILE__ . ':' . __LINE__);
1332 aurelien 683
 
1326 gduche 684
		$total = $this->getFoundRows();
685
		$this->navigation->setTotal($total);
686
 
687
		return $observations;
688
	}
689
 
690
	/**
691
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
692
	 * Méthode appelée uniquement lorsque les paramètres concernent les images
693
	 * */
694
	private function chargerLiaisonsImages() {
695
 
696
		// FIXME : si on faisait une requete à part pour compter, ca irait plus vite
697
		// Récupérer tous les ID d'image en fonction des paramètres de recherche
698
		$requeteImages = ' SELECT SQL_CALC_FOUND_ROWS '.
1413 aurelien 699
						 ' doi.id_image as id_image, dob.id_observation as id_observation, dob.nom_referentiel, nom_sel, nom_sel_nn, nt, famille, ce_zone_geo, zone_geo, lieudit, station, milieu, '.
1326 gduche 700
						 ' date_observation, dob.mots_cles_texte as mots_cles_texte, dob.commentaire as commentaire, di.mots_cles_texte as mots_cles_texte_image , date_transmission, '.
701
						 ' di.ce_utilisateur as ce_utilisateur, prenom, nom, courriel, dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, nom_original '.
702
						 ' FROM del_obs_image doi '.
703
						 ' INNER JOIN del_image di ON doi.id_image = di.id_image '.
704
						 ' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation '.
705
						 ' LEFT JOIN del_image_tag dit ON dit.ce_image = di.id_image '.
706
						 ' LEFT JOIN del_utilisateur du ON du.id_utilisateur = di.ce_utilisateur ';
1328 aurelien 707
 
1326 gduche 708
		$conditionsImg = $this->getConditionsImages();
709
 
710
		if (!empty($conditionsImg)) {
711
			$where = ' WHERE ('.implode(' OR ', $conditionsImg).') ';
712
			$where .= ' AND dit.actif = 1 ';
713
			$requeteImages .= $where;
714
		}
715
 
716
		// Gérer le tri, sur les votes ou sur les tags
717
		if (isset($this->parametres['tri'])) {
718
 
719
			$chaineTri = '';
720
			$chaineOrdre = '';
721
 
722
			if ($this->parametres['tri'] == 'votes') {
723
 
724
				$protocole = isset($this->parametres['protocole']) ? $this->parametres['protocole'] : 1;
725
 
726
				$requeteVotes = ' SELECT doi.id_image as id_image, IF(divo.ce_protocole = '.$protocole.', AVG(divo.valeur), 0) as total_votes '.
1330 aurelien 727
								' FROM del_obs_image doi '.
728
								' INNER JOIN del_image di ON doi.id_image = di.id_image '.
729
								' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation '.
1331 gduche 730
								' LEFT JOIN del_image_vote divo ON doi.id_image = divo.ce_image '.
731
                                ' AND ce_protocole = '.$protocole.' ';
1326 gduche 732
 
733
				// Et si on a cherché par tag ?
734
				if (isset($this->parametres['masque.tag'])) {
735
					$tag = $this->parametres['masque.tag'];
736
					$requeteVotes .= ' LEFT JOIN del_image_tag dit ON dit.ce_image = di.id_image ';
737
					$requeteVotes .= " WHERE (dit.tag_normalise LIKE '$tag%' OR  di.mots_cles_texte LIKE '%$tag%') AND dit.actif = 1 ";
738
				}
739
 
740
				$requeteVotes .= ' GROUP BY doi.id_image, doi.id_observation '.
1330 aurelien 741
								 ' ORDER by total_votes '.$this->directionTri .', doi.id_observation '.$this->directionTri.' '.
1326 gduche 742
						$this->gestionBdd->getLimitSql();
743
 
1331 gduche 744
 
1438 raphael 745
				$resultatsVotes = $this->bdd->recupererTous($requeteVotes . ' -- ' . __FILE__ . ':' . __LINE__);
1326 gduche 746
				$tabVotes = array();
747
				foreach ($resultatsVotes as $vote) {
748
					$tabVotes[] = $vote['id_image'];
749
				}
750
 
751
				$strVotes = empty($tabVotes) ? "''" : implode(',', $tabVotes);
752
 
753
				// Et si on a cherché par tag ?
754
				if (isset($this->parametres['masque.tag'])) {
755
					$chaineTri .= ' AND ';
756
				} else {
757
					$chaineTri .= ' WHERE ';
758
				}
759
 
760
 
761
				$chaineTri .= '  doi.id_image IN ('.$strVotes.') ';
762
				$chaineOrdre = ' ORDER BY FIELD(doi.id_image, '.$strVotes.')  ';
763
			}
1328 aurelien 764
 
1326 gduche 765
			if ($this->parametres['tri'] == 'tags') {
766
 
1328 aurelien 767
				$requetetags = ' SELECT SQL_CALC_FOUND_ROWS doi.id_image, COUNT(id_tag) as total_tags '.
1330 aurelien 768
						' FROM del_obs_image doi LEFT JOIN del_image_tag dit ON dit.ce_image = doi.id_image AND dit.actif = 1 '.
769
						 ' INNER JOIN del_image di ON doi.id_image = di.id_image '.
770
						 ' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation ';
771
 
1326 gduche 772
				if (isset($this->parametres['masque.tag'])) {
773
					// Et si on a cherché par tag ?
774
					$tag = $this->parametres['masque.tag'];
1328 aurelien 775
					$requetetags .= " WHERE (dit.tag_normalise LIKE '$tag%' OR  di.mots_cles_texte LIKE '%$tag%') ";
1326 gduche 776
				}
777
 
1330 aurelien 778
				$requetetags .= ' GROUP BY doi.id_image, doi.id_observation '.
779
						' ORDER by total_tags '.$this->directionTri.', doi.id_observation '.$this->directionTri.
1326 gduche 780
						$this->gestionBdd->getLimitSql();
1328 aurelien 781
 
1438 raphael 782
				$resultatstags = $this->bdd->recupererTous($requetetags . ' -- ' . __FILE__ . ':' . __LINE__);
1326 gduche 783
				$tabtags = array();
784
				foreach ($resultatstags as $tag) {
785
					$tabtags[] = $tag['id_image'];
786
				}
787
				$strtags = empty($tabtags) ? "''" : implode(',', $tabtags);
788
 
789
 
790
				// Et si on a cherché par tag ?
791
				if (isset($this->parametres['masque.tag'])) {
792
					$chaineTri .= ' AND ';
793
				} else {
794
					$chaineTri .= ' WHERE ';
795
				}
796
 
797
				$chaineTri .= ' doi.id_image IN ('.$strtags.') ';
798
				$chaineOrdre = ' ORDER BY FIELD(doi.id_image, '.$strtags.') ';
799
			}
1328 aurelien 800
 
1326 gduche 801
			$requeteImages .= $chaineTri.' GROUP BY doi.id_image, doi.id_observation '.$chaineOrdre;
802
		} else {
803
			$requeteImages .= ' GROUP BY doi.id_image, doi.id_observation'; // des fois, on a plusieurs observations pour la même image ...
804
			$requeteImages .= $this->gestionBdd->getLimitSql();
805
		}
1328 aurelien 806
 
1438 raphael 807
		$retour  = $this->bdd->recupererTous($requeteImages . ' -- ' . __FILE__ . ':' . __LINE__);
1326 gduche 808
		$total = $this->getFoundRows();
809
		$this->navigation->setTotal($total);
810
 
811
		return $retour;
812
	}
813
 
814
	/**
815
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
816
	 * */
817
	private function chargerLiaisons() {
818
 
1439 raphael 819
		$champs = array('dob.id_observation as id_observation', 'nom_sel', 'nom_sel_nn', 'nt', 'famille', 'dob.nom_referentiel', 'ce_zone_geo', 'zone_geo',
1326 gduche 820
				'lieudit', 'station', 'milieu', 'date_observation', 'dob.mots_cles_texte as mots_cles_texte', 'dob.commentaire as commentaire',
821
				'di.mots_cles_texte as mots_cles_texte_image ', 'date_transmission', 'di.id_image as id_image', 'di.ce_utilisateur as ce_utilisateur',
822
				'prenom', 'nom', 'courriel', 'dob.prenom_utilisateur', 'dob.nom_utilisateur', 'dob.courriel_utilisateur', 'nom_original');
823
		// Attention le LEFT JOIN est indispensable pour ramener les images n'ayant pas de votes
824
		// en cas de tri par votes
825
		$requeteLiaisons = 'SELECT DISTINCT SQL_CALC_FOUND_ROWS '.implode(', ',$champs).' '.
826
				($this->doitJoindreTableVotes() ?
827
						', IF(dvote.ce_protocole = '.$this->parametres['protocole'].', AVG(dvote.valeur), 0) as total_votes ' :
828
						''
829
				).
830
				($this->doitJoindreTableTags() ?
831
						// attention le DISTINCT est indispensable !
832
						', (COUNT(DISTINCT dtag.id_tag) + '.$this->assemblercomptageOccurencesMotsClesCel().') as total_tags ' :
833
						''
834
				).
835
				'FROM '.$this->gestionBdd->formaterTable('del_obs_image', 'doi').
836
				'INNER JOIN del_image di '.
837
				'ON doi.id_image = di.id_image '.
838
				'INNER JOIN del_observation dob '.
839
				'ON doi.id_observation = dob.id_observation '.
840
				'LEFT JOIN del_utilisateur du '.
841
				'ON du.id_utilisateur = di.ce_utilisateur '.
842
				($this->doitJoindreTableTags() ?
843
						'LEFT JOIN del_image_tag dtag '.
844
						'ON doi.id_image = dtag.ce_image AND dtag.actif = 1 ' :
845
						''
846
								).
847
								($this->doitJoindreTableVotes() ?
848
								'LEFT JOIN del_image_vote dvote '.
849
								'ON doi.id_image = dvote.ce_image AND dvote.ce_protocole = '.$this->parametres['protocole'] :
850
								''
851
										);
1332 aurelien 852
				$requeteLiaisons .= $this->chargerClauseWhere();
853
				$requeteLiaisons .= $this->getTri();
854
				$requeteLiaisons .= $this->gestionBdd->getLimitSql();
1366 aurelien 855
 
1438 raphael 856
				$retour = $this->bdd->recupererTous($requeteLiaisons . ' -- ' . __FILE__ . ':' . __LINE__);
1332 aurelien 857
				$total = $this->getFoundRows();
858
				$this->navigation->setTotal($total);
859
				return $retour;
1326 gduche 860
	}
861
 
862
 
863
	/**
864
	 * Retourner un tableau d'images formaté en fonction des liaisons trouvées
865
	 * @param $liaisons les liaisons de la table del_obs_images
866
	 * */
709 gduche 867
	private function chargerImage($liaisons) {
868
 
869
		$images = array();
870
		foreach ($liaisons as $liaison) {
736 gduche 871
			$idImage = $liaison['id_image'];
1326 gduche 872
 
1274 aurelien 873
			if($liaison['ce_utilisateur'] == 0) {
1326 gduche 874
				$liaison['prenom'] = $liaison['prenom_utilisateur'];
1274 aurelien 875
				$liaison['nom'] = $liaison['nom_utilisateur'];
876
			}
709 gduche 877
			// On enregistre l'ID de l'image pour n'effectuer qu'une seule requête par la suite
878
			$this->imageIds[] = $idImage;
800 aurelien 879
			$index = $liaison['id_image'].'-'.$liaison['id_observation'];
1025 aurelien 880
			$images[$index] = array('id_image' => $idImage, 'binaire.href' => $this->formaterLienImage($idImage),
1326 gduche 881
					'protocoles_votes' => array(),
882
					'mots_cles_texte' => $liaison['mots_cles_texte_image'], 'observation' => $this->formaterObservation($liaison));
709 gduche 883
		}
884
		return $images;
885
	}
886
 
887
	/**
1326 gduche 888
	 * Charger les votes pour chaque image
889
	 * */
709 gduche 890
	private function chargerVotes($images) {
1326 gduche 891
		$requeteVotes = 'SELECT v.*, p.* FROM '.
892
				$this->gestionBdd->formaterTable('del_image_vote', 'v').
893
				' INNER JOIN del_image_protocole p '.
894
				'ON v.ce_protocole = p.id_protocole '.
895
				$this->chargerClauseWhereVotes();
1438 raphael 896
		$resultatsVotes = $this->bdd->recupererTous($requeteVotes . ' -- ' . __FILE__ . ':' . __LINE__);
709 gduche 897
 
857 gduche 898
 
1326 gduche 899
		//TODO : faire une méthode formater vote
900
		$votes = $this->formaterVotes($resultatsVotes);
709 gduche 901
 
1326 gduche 902
		foreach ($images as $id => $image) {
903
			if (isset($votes[$image['id_image']])) {
904
				$images[$id]['protocoles_votes'] = $votes[$image['id_image']];
709 gduche 905
			}
1326 gduche 906
		}
907
 
709 gduche 908
		return $images;
909
	}
910
 
759 delphine 911
	private function chargerClauseWhereVotes() {
912
		if (sizeof($this->imageIds) > 0) {
913
			$chaineImageIds = implode(',', $this->imageIds);
760 delphine 914
			$where[] = 'v.ce_image  IN ('.$chaineImageIds.')';
759 delphine 915
		}
916
		if (isset($this->parametres['protocole'])) {
760 delphine 917
			$where[] = 'v.ce_protocole = '.$this->proteger($this->parametres['protocole']);
759 delphine 918
		}
1326 gduche 919
 
775 aurelien 920
		$where = (!empty($where)) ? 'WHERE '.implode(' AND ', $where) : '';
921
		return $where;
759 delphine 922
	}
923
 
1326 gduche 924
	/**************************************************************************************
925
	 *		    			FONCTION DE CONFIGURATION ET UTILITAIRES					  *
926
	***************************************************************************************/
927
	/**
928
	 * Enregistrer dans les variables de classe les paramètres et ressources
929
	 * @param $ressources
930
	 * @param $parametres de recherche
931
	 * */
932
	private function initialiserRessourcesEtParametres($ressources, $parametres) {
933
		$this->ressources = $ressources;
934
		$this->parametres = $parametres;
935
	}
936
 
937
	/**
938
	 * Configuration du service en fonction du fichier de config config_del.ini
939
	 * */
940
	public function configurer() {
941
		$this->mappingFiltre = $this->conteneur->getParametre('mapping_masque');
942
		$this->mappingObservation = $this->conteneur->getParametre('mapping_observation');
943
		$this->mappingVotes = $this->conteneur->getParametre('mapping_votes');
944
	}
945
 
946
	/**
947
	 * Vérifier que le service est bien configuré
948
	 * */
949
	public function verifierConfiguration() {
950
 
951
		$erreurs = array();
952
		$tableauImages = $this->conteneur->getParametre('images');
953
		if (empty($tableauImages)) {
954
			$erreurs[] = '- le fichier de configuration ne contient pas le tableau [images] ou celui-ci est vide ;';
955
		} else {
956
			if ($this->conteneur->getParametre('url_service') == null) {
957
				$erreurs[] = '- paramètre "url_service" manquant ;';
958
			}
959
 
960
			if ($this->conteneur->getParametre('url_images') == null) {
961
				$erreurs[] = '- paramètre "url_images" manquant ;';
962
			}
963
 
964
		}
965
 
966
		if (empty($this->mappingObservation)) {
967
			$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_observation] ou celui-ci est vide ;';
968
		} else {
969
			$champsMappingObs = array('id_observation', 'date_observation', 'date_transmission', 'famille', 'nom_sel', 'nom_sel_nn', 'nt',
970
					'ce_zone_geo', 'lieudit', 'station', 'milieu', 'ce_utilisateur', 'nom', 'prenom');
971
			foreach ($champsMappingObs as $champ) {
972
				if (!isset($this->mappingObservation[$champ])) {
973
					$erreurs[] = '- le mapping du champ "'.$champ.'" pour l\'observation est manquant ;';
974
				}
975
			}
976
		}
977
 
978
		if (empty($this->mappingFiltre)) {
979
			$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_masque] ou celui-ci est vide ;';
980
		} else {
981
			$champsMappingFiltre = array('famille', 'ns', 'nn', 'date', 'tag', 'commune');
982
			foreach ($champsMappingFiltre as $champ) {
983
				if (!isset($this->mappingFiltre[$champ])) {
984
					$erreurs[] = '- le mapping du champ "'.$champ.'" pour l\'observation est manquant ;';
985
				}
986
			}
987
		}
988
 
989
		$tris_possibles = $this->conteneur->getParametre('tris_possibles');
990
		if (empty($tris_possibles)) {
991
			$erreurs[] = '- le fichier de configuration ne contient pas le parametre tris_possibles ou celui-ci est vide ;';
992
		}
993
 
994
		if (!empty($erreurs)) {
995
			$e = 'Erreur lors de la configuration : '."\n";
996
			$e .= implode("\n", $erreurs);
997
			throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);
998
		}
999
	}
1000
 
1001
	/**
1002
	 * Verifier que les paramètres de tri sont bien autorisés et qu'ils sont au bon format.
1003
	 */
1004
	private function verifierParametresTri() {
1005
 
1006
		$erreurs = array();
1007
		$tris_possibles = $this->conteneur->getParametre('tris_possibles');
1008
		$tris_possibles_tableau = explode(',', $tris_possibles);
1009
		if(isset($this->parametres['tri']) && !in_array($this->parametres['tri'], $tris_possibles_tableau)) {
1010
			$erreurs[] = '- le type de tri demandé est incorrect, les valeurs possibles sont '.$tris_possibles.' ;';
1011
		}
1012
 
1013
		if(isset($this->parametres['tri']) && $this->parametres['tri'] == "votes") {
1014
			if(!isset($this->parametres['protocole']) || !is_numeric($this->parametres['protocole'])) {
1015
				$erreurs[] = '- Le paramètre protocole est obligatoire en cas de tri par vote et doit être un entier ;';
1016
			}
1017
		}
1018
 
1019
		$directions_tri = array('asc', 'desc');
1020
		if(isset($this->parametres['ordre']) && !in_array($this->parametres['ordre'], $directions_tri)) {
1021
			$erreurs[] = '- la direction du tri demandé est incorrecte, les valeurs supportées sont asc ou desc ;';
1022
		}
1023
 
1024
		if (!empty($erreurs)) {
1025
			$e = 'Erreur lors de l\'analyse des parametres du tri : '."\n";
1026
			$e .= implode("\n", $erreurs);
1027
			throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);
1028
		}
1029
	}
1345 aurelien 1030
 
1031
	private function verifierParametreFormatRetour() {
1032
		$erreurs = array();
1033
		$formats_possibles_str = $this->conteneur->getParametre('formats_possibles');
1034
		$formats_possibles = explode(',',$formats_possibles_str);
1326 gduche 1035
 
1345 aurelien 1036
		if(isset($this->parametres['format']) && !in_array($this->parametres['format'], $formats_possibles)) {
1037
			$erreurs[] = "- le format d'image demandé n'est pas supporté ; Les format supportés sont : ".$formats_possibles_str;
1038
		}
1039
 
1040
		if (!empty($erreurs)) {
1041
			$e = 'Erreur lors de l\'analyse du format de retour demandé : '."\n";
1042
			$e .= implode("\n", $erreurs);
1043
			throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);
1044
		}
1045
	}
1046
 
1047
	private function initialiserFormatRetour() {
1048
		$this->formatRetour = isset($this->parametres['format']) ? $this->parametres['format'] : $this->formatRetour;
1049
	}
1050
 
1326 gduche 1051
	/**
1052
	 * Initialiser les variables de tri depuis les paramètres
1053
	 * */
1054
	private function initialiserTri() {
1055
		$this->tri = isset($this->parametres['tri']) ? $this->parametres['tri'] : $this->tri;
1056
		$this->directionTri = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : $this->directionTri;
1057
	}
1058
 
1059
	/** Pour eviter les requêtes trop gourmandes, on supprime les caractères passe-partout
1060
	 * @param les paramètres de l'application
1061
	 * */
1062
	public function nettoyerParametres($parametres) {
1063
		$parametresRetour = array();
1064
		foreach ($parametres as $cle => $valeur) {
1065
			$valSansPourcent = trim($valeur, "% ");
1066
			if ($valSansPourcent != '') {
1067
				$parametresRetour[$cle] = $valeur;
1068
			}
1069
		}
1070
 
1071
		return $parametresRetour;
1072
	}
1073
 
1074
	/**
1075
	 * Nettoyer les jokers
1076
	 * @param la valeur du masque
1077
	 * */
1078
	private function remplacerParJokerCaractere($valeurMasque) {
1079
		return str_replace(array('-',' '), '_', $valeurMasque);
1080
	}
1081
 
1082
	//TODO: déplacer les fonctions ci dessus et dessous dans une classe
1083
	// utilitaire
1084
 
1085
	/**
1086
	 * Supprimer les accents des chaines de caractères
1087
	 * */
1088
	function supprimerAccents($str, $charset='utf-8')
1089
	{
1090
		$str = htmlentities($str, ENT_NOQUOTES, $charset);
1091
 
1092
		$str = preg_replace('#&([A-za-z])(?:acute|cedil|circ|grave|orn|ring|slash|th|tilde|uml);#', '\1', $str);
1093
		$str = preg_replace('#&([A-za-z]{2})(?:lig);#', '\1', $str); // pour les ligatures e.g. '&oelig;'
1094
		$str = preg_replace('#&[^;]+;#', '', $str); // supprime les autres caractères
1095
 
1096
		return $str;
1097
	}
1098
 
1099
	/**
1100
	 * Normaliser en supprimant les accents et en mettant en minuscule
1101
	 * @param $mot_cle le mot recherché
1102
	 * */
1103
	private function normaliserMotCle($mot_cle) {
1104
		return mb_strtolower($this->supprimerAccents(trim($mot_cle)));
1105
	}
1106
 
1107
	/**
1108
	 * Récupérer le numéro du département d'un fichier de configuration
1109
	 * */
1110
	private function obtenirIdDepartement($nomDpt) {
1111
 
1112
		$nomDpt = $this->supprimerAccents($nomDpt);
1113
		$nomDpt = strtolower(str_replace(' ','-',$nomDpt));
1114
 
1115
		$idDpt = $this->conteneur->getParametre($nomDpt);
1116
		if($idDpt == null || $idDpt == ' ') {
1117
			$idDpt = ' ';
1118
		}
1119
		return $idDpt;
1120
	}
1121
 
1122
 
1123
	/**
1124
	 * Obtenir le type de requête à exécuter en fonction des paramètres de recherche
1125
	 * @param $parametres les paramètres de l'application
1126
	 * */
1443 mathias 1127
	private function getTypeRequete($ressources, $parametres) {
1326 gduche 1128
 
1443 mathias 1129
		// une image par id
1130
		if ((count($ressources) == 1) && is_numeric($ressources[0])) {
1131
			return 'id';
1132
		}
1133
 
1326 gduche 1134
		$typeRequete = 'simple';
1135
 
1136
		// Dans ce cas précis, les informations concernant le depart, la limite ou l'ordre ne
1137
		// rentre pas en compte dans le type de requête ; ce ne sont que des compléments.
1138
		unset($parametres['navigation.depart']);
1139
		unset($parametres['navigation.limite']);
1140
		unset($parametres['ordre']);
1141
 
1142
		// En revanche, chaque masque est associé à un type de requête particulier.
1366 aurelien 1143
		$masquesObservation = array('masque', 'masque.departement', 'masque.ns', 'masque.genre', 'masque.date', 'masque.commune', 'masque.famille', 'masque.auteur', 'masque.nn', 'masque.referentiel');
1326 gduche 1144
		$masquesImage = array('masque', 'masque.tag');
1145
 
1146
		// Le type de requête est défini par les tables qu'il doit inclure (observation, image, ou les deux)
1147
		$requeteSimple = false;
1148
		$pourObservation = false;
1149
		$pourImage = false;
1150
 
1151
		// S'il n'y a aucun paramètre, on lance une requête simple
1152
		if (empty($parametres)) {
1153
			$requeteSimple = true;
1154
		}
1155
 
1156
		// Si l'un des masques demandé concerne l'observation
1157
		foreach ($masquesObservation as $masque) {
1158
			if (isset($parametres[$masque])) {
1159
				$pourObservation = true;
1160
				break;
1161
			}
1162
		}
1163
 
1164
		// Si l'un des masques demandé concerne les images
1165
		foreach ($masquesImage as $masque) {
1166
			if (isset($parametres[$masque])) {
1167
				$pourImage = true;
1168
				break;
1169
			}
1170
		}
1171
 
1172
		// Selon les tri
1173
		if (isset($parametres['tri'])) {
1174
			switch ($parametres['tri']) {
1175
				case 'votes' :
1176
				case 'tags' :
1177
					$pourImage = true;
1178
					break;
1179
				default : //case 'date_observation' :
1180
					if (sizeof($parametres) > 1) {
1181
						$pourObservation = true;
1182
					}
1183
			}
1184
		}
1185
 
1186
		// Vérifier la combinaison des booléens pour en déduire le type de requête
1187
		if ($pourObservation && $pourImage) {
1188
			$typeRequete = 'obs-images';
1189
		} else {
1190
			if ($pourImage) {
1191
				$typeRequete = 'images';
1192
			} else if ($pourObservation) {
1193
				$typeRequete = 'obs';
1194
			} else { // if ($requeteSimple)
1195
				$typeRequete = 'simple';
1196
			}
1197
		}
1198
 
1199
		return $typeRequete;
1200
	}
1201
 
1202
 
1203
	private function doitJoindreTableVotes() {
1204
		return ($this->tri == 'votes');
1205
	}
1206
 
1207
	private function doitJoindreTableTags() {
1208
		return ($this->tri == 'tags');
1209
	}
1210
 
709 gduche 1211
	/*-------------------------------------------------------------------------------
1212
								FORMATER ET METTRE EN FORME
1213
	--------------------------------------------------------------------------------*/
1214
 
1215
	/**
1216
	*  Formater une observation depuis une ligne liaison
1217
	*  @param $liaison liaison issue de la recherche
1218
	*  @return $observation l'observation mise en forme
1219
	* */
1220
	private function formaterObservation($liaison) {
1221
		$observation = array();
1222
		foreach ($this->mappingObservation as $nomOriginal => $nomFinal) {
1223
			$observation[$nomFinal] = $liaison[$nomOriginal];
1224
		}
1225
 
1226
		return $observation;
1227
	}
1228
 
1229
	/**
1230
	*  Formater une observation depuis une ligne liaison
1231
	*  @param $liaison liaison issue de la recherche
1232
	*  @return $observation l'observation mise en forme
1233
	* */
1234
	private function formaterVotes($votes) {
1235
		$retour = array();
1236
		foreach ($votes as $vote) {
759 delphine 1237
			$retour_vote = array();
1238
			foreach ($vote as $param=>$valeur) {
760 delphine 1239
				if (strpos($this->mappingVotes[$param], 'protocole.') === 0) {
1240
					$retour_protocole[$this->mappingVotes[$param]] = $valeur;
1241
				} else {
1242
					$retour_vote[$this->mappingVotes[$param]] = $valeur;
1243
				}
759 delphine 1244
			}
760 delphine 1245
			if (!isset($retour[$vote['ce_image']][$vote['ce_protocole']])) {
1246
				$retour[$vote['ce_image']][$vote['ce_protocole']] = $retour_protocole;
1247
			}
1248
			$retour[$vote['ce_image']][$vote['ce_protocole']]['votes'][$vote['id_vote']] = $retour_vote;
709 gduche 1249
		}
1250
 
1251
		return $retour;
1252
	}
1253
 
1254
	/**
1255
	 * Formater le lien de l'image en fonction du fichier de config et de l'identifiant de l'image
1256
	 * */
1257
	private function formaterLienImage($idImage) {
1258
		$idImage = sprintf('%09s', $idImage);
719 gduche 1259
		$url = $this->conteneur->getParametre('url_images');
1345 aurelien 1260
		$urlImage = sprintf($url, $idImage, $this->formatRetour);
709 gduche 1261
		return $urlImage;
1262
	}
719 gduche 1263
 
1264
	private function proteger($valeur) {
1265
		if (is_array($valeur)) {
1266
			return $this->bdd->protegerTableau($valeur);
1267
		} else {
1268
			return $this->bdd->proteger($valeur);
709 gduche 1269
		}
1270
	}
1271
}
1272
?>