[PLUGINS] +les plugins de bases
[ptitvelo/web/www.git] / www / plugins / saisies / formulaires / construire_formulaire.php
1 <?php
2
3 // Sécurité
4 if (!defined("_ECRIRE_INC_VERSION")) return;
5
6 function formulaires_construire_formulaire_charger($identifiant, $formulaire_initial=array(), $options=array()){
7 include_spip('inc/saisies');
8 $contexte = array();
9
10 // On ajoute un préfixe devant l'identifiant, pour être sûr
11 $identifiant = 'constructeur_formulaire_'.$identifiant;
12 $contexte['_identifiant_session'] = $identifiant;
13
14 // On vérifie ce qui a été passé en paramètre
15 if (!is_array($formulaire_initial)) $formulaire_initial = array();
16
17 // On initialise la session si elle est vide
18 if (is_null($formulaire_actuel = session_get($identifiant))){
19 session_set($identifiant, $formulaire_initial);
20 $formulaire_actuel = $formulaire_initial;
21 }
22
23 // Si le formulaire actuel est différent du formulaire initial on agite un drapeau pour le dire
24 if ($formulaire_actuel != $formulaire_initial){
25 $contexte['formulaire_modifie'] = true;
26 }
27 $contexte['_message_attention'] = _T('saisies:construire_attention_modifie');
28
29 // On passe ça pour l'affichage
30 $contexte['_contenu'] = $formulaire_actuel;
31
32 // On passe ça pour la récup plus facile des champs
33 $contexte['_saisies_par_nom'] = saisies_lister_par_nom($formulaire_actuel);
34 // Pour déclarer les champs modifiables à CVT
35 foreach(array_keys($contexte['_saisies_par_nom']) as $nom){
36 $contexte["saisie_modifiee_$nom"] = array();
37 }
38
39 // La liste des saisies
40 $saisies_disponibles = saisies_lister_disponibles();
41 $contexte['_saisies_disponibles'] = $saisies_disponibles;
42
43 $contexte['fond_generer'] = 'formulaires/inc-generer_saisies_configurables';
44
45 // On cherche jquery UI pour savoir si on pourra glisser-déplacer
46 // SPIP 3 - jquery_ui
47 if (find_in_path('javascript/ui/jquery.ui.sortable.js') and find_in_path('javascript/ui/jquery.ui.draggable.js')){
48 $contexte['_chemin_ui'] = 'javascript/ui/';
49 }
50 // plugin jquery_ui >= 1.8
51 elseif (find_in_path('javascript/jquery-ui/ui/jquery.ui.sortable.js') and find_in_path('javascript/jquery-ui/ui/jquery.ui.draggable.js')){
52 $contexte['_chemin_ui'] = 'javascript/jquery-ui/ui/';
53 }
54 // plugin jquery_ui < 1.8
55 elseif (find_in_path('javascript/jquery-ui-1.8.16/ui/jquery.ui.sortable.js') and find_in_path('javascript/jquery-ui-1.8.16/ui/jquery.ui.draggable.js')){
56 $contexte['_chemin_ui'] = 'javascript/jquery-ui-1.8.16/ui/';
57 }
58 else{
59 $contexte['_chemin_ui'] = false;
60 }
61
62 return $contexte;
63 }
64
65 function formulaires_construire_formulaire_verifier($identifiant, $formulaire_initial=array(), $options=array()){
66 include_spip('inc/saisies');
67 $erreurs = array();
68 // l'une ou l'autre sera presente
69 $configurer_saisie = $enregistrer_saisie = '';
70
71 // Pas d'erreur si l'on ne demande rien
72 if (!($nom_ou_id = $configurer_saisie = _request('configurer_saisie')
73 OR $nom_ou_id = $enregistrer_saisie = _request('enregistrer_saisie'))) {
74 return $erreurs;
75 }
76
77 // On ajoute un préfixe devant l'identifiant
78 $identifiant = 'constructeur_formulaire_'.$identifiant;
79 // On récupère le formulaire à son état actuel
80 $formulaire_actuel = session_get($identifiant);
81
82 // On récupère les saisies actuelles, par identifiant ou par nom
83 if ($nom_ou_id[0] == '@') {
84 $saisies_actuelles = saisies_lister_par_identifiant($formulaire_actuel);
85 $nom = $saisies_actuelles[$nom_ou_id]['options']['nom'];
86 } else {
87 $saisies_actuelles = saisies_lister_par_nom($formulaire_actuel);
88 $nom = $nom_ou_id;
89 }
90 $noms_autorises = array_keys($saisies_actuelles);
91
92 // le nom (ou identifiant) doit exister
93 if (!in_array($nom_ou_id, $noms_autorises)) {
94 return $erreurs;
95 }
96
97 // La liste des saisies
98 $saisies_disponibles = saisies_lister_disponibles();
99
100 $saisie = $saisies_actuelles[$nom_ou_id];
101 $formulaire_config = $saisies_disponibles[$saisie['saisie']]['options'];
102 array_walk_recursive($formulaire_config, 'formidable_transformer_nom', "saisie_modifiee_${nom}[options][@valeur@]");
103 $formulaire_config = saisie_identifier(array('saisies'=>$formulaire_config));
104 $formulaire_config = $formulaire_config['saisies'];
105
106 // Si la saisie possede un identifiant, on l'ajoute
107 // au formulaire de configuration pour ne pas le perdre en route
108 if (isset($saisie['identifiant']) and $saisie['identifiant']) {
109 $formulaire_config = saisies_inserer(
110 $formulaire_config,
111 array(
112 'saisie' => 'hidden',
113 'options' => array(
114 'nom' => "saisie_modifiee_${nom}[identifiant]",
115 'defaut' => $saisie['identifiant']
116 ),
117 )
118 );
119 }
120
121 // S'il y a l'option adéquat, on ajoute le champ pour modifier le nom
122 if (isset($options['modifier_nom']) and $options['modifier_nom']
123 and $chemin_nom = saisies_chercher($formulaire_config, "saisie_modifiee_${nom}[options][description]", true))
124 {
125 $chemin_nom[] = 'saisies';
126 $chemin_nom[] = '0';
127
128 $formulaire_config = saisies_inserer(
129 $formulaire_config,
130 array(
131 'saisie' => 'input',
132 'options' => array(
133 'nom' => "saisie_modifiee_${nom}[options][nom]",
134 'label' => _T('saisies:option_nom_label'),
135 'explication' => _T('saisies:option_nom_explication'),
136 'obligatoire' => 'oui',
137 'size' => 50
138 ),
139 'verifier' => array(
140 'type' => 'regex',
141 'options' => array(
142 'modele' => '/^[\w]+$/'
143 )
144 )
145 ),
146 $chemin_nom
147 );
148 }
149
150 // liste des options de vérification
151 $verif_options = array();
152
153 // S'il y a un groupe "validation" alors on va construire le formulaire des vérifications
154 if ($chemin_validation = saisies_chercher($formulaire_config, "saisie_modifiee_${nom}[options][validation]", true)){
155 include_spip('inc/verifier');
156 $liste_verifications = verifier_lister_disponibles();
157 $chemin_validation[] = 'saisies';
158 $chemin_validation[] = 1000000; // à la fin
159
160 // On construit la saisie à insérer et les fieldset des options
161 $saisie_liste_verif = array(
162 'saisie' => 'selection',
163 'options' => array(
164 'nom' => "saisie_modifiee_${nom}[verifier][type]",
165 'label' => _T('saisies:construire_verifications_label'),
166 'option_intro' => _T('saisies:construire_verifications_aucune'),
167 'li_class' => 'liste_verifications',
168 'datas' => array()
169 )
170 );
171
172 foreach ($liste_verifications as $type_verif => $verif){
173 $saisie_liste_verif['options']['datas'][$type_verif] = $verif['titre'];
174 // Si le type de vérif a des options, on ajoute un fieldset
175 if (isset($verif['options']) and $verif['options'] and is_array($verif['options'])){
176 $groupe = array(
177 'saisie' => 'fieldset',
178 'options' => array(
179 'nom' => 'options',
180 'label' => $verif['titre'],
181 'li_class' => "$type_verif options_verifier"
182 ),
183 'saisies' => $verif['options']
184 );
185 array_walk_recursive($groupe, 'formidable_transformer_nom', "saisie_modifiee_${nom}[verifier][$type_verif][@valeur@]");
186 $verif_options[$type_verif] = $groupe;
187 }
188 }
189 $verif_options = array_merge(array($saisie_liste_verif), $verif_options);
190 }
191
192
193 if ($enregistrer_saisie){
194 // La saisie modifié
195 $saisie_modifiee = _request("saisie_modifiee_${nom}");
196 // On cherche les erreurs de la configuration
197 $vraies_erreurs = saisies_verifier($formulaire_config);
198 // Si on autorise à modifier le nom ET qu'il doit être unique : on vérifie
199 if (isset($options['modifier_nom']) and $options['modifier_nom']
200 and isset($options['nom_unique']) and $options['nom_unique'])
201 {
202 $nom_modifie = $saisie_modifiee['options']['nom'];
203 if ($nom_modifie != $enregistrer_saisie and saisies_chercher($formulaire_actuel, $nom_modifie))
204 $vraies_erreurs["saisie_modifiee_${nom}[options][nom]"] = _T('saisies:erreur_option_nom_unique');
205 }
206 // On regarde s'il a été demandé un type de vérif
207 if (isset($saisie_modifiee['verifier']['type'])
208 and (($type_verif = $saisie_modifiee['verifier']['type']) != '')
209 and $verif_options[$type_verif])
210 {
211 // On ne vérifie que les options du type demandé
212 $vraies_erreurs = array_merge($vraies_erreurs, saisies_verifier($verif_options[$type_verif]['saisies']));
213 }
214 }
215
216 // On insère chaque saisie des options de verification
217 if ($verif_options){
218 foreach ($verif_options as $saisie_verif){
219 $formulaire_config = saisies_inserer($formulaire_config, $saisie_verif, $chemin_validation);
220 }
221 }
222 $erreurs['configurer_'.$nom] = $formulaire_config;
223 $erreurs['positionner'] = '#configurer_'.$nom;
224
225 if ($enregistrer_saisie) {
226 if ($vraies_erreurs)
227 $erreurs = array_merge($erreurs, $vraies_erreurs);
228 else
229 $erreurs = array();
230 }
231
232 return $erreurs;
233 }
234
235 function formulaires_construire_formulaire_traiter($identifiant, $formulaire_initial=array(), $options=array()){
236 include_spip('inc/saisies');
237 $retours = array();
238 $saisies_disponibles = saisies_lister_disponibles();
239
240 // On ajoute un préfixe devant l'identifiant
241 $identifiant = 'constructeur_formulaire_'.$identifiant;
242 // On récupère le formulaire à son état actuel
243 $formulaire_actuel = session_get($identifiant);
244
245 // Si on demande à ajouter une saisie
246 if ($ajouter_saisie = _request('ajouter_saisie')){
247 $nom = saisies_generer_nom($formulaire_actuel, $ajouter_saisie);
248 $saisie = array(
249 'saisie' => $ajouter_saisie,
250 'options' => array(
251 'nom' => $nom
252 )
253 );
254 // S'il y a des valeurs par défaut pour ce type de saisie, on les ajoute
255 if (($defaut = $saisies_disponibles[$ajouter_saisie]['defaut']) and is_array($defaut)){
256 $defaut = _T_ou_typo($defaut, 'multi');
257
258 //Compatibilite PHP<5.3.0
259 //source : http://www.php.net/manual/en/function.array-replace-recursive.php#92574
260 if (!function_exists('array_replace_recursive'))
261 {
262 function array_replace_recursive($array, $array1)
263 {
264 function recurse($array, $array1)
265 {
266 foreach ($array1 as $key => $value)
267 {
268 // create new key in $array, if it is empty or not an array
269 if (!isset($array[$key]) || (isset($array[$key]) && !is_array($array[$key])))
270 {
271 $array[$key] = array();
272 }
273 // overwrite the value in the base array
274 if (is_array($value))
275 {
276 $value = recurse($array[$key], $value);
277 }
278 $array[$key] = $value;
279 }
280 return $array;
281 }
282
283 // handle the arguments, merge one by one
284 $args = func_get_args();
285 $array = $args[0];
286 if (!is_array($array))
287 {
288 return $array;
289 }
290 for ($i = 1; $i < count($args); $i++)
291 {
292 if (is_array($args[$i]))
293 {
294 $array = recurse($array, $args[$i]);
295 }
296 }
297 return $array;
298 }
299 }
300 $saisie = array_replace_recursive($saisie, $defaut);
301 }
302 $formulaire_actuel = saisies_inserer($formulaire_actuel, $saisie);
303 }
304
305 // Si on demande à dupliquer une saisie
306 if ($dupliquer_saisie = _request('dupliquer_saisie')) {
307 $formulaire_actuel = saisies_dupliquer($formulaire_actuel, $dupliquer_saisie);
308 }
309
310 // Si on demande à supprimer une saisie
311 if ($supprimer_saisie = _request('supprimer_saisie')){
312 $formulaire_actuel = saisies_supprimer($formulaire_actuel, $supprimer_saisie);
313 }
314
315 // Si on enregistre la conf d'une saisie
316 if ($nom = _request('enregistrer_saisie')){
317 // On récupère ce qui a été modifié
318 $saisie_modifiee = _request("saisie_modifiee_$nom");
319
320 // On regarde s'il y a une position à modifier
321 if (isset($saisie_modifiee['position'])){
322 $position = $saisie_modifiee['position'];
323 unset($saisie_modifiee['position']);
324 // On ne déplace que si ce n'est pas la même chose
325 if ($position != $nom)
326 $formulaire_actuel = saisies_deplacer($formulaire_actuel, $nom, $position);
327 }
328
329 // On regarde s'il y a des options de vérification à modifier
330 if (isset($saisie_modifiee['verifier']['type'])
331 and ($type_verif = $saisie_modifiee['verifier']['type']) != '')
332 {
333 $saisie_modifiee['verifier'] = array(
334 'type' => $type_verif,
335 'options' => $saisie_modifiee['verifier'][$type_verif]
336 );
337 }
338 else {
339 unset($saisie_modifiee['verifier']);
340 }
341
342 // On récupère les options postées en enlevant les chaines vides
343 $saisie_modifiee['options'] = array_filter($saisie_modifiee['options'], 'saisie_option_contenu_vide');
344 if (isset($saisie_modifiee['verifier']['options']) and $saisie_modifiee['verifier']['options']) {
345 $saisie_modifiee['verifier']['options'] = array_filter($saisie_modifiee['verifier']['options'], 'saisie_option_contenu_vide');
346 }
347
348 // On désinfecte à la main
349 if (is_array($saisie_modifiee['options']))
350 spip_desinfecte($saisie_modifiee['options']);
351
352 // On modifie enfin
353 $formulaire_actuel = saisies_modifier($formulaire_actuel, $nom, $saisie_modifiee);
354 }
355
356 // Si on demande à réinitialiser
357 if (_request('reinitialiser') == 'oui'){
358 $formulaire_actuel = $formulaire_initial;
359 }
360
361 // On enregistre en session la nouvelle version du formulaire
362 session_set($identifiant, $formulaire_actuel);
363
364 // Le formulaire reste éditable
365 $retours['editable'] = true;
366
367 return $retours;
368 }
369
370 // À utiliser avec un array_walk_recursive()
371 // Applique une transformation à la @valeur@ de tous les champs "nom" d'un formulaire, y compris loin dans l'arbo
372 function formidable_transformer_nom(&$valeur, $cle, $transformation){
373 if ($cle == 'nom' and is_string($valeur)){
374 $valeur = str_replace('@valeur@', $valeur, $transformation);
375 }
376 }
377
378 // Préparer une saisie pour la transformer en truc configurable
379 function formidable_generer_saisie_configurable($saisie, $env){
380 // On récupère le nom
381 $nom = $saisie['options']['nom'];
382 $identifiant = $saisie['identifiant'];
383 // On cherche si ya un formulaire de config
384 $formulaire_config = isset($env['erreurs']['configurer_'.$nom]) ? $env['erreurs']['configurer_'.$nom] : "";
385 // On ajoute une classe
386 if (!isset($saisie['options']['li_class'])) {
387 $saisie['options']['li_class'] = ''; // initialisation
388 }
389 $saisie['options']['li_class'] .= ' configurable';
390 // On ajoute l'option "tout_afficher"
391 $saisie['options']['tout_afficher'] = 'oui';
392
393 // On ajoute les boutons d'actions, mais seulement s'il n'y a pas de configuration de lancée
394 if (!$env['erreurs']) {
395 $saisie = saisies_inserer_html(
396 $saisie,
397 recuperer_fond(
398 'formulaires/inc-construire_formulaire-actions',
399 array(
400 'nom' => $nom,
401 'identifiant' => $identifiant,
402 'formulaire_config' => $formulaire_config,
403 'deplacable' => $env['_chemin_ui']
404 )
405 ),
406 'debut'
407 );
408 }
409
410 // On ajoute une ancre pour s'y déplacer
411 $saisie = saisies_inserer_html(
412 $saisie,
413 "\n<a id=\"configurer_$nom\"></a>\n",
414 'debut'
415 );
416
417 // Si ya un form de config on l'ajoute à la fin
418 if (is_array($formulaire_config)){
419 // On double l'environnement
420 $env2 = $env;
421 // On ajoute une classe
422 $saisie['options']['li_class'] .= ' en_configuration';
423
424 // Si possible on met en readonly
425 $saisie['options']['readonly'] = 'oui';
426
427 // On vire les sous-saisies s'il y en a
428 if (isset($saisie['saisies']) and $saisie['saisies'] and is_array($saisie['saisies'])){
429 $nb_champs_masques = count(saisies_lister_champs($saisie['saisies']));
430 $saisie['saisies'] = array(
431 array(
432 'saisie' => 'explication',
433 'options' => array(
434 'nom' => 'truc',
435 'texte' => _T('saisies:construire_info_nb_champs_masques', array('nb'=>$nb_champs_masques)),
436 )
437 )
438 );
439 }
440
441 // On va ajouter le champ pour la position
442 if (!($chemin_description = saisies_chercher($formulaire_config, "saisie_modifiee_${nom}[options][description]", true))){
443 $chemin_description = array(0);
444 $formulaire_config = saisies_inserer(
445 $formulaire_config,
446 array(
447 'saisie' => 'fieldset',
448 'options' => array(
449 'nom' => "saisie_modifiee_${nom}[options][description]",
450 'label' => _T('saisies:option_groupe_description')
451 ),
452 'saisies' => array()
453 ),
454 0
455 );
456 }
457 $chemin_description[] = 'saisies';
458 $chemin_description[] = '0'; // tout au début
459 $formulaire_config = saisies_inserer(
460 $formulaire_config,
461 array(
462 'saisie' => 'position_construire_formulaire',
463 'options' => array(
464 'nom' => "saisie_modifiee_${nom}[position]",
465 'label' => _T('saisies:construire_position_label'),
466 'explication' => _T('saisies:construire_position_explication'),
467 'formulaire' => $env['_contenu'],
468 'saisie_a_positionner' => $nom
469 )
470 ),
471 $chemin_description
472 );
473
474 $env2['saisies'] = $formulaire_config;
475
476 // Un test pour savoir si on prend le _request ou bien
477 $erreurs_test = $env['erreurs'];
478 unset($erreurs_test['configurer_'.$nom]);
479 unset($erreurs_test['positionner']);
480 if ($erreurs_test){
481 // Là aussi on désinfecte à la main
482 if (is_array($env2["saisie_modifiee_$nom"]['options']))
483 spip_desinfecte($env2["saisie_modifiee_$nom"]['options']);
484 }
485 else{
486 $env2["saisie_modifiee_$nom"] = $env2['_saisies_par_nom'][$nom];
487 // il n'y a pas toujours de verification...
488 if (isset($env2["saisie_modifiee_$nom"]['verifier'])) {
489 $env2["saisie_modifiee_$nom"]['verifier'][ $env2["saisie_modifiee_$nom"]['verifier']['type'] ]
490 = $env2["saisie_modifiee_$nom"]['verifier']['options'];
491 }
492 }
493
494 $env2['fond_generer'] = 'inclure/generer_saisies';
495 $saisie = saisies_inserer_html(
496 $saisie,
497 '<div class="formulaire_configurer"><ul class="formulaire_configurer-contenus">'
498 .recuperer_fond(
499 'inclure/generer_saisies',
500 $env2
501 )
502 .'<li class="boutons">
503 <input type="hidden" name="enregistrer_saisie" value="'.$nom.'" />
504 <button type="submit" class="submit link" name="enregistrer_saisie" value="">'._T('bouton_annuler').'</button>
505 <input type="submit" class="submit" name="enregistrer" value="'._T('bouton_valider').'" />
506 </li>'
507 .'</ul></div>',
508 'fin'
509 );
510 }
511 // On génère le HTML de la saisie
512 $html = saisies_generer_html($saisie, $env);
513 return $html;
514 }
515
516 /**
517 * Callback d'array_filter()
518 * Permet de retourner tout ce qui n'est pas un contenu vide.
519 * La valeur '0' est par contre retournée.
520 *
521 * @param $var La variable a tester
522 * @return bool L'accepte-t-on ?
523 **/
524 function saisie_option_contenu_vide($var) {
525 if (!$var) {
526 if (is_string($var) AND strlen($var)) {
527 return true;
528 }
529 return false;
530 }
531 return true;
532 }
533 ?>