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