Zend_Form

Zend_Form démarrage rapide

Ce guide rapide couvre les bases de la création, de la validation, et du rendu des formulaires Zend_Form.

Créer un objet de formulaire

Instanciez simplement un objet Zend_Form :

  1. $form = new Zend_Form;

Pour des usages avancés, vous voudriez probablement dériver Zend_Form, mais pour les formulaires simples, vous pouvez créez un formulaire depuis une instance de Zend_Form.

Vous pouvez spécifier (c'est une bonne idée) l'action et la méthode d'envoi du formulaire grâce à setAction() et setMethod() :

  1. $form->setAction('/resource/process')
  2.      ->setMethod('post');

Le code ci-dessus indique au formulaire d'être envoyé vers l'URL "/resource/process" avec la méthode HTTP POST. Ceci va impacter le rendu du formulaire (la balise <form>).

Il est possible d'assigner les attributs HTML supplémentaires à la balise <form> via la méthode setAttrib() ou encore setAttribs(). Par exemple, indiquons un attribut "id" au formulaire :

  1. $form->setAttrib('id', 'login');

Ajouter des éléments au formulaire

Un formulaire ne sert à rien sans éléments. Le composant Zend_Form est fourni avec un ensemble d'éléments qui rendent du code XHTML via les aides de Zend_View. Voici ces aides :

  • button

  • checkbox (ou plusieurs checkboxes avec multiCheckbox)

  • hidden

  • image

  • password

  • radio

  • reset

  • select (régulier ou à selection multiple)

  • submit

  • text

  • textarea

Vous avez 2 manières de procéder pour ajouter les éléments au formulaire : instanciez vous même les objets des éléments, ou passer le type d'élément à Zend_Form, qui va alors créer le bon objet pour vous.

Quelques exemples :

  1. // Ajout d'un objet élément :
  2. $form->addElement(new Zend_Form_Element_Text('username'));
  3.  
  4. // Passage d'un texte décrivant le futur objet élément, à Zend_Form :
  5. $form->addElement('text', 'username');

Par défaut, ces éléments n'ont ni validateurs, ni filtres. Vous devrez donc ajoutez des validateurs et/ou des filtres, manuellement. Ceci est possible soit (a) avant de passer l'élément au formulaire, (b) via les options de configuration passés lors de la création de l'élément, ou (c) en récupérant l'objet déjà enregistré, depuis le formulaire, et en le configurant ensuite.

Voyons comment passer un validateur à un élément dont nous créons l'objet. On peut passer soit l'objet Zend_Validate_*, soit une chaîne le décrivant :

  1. $username = new Zend_Form_Element_Text('username');
  2.  
  3. // Passage d'un objet Zend_Validate_*:
  4. $username->addValidator(new Zend_Validate_Alnum());
  5.  
  6. // Passage du nom du validateur:
  7. $username->addValidator('alnum');

En utilisant la technique de passage par le nom, vous pouvez ajouter un tableau d'options à passer au constructeur de l'objet validateur. Ceci se fait en troisième paramètre :

  1. // Passage d'options au validateur
  2. $username->addValidator('regex', false, array('/^[a-z]/i'));

(Le second paramètre permet d'indiquer au validateur s'il doit briser la chaîne de validation ou non. Par défaut, FALSE : ce n'est donc pas le cas.)

Vous pouvez avoir besoin de spécifier qu'un élément est requis. Ceci peut être fait en utilisant un accesseur ou en passant une option à la création de l'élément. Voici un exemple :

  1. // Cet élément est requis:
  2. $username->setRequired(true);

Lorsqu'un élément est requis, un validateur "NotEmpty" lui est ajouté, sur le dessus de sa pile de validateurs.

La gestion des filtres est très semblable à celle des validateurs. Voyons comment ajouter un filtre qui retourne la donnée en minuscules :

  1. $username->addFilter('StringtoLower');

Finalement, la configuration complète de l'élément pourra ressembler à cela :

  1. $username->addValidator('alnum')
  2.          ->addValidator('regex', false, array('/^[a-z]/'))
  3.          ->setRequired(true)
  4.          ->addFilter('StringToLower');
  5.  
  6. // ou, de manière plus compacte:
  7. $username->addValidators(array('alnum',
  8.         array('regex', false, '/^[a-z]/i')
  9.     ))
  10.     ->setRequired(true)
  11.     ->addFilters(array('StringToLower'));

Aussi simple que cela puisse paraître, cela peut très vite devenir fastidieux de répéter ces opérations sur tous les éléments du formulaire. Reprenons le cas (b) d'au dessus : lorsque l'on crée un élément, Zend_Form::addElement() agit comme une fabrique et on peut lui passer des options de configuration. Par exemple, des validateurs ou des filtres. Essayons ceci :

  1. $form->addElement('text', 'username', array(
  2.     'validators' => array(
  3.         'alnum',
  4.         array('regex', false, '/^[a-z]/i')
  5.     ),
  6.     'required' => true,
  7.     'filters'  => array('StringToLower'),
  8. ));

Note: Si vous vous apercevez que vous créez des éléments basés sur les mêmes options, étendre Zend_Form_Element peut devenir une bonne option. Votre nouvelle classe configurera directement vos objets.

Rendre (visuellement) un formulaire

Rendre un formulaire est très simple. La plupart des éléments nécessitent les aides de vue Zend_View pour être rendus. Ils ont donc besoin d'un objet de vue. Pour rendre un formulaire, appelez sa méthode render() ou faites un echo devant l'objet.

  1. // Appel explicite de render() :
  2. echo $form->render($view);
  3.  
  4. // Supposant que setView() avec passage d'un objet Zend_View a été appelée avant :
  5. echo $form;

Par défaut, Zend_Form et les Zend_Form_Element vont essayer de récupérer l'objet de vue depuis l'aide d'action ViewRenderer, ce qui signifie que vous n'aurez pas besoin de spécifier un objet de vue manuellement si vous utilisez le système MVC de Zend Framework. Pour rendre un formulaire dans une vue MVC, un simple echo suffit :

  1. <?php echo $this->form ?>

Techniquement, Zend_Form utilise des "décorateurs" pour effectuer le rendu visuel. Ces décorateurs peuvent remplacer le contenu, ou le placer avant ou après. Ils peuvent aussi introspecter l'élément qui leur est passé. Ainsi, vous pouvez chaîner plusieurs décorateurs pour utiliser des effets visuels. Par défaut, Zend_Form_Element combine quatre décorateurs pour s'afficher :

  1. $element->addDecorators(array(
  2.     'ViewHelper',
  3.     'Errors',
  4.     array('HtmlTag', array('tag' => 'dd')),
  5.     array('Label', array('tag' => 'dt')),
  6. ));

(Où <HELPERNAME> est le nom de l'aide de vue à utiliser, qui varie selon l'élément à rendre.)

Les décorateurs par défaut (rappelés ci-dessus), produisent le rendu suivant :

  1. <dt><label for="username" class="required">Username</dt>
  2. <dd>
  3.     <input type="text" name="username" value="123-abc" />
  4.     <ul class="errors">
  5.         <li>'123-abc' has not only alphabetic and digit characters</li>
  6.         <li>'123-abc' does not match against pattern '/^[a-z]/i'</li>
  7.     </ul>
  8. </dd>

(Le formatage peut un peu changer.)

Vous pouvez changer les décorateurs utilisés par un élément si vous voulez avoir un visuel différent ; voyez la section sur les décorateurs pour plus d'informations.

Le formulaire boucle sur ses éléments et entoure leur rendu d'une balise HTML <form>. Cette balise prend en compte la méthode, l'action, et les éventuels attributs passés via setAttribs().

Les éléments sont bouclés dans l'ordre dans lequel ils sont ajoutés, ou, si votre élément possède un attribut "order", celui-ci sera alors utilisé pour gérer sa place dans la pile des éléments :

  1. $element->setOrder(10);

Ou encore, à la création de l'élément via addElement() :

  1. $form->addElement('text', 'username', array('order' => 10));

Vérifier qu'un formulaire est valide

Après l'envoi du formulaire, il faut vérifier les valeurs que contiennent ses éléments. Tous les validateurs de chaque élément sont donc interrogés. Si l'élément était marqué comme requis et que l'élément ne reçoit aucune donnée, les validateurs suivants agiront sur une valeur NULL.

D'où proviennent les données ? Vous pouvez utiliser $_POST ou $_GET, ou n'importe quelle source de données (service Web par exemple) :

  1. if ($form->isValid($_POST)) {
  2.     // succès!
  3. } else {
  4.     // echec!
  5. }

Avec des requêtes AJAX, il arrive que l'on ait besoin de ne valider qu'un élément, ou un groupe d'élément. isValidPartial() validera un formulaire partiel. Contrairement à isValid(), si une valeur est absente, les autres validateurs ne seront pas interrogés :

  1. if ($form->isValidPartial($_POST)) {
  2.     // Tous les éléments présents dans $_POST ont passé la validation
  3. } else {
  4.     // un ou plusieurs éléments présent dans $_POST ont échoué
  5. }

La méthode processAjax() peut aussi être utilisée pour valider partiellement un formulaire. Contrairement à isValidPartial(), cette méthode retournera les messages d'erreur de validation au format JSON.

En supposant que les validateurs aient passé, vous pouvez dès lors récupérer les valeurs filtrées depuis les éléments :

  1. $values = $form->getValues();

Si vous désirez les valeurs non filtrées, utilisez :

  1. $unfiltered = $form->getUnfilteredValues();

Si d'un autre côté, vous ne souhaitez que les valeurs filtrées valides d'un formulaire partiellement valide, vous pouvez appeler :

  1. $values = $form->getValidValues($_POST);

Les statuts d'erreur

Votre formulaire a échoué à l'envoi ? Dans la plupart des cas, vous voudrez rendre à nouveau le formulaire, mais avec les messages d'erreur affichés :

  1. if (!$form->isValid($_POST)) {
  2.     echo $form;
  3.  
  4.     // ou en assignant un objet de vue (cas non-MVC typiquement)
  5.     $this->view->form = $form;
  6.     return $this->render('form');
  7. }

Si vous voulez inspecter les erreurs, 2 méthodes s'offrent à vous. getErrors() retourne un tableau associatif avec en clé le nom de l'élément et en valeur un tableau de codes d'erreurs. getMessages() retourne un tableau associatif avec en clé le nom de l'élément, et en valeur un tableau de messages d'erreurs (code=>message). Tout élément ne comportant pas d'erreur ne sera pas inclus dans le tableau.

Assembler le tout ensemble

Créons un formulaire de d'authentification. Il aura besoin d'élément représentant :

  • un nom

  • un mot de passe

  • un bouton d'envoi

Pour notre exemple, imaginons un nom composé de caractères alphanumériques uniquement. Le nom commencera par une lettre, et devra faire entre 6 et 20 caractères de long, qui seront normalisés en lettres minuscules. Les mots de passe feront 6 caractères minimum.

Nous allons utiliser la puissance de Zend_Form pour configurer le formulaire et effectuer le rendu :

  1. $form = new Zend_Form();
  2. $form->setAction('/user/login')
  3.      ->setMethod('post');
  4.  
  5. // élément nom :
  6. $username = $form->createElement('text', 'username');
  7. $username->addValidator('alnum')
  8.          ->addValidator('regex', false, array('/^[a-z]+/'))
  9.          ->addValidator('stringLength', false, array(6, 20))
  10.          ->setRequired(true)
  11.          ->addFilter('StringToLower');
  12.  
  13. // élément mot de passe :
  14. $password = $form->createElement('password', 'password');
  15. $password->addValidator('StringLength', false, array(6))
  16.          ->setRequired(true);
  17.  
  18. // Ajout des éléments au formulaire
  19. $form->addElement($username)
  20.      ->addElement($password)
  21.      // addElement() agit comme une fabrique qui crée un bouton 'Login'
  22.      ->addElement('submit', 'login', array('label' => 'Login'));

Il nous faut à présent un contrôleur pour gérer tout cela :

  1. class UserController extends Zend_Controller_Action
  2. {
  3.     public function getForm()
  4.     {
  5.         // Créer le formulaire comme décrit ci-dessus
  6.         return $form;
  7.     }
  8.  
  9.     public function indexAction()
  10.     {
  11.         // rend user/form.phtml
  12.         $this->view->form = $this->getForm();
  13.         $this->render('form');
  14.     }
  15.  
  16.     public function loginAction()
  17.     {
  18.         if (!$this->getRequest()->isPost()) {
  19.             return $this->_forward('index');
  20.         }
  21.         $form = $this->getForm();
  22.         if (!$form->isValid($_POST)) {
  23.             // Validation en echec
  24.             $this->view->form = $form;
  25.             return $this->render('form');
  26.         }
  27.  
  28.         $values = $form->getValues();
  29.         // les valeurs sont récupérées
  30.     }
  31. }

Le script de vue pour afficher le formulaire :

  1. <h2>Identifiez vous:</h2>
  2. <?php echo $this->form ?>

Comme vous le voyez sur le code du contrôleur, il reste du travail à faire une fois le formulaire validé. Par exemple, utiliser Zend_Auth pour déclencher un processus d'identification.

Utiliser un objet Zend_Config

Toutes les classes du composant Zend_Form sont configurables au moyen d'un objet Zend_Config ; vous pouvez passer un objet Zend_Config au constructeur ou via la méthode setConfig(). Voyons comment créer le formulaire ci-dessus, au moyen d'un fichier INI. Tout d'abord, nous nous baserons sur une section "developement", et nos instructions devront être imbriquées afin de refléter la configuration. Ensuite nous utiliserons un espace de nom "user" correspondant au contrôleur, puis un "login" concernant le formulaire (ceci permet de ranger les données correctement dans le fichier INI) :

  1. [development]
  2. ; informations générales du formulaire
  3. user.login.action = "/user/login"
  4. user.login.method = "post"
  5.  
  6. ; element username
  7. user.login.elements.username.type = "text"
  8. user.login.elements.username.options.validators.alnum.validator = "alnum"
  9. user.login.elements.username.options.validators.regex.validator = "regex"
  10. user.login.elements.username.options.validators.regex.options.pattern = "/^[a-z]/i"
  11. user.login.elements.username.options.validators.strlen.validator = "StringLength"
  12. user.login.elements.username.options.validators.strlen.options.min = "6"
  13. user.login.elements.username.options.validators.strlen.options.max = "20"
  14. user.login.elements.username.options.required = true
  15. user.login.elements.username.options.filters.lower.filter = "StringToLower"
  16.  
  17. ; element password
  18. user.login.elements.password.type = "password"
  19. user.login.elements.password.options.validators.strlen.validator = "StringLength"
  20. user.login.elements.password.options.validators.strlen.options.min = "6"
  21. user.login.elements.password.options.required = true
  22.  
  23. ; element submit
  24. user.login.elements.submit.type = "submit"

Le constructeur du formulaire ressemblera alors à ceci :

  1. $config = new Zend_Config_Ini($configFile, 'development');
  2. $form   = new Zend_Form($config->user->login);

et tout le formulaire sera défini.

Conclusion

Vous êtes maintenant capable de libérer la puissance de Zend_Form. Continuez de lire les chapitres suivants pour utiliser ce composant en profondeur !


Zend_Form