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