Créer et rendre des éléments composites

Dans la dernière section, nous avions un exemple traitant un élément "date de naissance":

  1. <div class="element">
  2.     <?php echo $form->dateOfBirth->renderLabel() ?>
  3.     <?php echo $this->formText('dateOfBirth[day]', '', array(
  4.         'size' => 2, 'maxlength' => 2)) ?>
  5.     /
  6.     <?php echo $this->formText('dateOfBirth[month]', '', array(
  7.         'size' => 2, 'maxlength' => 2)) ?>
  8.     /
  9.     <?php echo $this->formText('dateOfBirth[year]', '', array(
  10.         'size' => 4, 'maxlength' => 4)) ?>
  11. </div>

Comment représenteriez-vous cet élément en tant que Zend_Form_Element? Comment écrire un décorateur qui s'assure de son rendu ?

L'élément

Les questions à se poser sur le fonctionnement de l'élément sont:

  • Comment affecter et récupérer une valeur?

  • Comment valider la valeur?

  • Comment proposer l'affectation personnalisée d'une valeur composées de trois segments (jour, mois, année)?

Les deux première questions se positionnent sur l'élément de formulaire lui-même, comment vont fonctionner les méthodes setValue() et getValue()? L'autre question nous suggère de nous questionner sur comment récupérer les segments représentant la date, ou comment les affecter dans l'élément?

La solution est de surcharger la méthode setValue() dans l'élément pour proposer sa propre logique. Dans le cas de notre exemple, notre élément devrait avoir trois comportements distincts:

  • Si un timestamp entier est utilisé, il doit aider à la détermination des entités jour, mois, année.

  • Si une chaine est utilisée, elle devrait être transformée en timestamp, et cette valeur sera utiliser pour déterminer les entités jour, mois, année.

  • Si un tableau contenant les clés jour, mois, année est utilisé, alors les valeurs doivent être stockées.

En interne, les jour, mois et année seront stockés distinctement. Lorsque la valeur de l'élément sera demandée, nous récupèrerons une chaine formatée et normalisée. Nous surchargerons getValue() pour assembler les segments élémentaires composant la date.

Voici à quoi ressemblerait la classe:

  1. class My_Form_Element_Date extends Zend_Form_Element_Xhtml
  2. {
  3.     protected $_dateFormat = '%year%-%month%-%day%';
  4.     protected $_day;
  5.     protected $_month;
  6.     protected $_year;
  7.  
  8.     public function setDay($value)
  9.     {
  10.         $this->_day = (int) $value;
  11.         return $this;
  12.     }
  13.  
  14.     public function getDay()
  15.     {
  16.         return $this->_day;
  17.     }
  18.  
  19.     public function setMonth($value)
  20.     {
  21.         $this->_month = (int) $value;
  22.         return $this;
  23.     }
  24.  
  25.     public function getMonth()
  26.     {
  27.         return $this->_month;
  28.     }
  29.  
  30.     public function setYear($value)
  31.     {
  32.         $this->_year = (int) $value;
  33.         return $this;
  34.     }
  35.  
  36.     public function getYear()
  37.     {
  38.         return $this->_year;
  39.     }
  40.  
  41.     public function setValue($value)
  42.     {
  43.         if (is_int($value)) {
  44.             $this->setDay(date('d', $value))
  45.                  ->setMonth(date('m', $value))
  46.                  ->setYear(date('Y', $value));
  47.         } elseif (is_string($value)) {
  48.             $date = strtotime($value);
  49.             $this->setDay(date('d', $date))
  50.                  ->setMonth(date('m', $date))
  51.                  ->setYear(date('Y', $date));
  52.         } elseif (is_array($value)
  53.                   && (isset($value['day'])
  54.                       && isset($value['month'])
  55.                       && isset($value['year'])
  56.                   )
  57.         ) {
  58.             $this->setDay($value['day'])
  59.                  ->setMonth($value['month'])
  60.                  ->setYear($value['year']);
  61.         } else {
  62.             throw new Exception('Valeur de date invalide');
  63.         }
  64.  
  65.         return $this;
  66.     }
  67.  
  68.     public function getValue()
  69.     {
  70.         return str_replace(
  71.             array('%year%', '%month%', '%day%'),
  72.             array($this->getYear(), $this->getMonth(), $this->getDay()),
  73.             $this->_dateFormat
  74.         );
  75.     }
  76. }

Cette classe est fléxible : nous pouvons affecter les valeurs par défaut depuis une base de données et être certains qu'elles seront stockées correctement. Aussi, la valeur peut être affectée depuis un tableau provenant des entrées du formulaire. Enfin, nous avons tous les accesseurs distincts pour chaque segment de la date, un décorateur pourra donc créer l'élément comme il le voudra.

Le décorateur

Toujours en suivant notre exemple, imaginons que nous voulions que notre utilisateur saisissent chaque segment jour, mois, année séparément. Heureusement, PHP permet d'utiliser la notation tableau pour créer des éléments, ainsi nous pourrons capturer ces trois valeurs en une seule et nous crérons un élément Zend_Form traitant avec des valeurs en tableau.

