Subversion Repositories eFlore/Applications.cel

Rev

Rev 1516 | Rev 1536 | 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';
1050 jpm 25
	const WS_SAISIE = "CelWidgetSaisie";
1345 aurelien 26
	const WS_NOM = "noms";
1476 aurelien 27
	private $NS_PROJET_VERSION = "1.01";
1345 aurelien 28
	const EFLORE_API_VERSION = "0.1";
1476 aurelien 29
	private $NS_PROJET = "bdtfx";
1050 jpm 30
 
1476 aurelien 31
	private $projetsVersions = array();
719 jpm 32
	private $projet = null;
33
	private $configProjet = null;
1050 jpm 34
 
712 jpm 35
	/**
36
	 * Méthode appelée par défaut pour charger ce widget.
37
	 */
38
	public function executer() {
39
		$retour = null;
40
		extract($this->parametres);
41
 
1345 aurelien 42
		$this->projet = isset($projet) ? $projet : self::PROJET_DEFAUT;
719 jpm 43
		$this->chargerConfigProjet();
1476 aurelien 44
		$this->chargerProjetsVersion();
1050 jpm 45
 
1345 aurelien 46
		$service = isset($service) ? $service : 'widget';
719 jpm 47
		$methode = $this->traiterNomMethodeExecuter($service);
712 jpm 48
		if (method_exists($this, $methode)) {
49
			$retour = $this->$methode();
50
		} else {
51
			$this->messages[] = "Ce type de service '$methode' n'est pas disponible.";
52
		}
53
 
1050 jpm 54
		$contenu = null;
55
		$mime = null;
56
		if (is_array($retour) && array_key_exists('squelette', $retour)) {
719 jpm 57
			$ext = (isset($retour['squelette_ext'])) ? $retour['squelette_ext'] : '.tpl.html';
1475 aurelien 58
			if($this->projetASquelette()) {
59
				$squelette = dirname(__FILE__).self::DS.'squelettes'.self::DS.$this->projet.self::DS.$retour['squelette'].$ext;
60
			} else {
61
				$squelette = dirname(__FILE__).self::DS.'squelettes'.self::DS.'defaut'.self::DS.'defaut'.$ext;
62
			}
712 jpm 63
			$contenu = $this->traiterSquelettePhp($squelette, $retour['donnees']);
1050 jpm 64
			$mime = isset($retour['mime']) ? $retour['mime'] : null;
65
		} else {
66
			if (count($this->messages) == 0) {
67
				$this->messages[] = "La méthode du sous-service ne renvoie pas les données dans le bon format.";
68
			}
69
			$contenu = 'Un problème est survenu : '.print_r($this->messages, true);
712 jpm 70
		}
1050 jpm 71
 
72
		$this->envoyer($contenu, $mime);
712 jpm 73
	}
1050 jpm 74
 
719 jpm 75
	private function chargerConfigProjet() {
76
		$fichier_config = dirname(__FILE__).self::DS.'configurations'.self::DS.$this->projet.'.ini';
77
		if (file_exists($fichier_config)) {
1526 jpm 78
			if (!$this->configProjet = parse_ini_file($fichier_config)) {
719 jpm 79
				$this->messages[] = "Le fichier ini '$fichier_config' du projet n'a pu être chargé.";
1050 jpm 80
			}
719 jpm 81
		} else {
1345 aurelien 82
			$this->debug[] = "Le fichier ini '$fichier_config' du projet n'existe pas.";
719 jpm 83
		}
84
	}
1526 jpm 85
 
1476 aurelien 86
	private function chargerProjetsVersion() {
1526 jpm 87
		if (isset($this->configProjet)) {
88
			foreach ($this->configProjet as $config => $valeur) {
89
				if(strstr($config,'.version')) {
90
					$this->projetsVersions[str_replace('.version', '', $config)] = $valeur;
91
				}
1476 aurelien 92
			}
93
		}
94
	}
1050 jpm 95
 
