Subversion Repositories eFlore/Applications.cel

Rev

Rev 1840 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
712 jpm 1
<?php
2
// declare(encoding='UTF-8');
3
/**
4
 * Widget fournissant des interfaces de saisies simplifiée pour différent projets.
5
 * Encodage en entrée : utf8
6
 * Encodage en sortie : utf8
7
 *
8
 * Cas d'utilisation et documentation :
9
 * @link http://www.tela-botanica.org/wikini/eflore/wakka.php?wiki=AideCELWidgetSaisie
10
 *
11
 * Paramètres :
12
 * ===> projet = chaine  [par défaut : Biodiversite34]
13
 * Indique quel projet nous voulons charger
14
 *
15
 * @author		Jean-Pascal MILCENT <jpm@tela-botanica.org>
16
 * @license		GPL v3 <http://www.gnu.org/licenses/gpl.txt>
17
 * @license		CECILL v2 <http://www.cecill.info/licences/Licence_CeCILL_V2-en.txt>
18
 * @version		$Id$
19
 * @copyright	Copyright (c) 2010, Tela Botanica (accueil@tela-botanica.org)
20
 */
21
class Saisie extends WidgetCommun {
1050 jpm 22
 
712 jpm 23
	const DS = DIRECTORY_SEPARATOR;
1345 aurelien 24
	const PROJET_DEFAUT = 'defaut';
1580 jpm 25
	const WS_SAISIE = 'CelWidgetSaisie';
26
	const WS_OBS = 'CelObs';
27
	const WS_NOM = 'noms';
28
	private $NS_PROJET_VERSION = '1.01';
29
	const EFLORE_API_VERSION = '0.1';
30
	private $NS_PROJET = 'bdtfx';
1050 jpm 31
 
1476 aurelien 32
	private $projetsVersions = array();
719 jpm 33
	private $projet = null;
34
	private $configProjet = null;
1050 jpm 35
 
712 jpm 36
	/**
37
	 * Méthode appelée par défaut pour charger ce widget.
38
	 */
39
	public function executer() {
40
		$retour = null;
41
		extract($this->parametres);
42
 
1345 aurelien 43
		$this->projet = isset($projet) ? $projet : self::PROJET_DEFAUT;
719 jpm 44
		$this->chargerConfigProjet();
1476 aurelien 45
		$this->chargerProjetsVersion();
1050 jpm 46
 
1345 aurelien 47
		$service = isset($service) ? $service : 'widget';
719 jpm 48
		$methode = $this->traiterNomMethodeExecuter($service);
712 jpm 49
		if (method_exists($this, $methode)) {
50
			$retour = $this->$methode();
51
		} else {
52
			$this->messages[] = "Ce type de service '$methode' n'est pas disponible.";
53
		}
54
 
1050 jpm 55
		$contenu = null;
56
		$mime = null;
57
		if (is_array($retour) && array_key_exists('squelette', $retour)) {
719 jpm 58
			$ext = (isset($retour['squelette_ext'])) ? $retour['squelette_ext'] : '.tpl.html';
1475 aurelien 59
			if($this->projetASquelette()) {
60
				$squelette = dirname(__FILE__).self::DS.'squelettes'.self::DS.$this->projet.self::DS.$retour['squelette'].$ext;
61
			} else {
62
				$squelette = dirname(__FILE__).self::DS.'squelettes'.self::DS.'defaut'.self::DS.'defaut'.$ext;
63
			}
712 jpm 64
			$contenu = $this->traiterSquelettePhp($squelette, $retour['donnees']);
1050 jpm 65
			$mime = isset($retour['mime']) ? $retour['mime'] : null;
66
		} else {
67
			if (count($this->messages) == 0) {
68
				$this->messages[] = "La méthode du sous-service ne renvoie pas les données dans le bon format.";
69
			}
70
			$contenu = 'Un problème est survenu : '.print_r($this->messages, true);
712 jpm 71
		}
1050 jpm 72
 
73
		$this->envoyer($contenu, $mime);
712 jpm 74
	}
1050 jpm 75
 
719 jpm 76
	private function chargerConfigProjet() {
77
		$fichier_config = dirname(__FILE__).self::DS.'configurations'.self::DS.$this->projet.'.ini';
78
		if (file_exists($fichier_config)) {
1526 jpm 79
			if (!$this->configProjet = parse_ini_file($fichier_config)) {
719 jpm 80
				$this->messages[] = "Le fichier ini '$fichier_config' du projet n'a pu être chargé.";
1050 jpm 81
			}
719 jpm 82
		} else {
1345 aurelien 83
			$this->debug[] = "Le fichier ini '$fichier_config' du projet n'existe pas.";
719 jpm 84
		}
85
	}
1526 jpm 86
 
1476 aurelien 87
	private function chargerProjetsVersion() {
1526 jpm 88
		if (isset($this->configProjet)) {
89
			foreach ($this->configProjet as $config => $valeur) {
90
				if(strstr($config,'.version')) {
91
					$this->projetsVersions[str_replace('.version', '', $config)] = $valeur;
92
				}
1476 aurelien 93
			}
94
		}
95
	}
1050 jpm 96
 
1345 aurelien 97
	public function executerWidget() {
1476 aurelien 98
		$referentiel_impose = false;
1526 jpm 99
		if (isset($_GET['referentiel']) && $_GET['referentiel'] != '') {
1476 aurelien 100
			$this->NS_PROJET = isset($_GET['referentiel']) && $_GET['referentiel'] != '' ? $_GET['referentiel'] : $this->NS_PROJET;
101
			$this->NS_PROJET_VERSION = $this->projetsVersions[$this->NS_PROJET];
102
			$referentiel_impose = true;
103
		}
1526 jpm 104
 
1050 jpm 105
		$widget['squelette'] = $this->projet;
106
		$widget['donnees'] = array();
107
		$widget['donnees']['url_base'] = sprintf($this->config['chemins']['baseURLAbsoluDyn'], '');
108
		$widget['donnees']['url_ws_saisie'] = sprintf($this->config['chemins']['baseURLServicesCelTpl'], self::WS_SAISIE);
1580 jpm 109
		$widget['donnees']['url_ws_obs'] = sprintf($this->config['chemins']['baseURLServicesCelTpl'], self::WS_OBS);
1536 jpm 110
		$widget['donnees']['url_ws_annuaire'] = sprintf($this->config['chemins']['baseURLServicesAnnuaireTpl'], 'utilisateur/identite-par-courriel/');
1526 jpm 111
 
1536 jpm 112
 
1516 aurelien 113
		$widget['donnees']['logo'] = isset($_GET['logo']) ? $_GET['logo'] : 'defaut';
114
		$widget['donnees']['titre'] = isset($_GET['titre']) ? $_GET['titre'] : 'defaut';
115
		$widget['donnees']['titre'] = ($widget['donnees']['titre'] == '0') ? '' : $widget['donnees']['titre'];
1526 jpm 116
 
1475 aurelien 117
		// cas du projet par défaut ou bien d'un projet n'ayant pas de squelette spécifique
1526 jpm 118
		if ($this->projet == 'defaut' || $this->projet == 'florileges' || !$this->projetASquelette()) {
1345 aurelien 119
			$urlWsNsTpl = $this->config['chemins']['baseURLServicesEfloreTpl'];
1476 aurelien 120
			$urlWsNs = sprintf($urlWsNsTpl, self::EFLORE_API_VERSION, $this->NS_PROJET, self::WS_NOM);
121
			$urlWsNsSansRef = sprintf($urlWsNsTpl, self::EFLORE_API_VERSION, '{referentiel}', self::WS_NOM);
1345 aurelien 122
			$widget['donnees']['url_ws_autocompletion_ns'] = $urlWsNs;
1476 aurelien 123
			$widget['donnees']['url_ws_autocompletion_ns_tpl'] = $urlWsNsSansRef;
124
			$widget['donnees']['ns_referentiel'] = $this->NS_PROJET.':'.$this->NS_PROJET_VERSION;
125
			$widget['donnees']['ns_projet'] = $this->NS_PROJET;
126
			$widget['donnees']['ns_version'] = $this->NS_PROJET_VERSION;
127
			$widget['donnees']['referentiel_impose'] = $referentiel_impose;
128
			$widget['donnees']['projets_versions'] = $this->projetsVersions;
1526 jpm 129
			$widget['donnees']['espece_imposee'] = false;
130
			$widget['donnees']['nn_espece_defaut'] = '';
131
			$widget['donnees']['nom_sci_espece_defaut'] = '';
1536 jpm 132
			$widget['donnees']['infos_espece'] = '{}';
1526 jpm 133
 
134
			if ($this->especeEstImposee()) {
135
				$nom = $this->executerChargementInfosTaxon($_GET['num_nom']);
1418 aurelien 136
				$widget['donnees']['espece_imposee'] = true;
137
				$widget['donnees']['nn_espece_defaut'] = $_GET['num_nom'];
1419 aurelien 138
				$widget['donnees']['nom_sci_espece_defaut'] = $nom['nom_sci'];
139
				$widget['donnees']['infos_espece'] = $this->array2js($nom, true);
1418 aurelien 140
			}
1536 jpm 141
		}
142
		if ($this->projet != 'defaut' || $this->projetASquelette()) {
1613 jpm 143
			if ($this->projet == 'florileges') {
144
				$widget['donnees']['taxons'] = $this->recupererListeNoms();
145
			} else {
146
				$widget['donnees']['taxons'] = $this->recupererListeNomsSci();
147
			}
1345 aurelien 148
			$widget['donnees']['milieux'] = $this->parserMilieux();
719 jpm 149
		}
1345 aurelien 150
		return  $widget;
719 jpm 151
	}
1526 jpm 152
 
1475 aurelien 153
	private function projetASquelette() {
1526 jpm 154
		// fonction très simple qui ne teste que si le dossier du projet courant
1475 aurelien 155
		// existe, mais elle suffit pour le moment.
156
		return file_exists(dirname(__FILE__).self::DS.'squelettes'.self::DS.$this->projet);
157
	}
1050 jpm 158
 
1345 aurelien 159
	public function executerTaxons() {
160
		$widget['squelette'] = $this->projet.'_taxons';
161
		$widget['squelette_ext'] = '.tpl.js';
1613 jpm 162
		$widget['donnees'] = array();
1629 jpm 163
		$nomsAAfficher = $this->recupererListeNomsSci();
164
		$taxons_tries = array();
1691 raphael 165
		foreach ($nomsAAfficher as $taxon) {
1629 jpm 166
			$taxons_tries[$taxon['num_nom_sel']] = $taxon;
167
		}
168
		$widget['donnees']['taxons'] = json_encode($taxons_tries);
169
		return $widget;
1345 aurelien 170
	}
1613 jpm 171
 
172
	private function recupererListeNomsSci() {
173
		$taxons = $this->recupererListeTaxon();
174
		if (is_array($taxons)) {
175
			$taxons = self::trierTableauMd($taxons, array('nom_fr' => SORT_ASC));
176
		}
1629 jpm 177
		return $taxons;
1613 jpm 178
	}
179
 
180
	private function recupererListeNoms() {
181
		$taxons = $this->recupererListeTaxon();
182
		$nomsAAfficher = array();
183
		$nomsSpeciaux = array();
184
		if (is_array($taxons)) {
185
			foreach ($taxons as $taxon) {
186
				$nomSciTitle = $taxon['nom_ret'].
187
					($taxon['nom_fr'] != '' ? ' - '.$taxon['nom_fr'] : '' ).
188
					($taxon['nom_fr_autre'] != '' ? ' - '.$taxon['nom_fr_autre'] : '' );
189
				$nomFrTitle = $taxon['nom_sel'].
190
					($taxon['nom_ret'] != $taxon['nom_sel']? ' - '.$taxon['nom_ret'] : '' ).
191
					($taxon['nom_fr_autre'] != '' ? ' - '.$taxon['nom_fr_autre'] : '' );
192
 
193
				if ($taxon['groupe'] == 'special') {
194
					$nomsSpeciaux[] = array(
195
						'num_nom' => $taxon['num_nom_sel'],
196
						'nom_a_afficher' => $taxon['nom_fr'],
197
						'nom_a_sauver' => $taxon['nom_sel'],
198
						'nom_title' => $nomSciTitle,
199
						'nom_type' => 'nom-special');
200
				} else {
201
					$nomsAAfficher[] = array(
202
						'num_nom' => $taxon['num_nom_sel'],
203
						'nom_a_afficher' => $taxon['nom_sel'],
204
						'nom_a_sauver' => $taxon['nom_sel'],
205
						'nom_title' => $nomSciTitle,
206
						'nom_type' => 'nom-sci');
207
					$nomsAAfficher[] = array(
208
						'num_nom' => $taxon['num_nom_sel'],
209
						'nom_a_afficher' => $taxon['nom_fr'],
210
						'nom_a_sauver' => $taxon['nom_fr'],
211
						'nom_title' => $nomFrTitle,
212
						'nom_type' => 'nom-fr');
213
				}
214
			}
215
			$nomsAAfficher = self::trierTableauMd($nomsAAfficher, array('nom_a_afficher' => SORT_ASC));
216
			$nomsSpeciaux = self::trierTableauMd($nomsSpeciaux, array('nom_a_afficher' => SORT_ASC));
217
		}
218
		return array('speciaux' => $nomsSpeciaux, 'sci-et-fr' => $nomsAAfficher);
219
	}
1345 aurelien 220
 
221
	private function recupererListeTaxon() {
222
		$taxons = null;
223
		$fichier_tsv = dirname(__FILE__).self::DS.'configurations'.self::DS.$this->projet.'_taxons.tsv';
224
		if (file_exists($fichier_tsv) && is_readable($fichier_tsv)) {
712 jpm 225
			$taxons = $this->decomposerFichierTsv($fichier_tsv);
1345 aurelien 226
		} else {
227
			$this->debug[] = "Impossible d'ouvrir le fichier '$fichier_tsv'.";
228
		}
229
		return $taxons;
230
	}
231
 
232
	private function decomposerFichierTsv($fichier, $delimiter = "\t"){
1536 jpm 233
		$header = null;
1345 aurelien 234
		$data = array();
235
		if (($handle = fopen($fichier, 'r')) !== FALSE) {
236
			while (($row = fgetcsv($handle, 1000, $delimiter)) !== FALSE) {
237
				if (!$header) {
238
					$header = $row;
239
				} else {
240
					$data[] = array_combine($header, $row);
241
				}
242
			}
243
			fclose($handle);
244
		}
245
		return $data;
712 jpm 246
	}
1050 jpm 247
 
1345 aurelien 248
	private function parserMilieux() {
1050 jpm 249
		$infosMilieux = array();
1536 jpm 250
		if (isset($this->configProjet['milieux'])) {
251
			$milieux = explode('|', $this->configProjet['milieux']);
252
			foreach ($milieux as $milieu) {
253
				$details = explode(';', $milieu);
254
				if (isset($details[1])) {
255
					$infosMilieux[$details[0]] = $details[1];
256
				} else {
257
					$infosMilieux[$details[0]] = '';
258
				}
1050 jpm 259
			}
1536 jpm 260
			ksort($infosMilieux);
1050 jpm 261
		}
262
		return $infosMilieux;
263
	}
1526 jpm 264
 
1418 aurelien 265
	private function especeEstImposee() {
1536 jpm 266
		return (isset($_GET['num_nom']) && $_GET['num_nom'] != '');
1418 aurelien 267
	}
1526 jpm 268
 
1418 aurelien 269
	private function executerChargementInfosTaxon($num_nom) {
1476 aurelien 270
		$url_service_infos = sprintf($this->config['chemins']['infosTaxonUrl'], $this->NS_PROJET, $num_nom);
1418 aurelien 271
		$infos = json_decode(file_get_contents($url_service_infos));
1855 aurelien 272
		// trop de champs injectés dans les infos espèces peut
273
		// faire planter javascript
274
		$champs_a_garder = array('id', 'nom_sci','nom_sci_complet',
275
		'famille','nom_retenu.id', 'nom_retenu.libelle', 'num_taxonomique');
1419 aurelien 276
		$resultat = array();
1539 jpm 277
		if (isset($infos) && !empty($infos)) {
1419 aurelien 278
			$infos = (array)$infos;
1855 aurelien 279
			if(isset($infos['nom_sci']) && $infos['nom_sci'] != '') {
280
				$resultat = array_intersect_key($infos, array_flip($champs_a_garder));
281
				$resultat['retenu'] = ($infos['id'] == $infos['nom_retenu.id']) ? "true" : "false";
282
			}
1419 aurelien 283
		}
1418 aurelien 284
		return $resultat;
285
	}
1050 jpm 286
 
1345 aurelien 287
	public function executerUploadImage() {
288
		$retour = array(
289
				'squelette' => $this->projet.'_image',
290
				'squelette_ext' => '.tpl.xml',
291
				'mime' => 'text/xml',
292
				'donnees' => array(
293
						'urlMiniature' => '',
294
						'imageNom' => '',
295
						'message' => '',
296
						'debogage' => ''));
297
		$message = '';
298
		$debogage = '';
299
		if ($_FILES['fichier']['error'] == UPLOAD_ERR_OK) {
300
			if (is_uploaded_file($_FILES['fichier']['tmp_name'])) {
1355 aurelien 301
				if ($this->verifierFormatJpeg($_FILES['fichier']['tmp_name'])) {
1345 aurelien 302
					$dossierStockage = $this->config['chemins']['imagesTempDossier'];
303
 
304
					$nomFichierOriginal = preg_replace('/[.](jpeg|jpg)$/i', '.jpg', strtolower($_FILES['fichier']['name']));
305
					$originalChemin = $dossierStockage.$nomFichierOriginal;
306
					$deplacementOk = move_uploaded_file($_FILES['fichier']['tmp_name'], $originalChemin);
307
 
308
					if ($deplacementOk === true) {
309
						$miniatureFichier = str_replace('.jpg', '_min.jpg', $nomFichierOriginal);
310
						$miniatureChemin = $dossierStockage.$miniatureFichier;
311
 
312
						// Parametres
313
						$largeurIdeale = 100;
314
						$hauteurIdeale = 100;
315
						$qualite = 85;
316
 
317
						// Calcul de la hauteur et de la largeur optimale de la miniature
318
						$taillesImgOriginale = getimagesize($originalChemin);
319
						$largeurOrigine = $taillesImgOriginale[0];
320
						$hauteurOrigine = $taillesImgOriginale[1];
321
 
322
						$largeurMin = $largeurIdeale;
323
						$hauteurMin = (int) ($hauteurOrigine * ($largeurIdeale / $largeurOrigine));
324
						if ($hauteurMin > $hauteurIdeale) {
325
							$hauteurMin = $hauteurIdeale;
326
							$largeurMin = (int)($largeurOrigine * ($hauteurMin / $hauteurOrigine));
327
						}
328
 
329
						// Création de la miniature
330
						$imageOriginale = imagecreatefromjpeg($originalChemin);
331
						$imageMiniature = imagecreatetruecolor($largeurMin, $hauteurMin);
332
						$couleurFond = imagecolorallocate($imageMiniature, 255, 255, 255);
333
						imagefill($imageMiniature, 0, 0, $couleurFond);
334
						imagecopyresized($imageMiniature, $imageOriginale, 0, 0, 0, 0, $largeurMin, $hauteurMin, $largeurOrigine, $hauteurOrigine);
335
						imagejpeg($imageMiniature, $miniatureChemin, $qualite);
336
						imagedestroy($imageMiniature);
337
						imagedestroy($imageOriginale);
338
 
339
						// Retour des infos
340
						$retour['donnees']['urlMiniature'] = sprintf($this->config['chemins']['imagesTempUrlTpl'], $miniatureFichier);
341
						$retour['donnees']['imageNom'] = $nomFichierOriginal;
342
					} else {
343
						$message = "L'image n'a pu être déplacé sur le serveur.";
344
					}
1475 aurelien 345
				} else {
1345 aurelien 346
					$message = "L'image n'est pas au format JPEG.";
347
				}
348
			} else {
349
				$message = "L'image n'a pu être téléversée.";
350
				$debogage = $message.print_r($_FILES, true);
351
			}
352
		} else {
353
			if ($_FILES['fichier']['error'] == UPLOAD_ERR_FORM_SIZE) {
354
				$message = "L'image téléversée excède la taille maximum autorisée.".
355
						"Veuillez modifier votre image avant de la téléverser à nouveau.";
356
			} else {
357
				$message = "Une erreur de transfert a eu lieu (téléversement interrompu).";
358
			}
359
			$debogage = "Code erreur : {$_FILES['fichier']['error']}. ".
360
					"Voir : http://php.net/manual/fr/features.file-upload.errors.php";
361
		}
362
		// Retour des infos
363
		$retour['donnees']['message'] = $message;
364
		$retour['donnees']['debogage'] = $debogage;
365
		return $retour;
1050 jpm 366
	}
1526 jpm 367
 
368
	// Il ne faut pas utiliser l'index type du tableau files pour tester
1355 aurelien 369
	// si une image est en jpeg car le type renvoyé par les navigateurs
1526 jpm 370
	// peut varier (ex. sous ie qui renvoie image/pjpeg
1355 aurelien 371
	private function verifierFormatJpeg($chemin) {
372
		// get imagesize renvoie un résultat consistant par contre
373
		$infos = getimagesize($chemin, $infos);
1526 jpm 374
		return (isset($infos['mime']) && $infos['mime'] == 'image/jpeg');
1355 aurelien 375
	}
1526 jpm 376
 
377
	private function array2js($array,$show_keys) {
1536 jpm 378
		$tableauJs = '{}';
379
		if (!empty($array)) {
380
			$total = count($array) - 1;
381
			$i = 0;
382
			$dimensions = array();
383
			foreach ($array as $key => $value) {
384
				if (is_array($value)) {
385
					$dimensions[$i] = array2js($value,$show_keys);
386
					if ($show_keys) {
387
						$dimensions[$i] = '"'.$key.'":'.$dimensions[$i];
388
					}
389
				} else {
390
					$dimensions[$i] = '"'.addslashes($value).'"';
391
					if ($show_keys) {
392
						$dimensions[$i] = '"'.$key.'":'.$dimensions[$i];
393
					}
1526 jpm 394
				}
1536 jpm 395
				if ($i == 0) {
396
					$dimensions[$i] = '{'.$dimensions[$i];
1526 jpm 397
				}
1536 jpm 398
				if ($i == $total) {
399
					$dimensions[$i].= '}';
400
				}
401
				$i++;
1526 jpm 402
			}
1536 jpm 403
			$tableauJs = implode(',', $dimensions);
1526 jpm 404
		}
1536 jpm 405
		return $tableauJs;
1526 jpm 406
	}
1345 aurelien 407
}
408
?>