Le décorateur est relativement simple: Il va récupérer le jour, le mois et l'année de l'élément et passer chaque valeur à une aide de vue qui rendra chaque champ individuellement. Nous les rassemblerons ensuite dans le rendu final.

  1. class My_Form_Decorator_Date extends Zend_Form_Decorator_Abstract
  2. {
  3.     public function render($content)
  4.     {
  5.         $element = $this->getElement();
  6.         if (!$element instanceof My_Form_Element_Date) {
  7.             // Nous ne rendons que des éléments Date
  8.             return $content;
  9.         }
  10.  
  11.         $view = $element->getView();
  12.         if (!$view instanceof Zend_View_Interface) {
  13.             // Nous utilisons des aides de vue, si aucune vue n'existe
  14.             // nous ne rendons rien
  15.             return $content;
  16.         }
  17.  
  18.         $day   = $element->getDay();
  19.         $month = $element->getMonth();
  20.         $year  = $element->getYear();
  21.         $name  = $element->getFullyQualifiedName();
  22.  
  23.         $params = array(
  24.             'size'      => 2,
  25.             'maxlength' => 2,
  26.         );
  27.         $yearParams = array(
  28.             'size'      => 4,
  29.             'maxlength' => 4,
  30.         );
  31.  
  32.         $markup = $view->formText($name . '[day]', $day, $params)
  33.                 . ' / ' . $view->formText($name . '[month]', $month, $params)
  34.                 . ' / ' . $view->formText($name . '[year]', $year, $yearParams);
  35.  
  36.         switch ($this->getPlacement()) {
  37.             case self::PREPEND:
  38.                 return $markup . $this->getSeparator() . $content;
  39.             case self::APPEND:
  40.             default:
  41.                 return $content . $this->getSeparator() . $markup;
  42.         }
  43.     }
  44. }

Il faut maintenant préciser à notre élément d'utiliser notre décorateur par défaut. Pour ceci, il faut informer l'élément du chemin vers notre décorateur. Nous pouvons effectuer ceci par le constructeur:

  1. class My_Form_Element_Date extends Zend_Form_Element_Xhtml
  2. {
  3.     // ...
  4.  
  5.     public function __construct($spec, $options = null)
  6.     {
  7.         $this->addPrefixPath(
  8.             'My_Form_Decorator',
  9.             'My/Form/Decorator',
  10.             'decorator'
  11.         );
  12.         parent::__construct($spec, $options);
  13.     }
  14.  
  15.     // ...
  16. }

Notez que l'on fait cela en constructeur et non dans la méthode init(). Ceci pour deux raisons. D'abord, ceci permet d'étendre dans le futur notre élément afin d'y ajouter de la logique dans init sans se soucier de l'appel à parent::init(). Ensuite, celà permet aussi de redéfinir le décorateur par défaut Date dans le futur si celà devient nécessaire, via le constructeur ou la méthode init.

Ensuite, nous devons réécrire la méthode loadDefaultDecorators() pour lui indiquer d'utiliser notre décorateur Date:

  1. class My_Form_Element_Date extends Zend_Form_Element_Xhtml
  2. {
  3.     // ...
  4.  
  5.     public function loadDefaultDecorators()
  6.     {
  7.         if ($this->loadDefaultDecoratorsIsDisabled()) {
  8.             return;
  9.         }
  10.  
  11.         $decorators = $this->getDecorators();
  12.         if (empty($decorators)) {
  13.             $this->addDecorator('Date')
  14.                  ->addDecorator('Errors')
  15.                  ->addDecorator('Description', array(
  16.                      'tag'   => 'p',
  17.                      'class' => 'description'
  18.                  ))
  19.                  ->addDecorator('HtmlTag', array(
  20.                      'tag' => 'dd',
  21.                      'id'  => $this->getName() . '-element'
  22.                  ))
  23.                  ->addDecorator('Label', array('tag' => 'dt'));
  24.         }
  25.     }
  26.  
  27.     // ...
  28. }

A qyuoi ressemble le rendu final ? Considérons l'élément suivant:

  1. $d = new My_Form_Element_Date('dateOfBirth');
  2. $d->setLabel('Date de naissance: ')
  3.   ->setView(new Zend_View());
  4.  
  5. // Ces deux procédés sont équivalents:
  6. $d->setValue('20 April 2009');
  7. $d->setValue(array('year' => '2009', 'month' => '04', 'day' => '20'));

Si vous affichez cet élément, vous obtiendrez ce rendu (avec quelques modifications concernant la mise en page du manuel et sa lisibilité):

  1. <dt id="dateOfBirth-label"><label for="dateOfBirth" class="optional">
  2.     Date de naissance:
  3. </label></dt>
  4. <dd id="dateOfBirth-element">
  5.     <input type="text" name="dateOfBirth[day]" id="dateOfBirth-day"
  6.         value="20" size="2" maxlength="2"> /
  7.     <input type="text" name="dateOfBirth[month]" id="dateOfBirth-month"
  8.         value="4" size="2" maxlength="2"> /
  9.     <input type="text" name="dateOfBirth[year]" id="dateOfBirth-year"
  10.         value="2009" size="4" maxlength="4">
  11. </dd>

Conclusion

Nous avons maintenant un élément qui peut rendre de multiples champs de formulaire, et les traiter comme une seule entité -- la valeur dateOfBirth sera passée comme un tableau à l'élément et celui-ci créra les segments de date appropriés et retournera une valeur normalisée.

Aussi, nous pouvons toujours utiliser des décorateurs différents avec l'élément. Si nous avions voulu utiliser un décorateur » Dojo DateTextBox -- qui accepte et retourne des chaines -- we aurions pu, sans modification sur l'élément lui-même.

Enfin, vous avez une API uniforme pour décrire un élement se composant se plusieurs segments distincts.

blog comments powered by Disqus