1345 aurelien 96
	public function executerWidget() {
1476 aurelien 97
		$referentiel_impose = false;
1526 jpm 98
		if (isset($_GET['referentiel']) && $_GET['referentiel'] != '') {
1476 aurelien 99
			$this->NS_PROJET = isset($_GET['referentiel']) && $_GET['referentiel'] != '' ? $_GET['referentiel'] : $this->NS_PROJET;
100
			$this->NS_PROJET_VERSION = $this->projetsVersions[$this->NS_PROJET];
101
			$referentiel_impose = true;
102
		}
1526 jpm 103
 
1050 jpm 104
		$widget['squelette'] = $this->projet;
105
		$widget['donnees'] = array();
106
		$widget['donnees']['url_base'] = sprintf($this->config['chemins']['baseURLAbsoluDyn'], '');
107
		$widget['donnees']['url_ws_saisie'] = sprintf($this->config['chemins']['baseURLServicesCelTpl'], self::WS_SAISIE);
1526 jpm 108
 
1516 aurelien 109
		$widget['donnees']['logo'] = isset($_GET['logo']) ? $_GET['logo'] : 'defaut';
110
		$widget['donnees']['titre'] = isset($_GET['titre']) ? $_GET['titre'] : 'defaut';
111
		$widget['donnees']['titre'] = ($widget['donnees']['titre'] == '0') ? '' : $widget['donnees']['titre'];
1526 jpm 112
 
1475 aurelien 113
		// cas du projet par défaut ou bien d'un projet n'ayant pas de squelette spécifique
1526 jpm 114
		if ($this->projet == 'defaut' || $this->projet == 'florileges' || !$this->projetASquelette()) {
1345 aurelien 115
			$urlWsNsTpl = $this->config['chemins']['baseURLServicesEfloreTpl'];
1476 aurelien 116
			$urlWsNs = sprintf($urlWsNsTpl, self::EFLORE_API_VERSION, $this->NS_PROJET, self::WS_NOM);
117
			$urlWsNsSansRef = sprintf($urlWsNsTpl, self::EFLORE_API_VERSION, '{referentiel}', self::WS_NOM);
1345 aurelien 118
			$widget['donnees']['url_ws_autocompletion_ns'] = $urlWsNs;
1476 aurelien 119
			$widget['donnees']['url_ws_autocompletion_ns_tpl'] = $urlWsNsSansRef;
120
			$widget['donnees']['ns_referentiel'] = $this->NS_PROJET.':'.$this->NS_PROJET_VERSION;
121
			$widget['donnees']['ns_projet'] = $this->NS_PROJET;
122
			$widget['donnees']['ns_version'] = $this->NS_PROJET_VERSION;
123
			$widget['donnees']['referentiel_impose'] = $referentiel_impose;
124
			$widget['donnees']['projets_versions'] = $this->projetsVersions;
1526 jpm 125
			$widget['donnees']['espece_imposee'] = false;
126
			$widget['donnees']['nn_espece_defaut'] = '';
127
			$widget['donnees']['nom_sci_espece_defaut'] = '';
128
			$widget['donnees']['infos_espece'] = array();
129
 
130
			if ($this->especeEstImposee()) {
131
				$nom = $this->executerChargementInfosTaxon($_GET['num_nom']);
1418 aurelien 132
				$widget['donnees']['espece_imposee'] = true;
133
				$widget['donnees']['nn_espece_defaut'] = $_GET['num_nom'];
1419 aurelien 134
				$widget['donnees']['nom_sci_espece_defaut'] = $nom['nom_sci'];
135
				$widget['donnees']['infos_espece'] = $this->array2js($nom, true);
1418 aurelien 136
			}
1345 aurelien 137
		} else {
138
			$widget['donnees']['taxons'] = $this->recupererListeTaxon();
139
			$widget['donnees']['milieux'] = $this->parserMilieux();
719 jpm 140
		}
1345 aurelien 141
		return  $widget;
719 jpm 142
	}
1526 jpm 143
 
1475 aurelien 144
	private function projetASquelette() {
1526 jpm 145
		// fonction très simple qui ne teste que si le dossier du projet courant
1475 aurelien 146
		// existe, mais elle suffit pour le moment.
147
		return file_exists(dirname(__FILE__).self::DS.'squelettes'.self::DS.$this->projet);
148
	}
1050 jpm 149
 
1345 aurelien 150
	public function executerTaxons() {
151
		$widget['squelette'] = $this->projet.'_taxons';
152
		$widget['squelette_ext'] = '.tpl.js';
153
		$widget['donnees'] = array();
154
		$taxons = $this->recupererListeTaxon();
155
		$taxons_tries = array();
156
		foreach ($taxons as $taxon) {
157
			$taxons_tries[$taxon['num_nom_sel']] = $taxon;
158
		}
159
		$widget['donnees']['taxons'] = json_encode($taxons_tries);
160
		return  $widget;
161
	}
162
 
163
	private function recupererListeTaxon() {
164
		$taxons = null;
165
		$fichier_tsv = dirname(__FILE__).self::DS.'configurations'.self::DS.$this->projet.'_taxons.tsv';
166
		if (file_exists($fichier_tsv) && is_readable($fichier_tsv)) {
712 jpm 167
			$taxons = $this->decomposerFichierTsv($fichier_tsv);
1345 aurelien 168
			$taxons = self::trierTableauMd($taxons, array('nom_fr' => SORT_ASC));
169
		} else {
170
			$this->debug[] = "Impossible d'ouvrir le fichier '$fichier_tsv'.";
171
		}
172
		return $taxons;
173
	}
174
 
175
	private function decomposerFichierTsv($fichier, $delimiter = "\t"){
176
		$header = NULL;
177
		$data = array();
178
		if (($handle = fopen($fichier, 'r')) !== FALSE) {
179
			while (($row = fgetcsv($handle, 1000, $delimiter)) !== FALSE) {
180
				if (!$header) {
181
					$header = $row;
182
				} else {
183
					$data[] = array_combine($header, $row);
184
				}
185
			}
186
			fclose($handle);
187
		}
188
		return $data;
712 jpm 189
	}
1050 jpm 190
 
1345 aurelien 191
	private function parserMilieux() {
1050 jpm 192
		$infosMilieux = array();
193
		$milieux = explode('|', $this->configProjet['milieux']);
194
		foreach ($milieux as $milieu) {
195
			$details = explode(';', $milieu);
196
			if (isset($details[1])) {
197
				$infosMilieux[$details[0]] = $details[1];
198
			} else {
199
				$infosMilieux[$details[0]] = '';
200
			}
201
		}
1345 aurelien 202
		ksort($infosMilieux);
1050 jpm 203
		return $infosMilieux;
204
	}
1526 jpm 205
 
1418 aurelien 206
	private function especeEstImposee() {
207
		return isset($_GET['num_nom']) && $_GET['num_nom'] != '';
208
	}
1526 jpm 209
 
1418 aurelien 210
	private function executerChargementInfosTaxon($num_nom) {
1476 aurelien 211
		$url_service_infos = sprintf($this->config['chemins']['infosTaxonUrl'], $this->NS_PROJET, $num_nom);
1418 aurelien 212
		$infos = json_decode(file_get_contents($url_service_infos));
1419 aurelien 213
		$resultat = array();
214
		if(isset($infos) && !empty($infos)) {
215
			$infos = (array)$infos;
216
			$resultat = (isset($infos['nom_sci']) && $infos['nom_sci'] != '') ? $infos : array();
217
		}
1418 aurelien 218
		return $resultat;
219
	}
1050 jpm 220
 
1345 aurelien 221
	public function executerUploadImage() {
222
		$retour = array(
223
				'squelette' => $this->projet.'_image',
224
				'squelette_ext' => '.tpl.xml',
225
				'mime' => 'text/xml',
226
				'donnees' => array(
227
						'urlMiniature' => '',
228
						'imageNom' => '',
229
						'message' => '',
230
						'debogage' => ''));
231
		$message = '';
232
		$debogage = '';
233
		if ($_FILES['fichier']['error'] == UPLOAD_ERR_OK) {
234
			if (is_uploaded_file($_FILES['fichier']['tmp_name'])) {
1355 aurelien 235
				if ($this->verifierFormatJpeg($_FILES['fichier']['tmp_name'])) {
1345 aurelien 236
					$dossierStockage = $this->config['chemins']['imagesTempDossier'];
237
 
238
					$nomFichierOriginal = preg_replace('/[.](jpeg|jpg)$/i', '.jpg', strtolower($_FILES['fichier']['name']));
239
					$originalChemin = $dossierStockage.$nomFichierOriginal;
240
					$deplacementOk = move_uploaded_file($_FILES['fichier']['tmp_name'], $originalChemin);
241
 
242
					if ($deplacementOk === true) {
243
						$miniatureFichier = str_replace('.jpg', '_min.jpg', $nomFichierOriginal);
244
						$miniatureChemin = $dossierStockage.$miniatureFichier;
245
 
246
						// Parametres
247
						$largeurIdeale = 100;
248
						$hauteurIdeale = 100;
249
						$qualite = 85;
250
 
251
						// Calcul de la hauteur et de la largeur optimale de la miniature
252
						$taillesImgOriginale = getimagesize($originalChemin);
253
						$largeurOrigine = $taillesImgOriginale[0];
254
						$hauteurOrigine = $taillesImgOriginale[1];
255
 
256
						$largeurMin = $largeurIdeale;
257
						$hauteurMin = (int) ($hauteurOrigine * ($largeurIdeale / $largeurOrigine));
258
						if ($hauteurMin > $hauteurIdeale) {
259
							$hauteurMin = $hauteurIdeale;
260
							$largeurMin = (int)($largeurOrigine * ($hauteurMin / $hauteurOrigine));
261
						}
262
 
263
						// Création de la miniature
264
						$imageOriginale = imagecreatefromjpeg($originalChemin);
265
						$imageMiniature = imagecreatetruecolor($largeurMin, $hauteurMin);
266
						$couleurFond = imagecolorallocate($imageMiniature, 255, 255, 255);
267
						imagefill($imageMiniature, 0, 0, $couleurFond);
268
						imagecopyresized($imageMiniature, $imageOriginale, 0, 0, 0, 0, $largeurMin, $hauteurMin, $largeurOrigine, $hauteurOrigine);
269
						imagejpeg($imageMiniature, $miniatureChemin, $qualite);
270
						imagedestroy($imageMiniature);
271
						imagedestroy($imageOriginale);
272
 
273
						// Retour des infos
274
						$retour['donnees']['urlMiniature'] = sprintf($this->config['chemins']['imagesTempUrlTpl'], $miniatureFichier);
275
						$retour['donnees']['imageNom'] = $nomFichierOriginal;
276
					} else {
277
						$message = "L'image n'a pu être déplacé sur le serveur.";
278
					}
1475 aurelien 279
				} else {
1345 aurelien 280
					$message = "L'image n'est pas au format JPEG.";
281
				}
282
			} else {
283
				$message = "L'image n'a pu être téléversée.";
284
				$debogage = $message.print_r($_FILES, true);
285
			}
286
		} else {
287
			if ($_FILES['fichier']['error'] == UPLOAD_ERR_FORM_SIZE) {
288
				$message = "L'image téléversée excède la taille maximum autorisée.".
289
						"Veuillez modifier votre image avant de la téléverser à nouveau.";
290
			} else {
291
				$message = "Une erreur de transfert a eu lieu (téléversement interrompu).";
292
			}
293
			$debogage = "Code erreur : {$_FILES['fichier']['error']}. ".
294
					"Voir : http://php.net/manual/fr/features.file-upload.errors.php";
295
		}
296
		// Retour des infos
297
		$retour['donnees']['message'] = $message;
298
		$retour['donnees']['debogage'] = $debogage;
299
		return $retour;
1050 jpm 300
	}
1526 jpm 301
 
302
	// Il ne faut pas utiliser l'index type du tableau files pour tester
1355 aurelien 303
	// si une image est en jpeg car le type renvoyé par les navigateurs
1526 jpm 304
	// peut varier (ex. sous ie qui renvoie image/pjpeg
1355 aurelien 305
	private function verifierFormatJpeg($chemin) {
306
		// get imagesize renvoie un résultat consistant par contre
307
		$infos = getimagesize($chemin, $infos);
1526 jpm 308
		return (isset($infos['mime']) && $infos['mime'] == 'image/jpeg');
1355 aurelien 309
	}
1526 jpm 310
 
311
	private function array2js($array,$show_keys) {
312
		$dimensions = array();
313
		$valeurs = array();
314
 
315
		$total = count($array) - 1;
316
		$i = 0;
317
		foreach ($array as $key => $value) {
318
			if (is_array($value)) {
319
				$dimensions[$i] = array2js($value,$show_keys);
320
				if ($show_keys) {
321
					$dimensions[$i] = '"'.$key.'":'.$dimensions[$i];
322
				}
323
			} else {
324
				$dimensions[$i] = '"'.addslashes($value).'"';
325
				if ($show_keys) {
326
					$dimensions[$i] = '"'.$key.'":'.$dimensions[$i];
327
				}
328
			}
329
			if ($i == 0) {
330
				$dimensions[$i] = '{'.$dimensions[$i];
331
			}
332
			if ($i == $total) {
333
				$dimensions[$i].= '}';
334
			}
335
			$i++;
336
		}
337
		return implode(',',$dimensions);
338
	}
1345 aurelien 339
}
340
?>