Zend_Filter_Input

Zend_Filter_Inflector

Zend_Filter_Inflector est un outil de conversion de règles (sous forme de chaîne de caractères), vers une cible. Ce procédé est appelé inflexion.

Par exemple, transformer des MotsEncasseMélangée ou des motsEnCamelCase vers un chemin. Vous pourriez avoir besoin de passer les caractères en minuscules, et séparer les mots en utilisant un tiret ("-"). Un inflecteur sert à ceci.

Zend_Filter_Inflector implémente Zend_Filter_Interface ; pour utiliser l'inflexion, vous appelez filter() sur votre instance.

Example #1 Transformer du texteEncasseMelangée ou du texteCamelCase vers un autre format

  1. $inflector = new Zend_Filter_Inflector('pages/:page.:suffix');
  2. $inflector->setRules(array(
  3.     ':page'  => array('Word_CamelCaseToDash', 'StringToLower'),
  4.     'suffix' => 'html'
  5. ));
  6.  
  7. $string   = 'motsEnNotationCamel';
  8. $filtered = $inflector->filter(array('page' => $string));
  9. // pages/mots-en-notation-camel.html
  10.  
  11. $string   = 'ceci_n_est_pas_en_notation_camel';
  12. $filtered = $inflector->filter(array('page' => $string));
  13. // pages/ceci_n_est_pas_en_notation_camel.html

Opération

Un inflecteur a besoin d'une cible et d'une ou plusieurs règles. Une cible est une chaîne dans laquelle des jokers pour les variables à remplacer sont présents. Ils sont préfixés de doubles-points, par exemple :script.

L'appel à filter(), nécessite un tableau de clés/valeurs pour les jokers présents dans la cible.

Chaque variable dans la cible peut avoir zéro ou plusieurs règles associées. Les règles peuvent être statiques ou faire référence à une classe Zend_Filter. Les règles statiques sont des remplacements purs et simples. Sinon, la classe qui correspond à la règle sera utilisée pour analyser le texte. Ces classes sont spécifiées par leur nom (du filtre), non préfixé.

Par exemple, vous pouvez utiliser n'importe quelle instance de Zend_Filter. Cependant, plutôt que d'y faire référence via "Zend_Filter_Alpha" ou "Zend_Filter_StringToLower", vous spécifierez seulement "Alpha" ou encore "StringToLower".

Créer des chemins vers des filtres alternatifs

Zend_Filter_Inflector utilise Zend_Loader_PluginLoader pour gérer les filtres chargés. Par défaut, n'importe quel filtre préfixé par Zend_Filter sera disponible. Pour accéder aux filtres ayant d'autres préfixes plus profonds, enlevez leur préfixe "Zend_Filter" tout simplement :

  1. // utilise Zend_Filter_Word_CamelCaseToDash comme règle
  2. $inflector->addRules(array('script' => 'Word_CamelCaseToDash'));

Pour spécifier d'autres chemins, Zend_Filter_Inflector possède une méthode qui proxie vers le plugin loader, addFilterPrefixPath() :

  1. $inflector->addFilterPrefixPath('Mes_Filtres', 'Mes/Filtres/');

Il est possible également de récupérer le plugin loader, et d'intervenir sur son instance de manière directe :

  1. $loader = $inflector->getPluginLoader();
  2. $loader->addPrefixPath('Mes_Filtres', 'Mes/Filtres/');

Pour plus d'informations sur la modification des chemins vers les filtres voyez la documentation de PluginLoader.

Paramétrer la cible de l'inflecteur

La cible de l'inflecteur est un mot joker (ou un identifiant), précédé du caractère spécial, double-point. ":script", ":path", etc. La méthode filter() cherche ces identifiants pour les traiter (les remplacer).

Il est possible de changer le caractère spécial double-point avec setTargetReplacementIdentifier(), ou en troisième paramètre de constructeur :

  1. // Via le constructeur :
  2. $inflector = new Zend_Filter_Inflector('#foo/#bar.#sfx', null, '#');
  3.  
  4. // Via l'accesseur :
  5. $inflector->setTargetReplacementIdentifier('#');

En général, concernant la cible, on la passe en constructeur. C'est le cas classique. Il peut être en revanche nécessaire de pouvoir passer une cible après la construction de l'objet. (Par exemple modifier l'inflecteur des composants Zend intégrés tels que ViewRenderer ou Zend_Layout). setTarget() peut vous y aider :

  1. $inflector = $layout->getInflector();
  2. $inflector->setTarget('layouts/:script.phtml');

De plus, vous pouvez agréger la cible dans un membre de votre classe, si cela vous permet d'éviter trop d'appels de méthodes. setTargetReference() permet ceci :

  1. class Foo
  2. {
  3.     /**
  4.      * @var string Inflector target
  5.      */
  6.     protected $_target = 'foo/:bar/:baz.:suffix';
  7.  
  8.     /**
  9.      * Constructor
  10.      * @return void
  11.      */
  12.     public function __construct()
  13.     {
  14.         $this->_inflector = new Zend_Filter_Inflector();
  15.         $this->_inflector->setTargetReference($this->_target);
  16.     }
  17.  
  18.     /**
  19.      * Set target; updates target in inflector
  20.      *
  21.      * @param  string $target
  22.      * @return Foo
  23.      */
  24.     public function setTarget($target)
  25.     {
  26.         $this->_target = $target;
  27.         return $this;
  28.     }
  29. }

Règles d'inflexion

Comme dit en introduction, il existe 2 types de règles : statiques et basées sur des filtres.

Note: Notez bien que quelle que soit la méthode que vous utilisez pour spécifier vos règles dans l'inflecteur, leur ordre est très important. Vous devez ajouter de la règle la plus spécifique, à la plus générique. Par exemple, 2 règles nommées "moduleDir" et "module", la règle "moduleDir" devrait être ajoutée avant la règle "module", car cette dernière est contenue dans "moduleDir".

Règles statiques

Les règles statiques permettent des remplacements simples. Si vous avez un segment statique dans votre cible, ce type de règle est idéal. setStaticRule() permet de les manipuler :

  1. $inflector = new Zend_Filter_Inflector(':script.:suffix');
  2. $inflector->setStaticRule('suffix', 'phtml');
  3.  
  4. // ici des opérations ...
  5. // changement de la règle:
  6. $inflector->setStaticRule('suffix', 'php');

Bien sur il est possible d'agréger la règle dans une propriété de classe, ceci permettra d'éviter l'appel de méthodes. Ce cas se produit typiquement lorsque l'inflecteur est embarqué (encapsulé) dans une classe. Vous pouvez à ce moment là interdire la récupération de l'inflecteur depuis l'extérieur de la classe, par exemple. La méthode setStaticRuleReference() vous y aidera :

  1. class Foo
  2. {
  3.     /**
  4.      * @var string Suffix
  5.      */
  6.     protected $_suffix = 'phtml';
  7.  
  8.     /**
  9.      * Constructor
  10.      * @return void
  11.      */
  12.     public function __construct()
  13.     {
  14.         $this->_inflector =
  15.             new Zend_Filter_Inflector(':script.:suffix');
  16.         $this->_inflector
  17.              ->setStaticRuleReference('suffix', $this->_suffix);
  18.     }
  19.  
  20.     /**
  21.      * Set suffix; updates suffix static rule in inflector
  22.      *
  23.      * @param  string $suffix
  24.      * @return Foo
  25.      */
  26.     public function setSuffix($suffix)
  27.     {
  28.         $this->_suffix = $suffix;
  29.         return $this;
  30.     }
  31. }

Règles non statiques : basées sur des filtres

Des filtres de type Zend_Filter peuvent être utilisés comme règles dans l'inflecteur. Ils sont donc liés à des variables cibles, mais vous pouvez lier plusieurs filtres pour une même cible. Ils sont alors procédés dans l'ordre (FIFO), prenez donc garde à ceci.

Les règles des filtres sont ajoutées avec setFilterRule(). Cette méthode écrase toute règle déjà définie. addFilterRule() au contraire, n'écrase pas mais gère une pile de filtres pour une variable. Les noms des filtres passés à ces 2 méthodes sont de la forme :

  • String : une chaîne de caractères représentant le nom de la classe du filtre, ou alors le nom de la classe moins le préfixe utilisé par le plugin loader. (le préfixe par défaut étant "Zend_Filter").

  • Objet filtre : une instance d'objet implémentant Zend_Filter_Interface.

  • Array : un tableau de chaînes ou d'objets.

  1. $inflector = new Zend_Filter_Inflector(':script.:suffix');
  2.  
  3. // Affecte une règle pour utiliser le filtre
  4. //Zend_Filter_Word_CamelCaseToDash
  5. $inflector->setFilterRule('script', 'Word_CamelCaseToDash');
  6.  
  7. // Ajoute une règle vers un filtre de casse minuscule
  8. $inflector->addFilterRule('script', new Zend_Filter_StringToLower());
  9.  
  10. // Affectation de plusieurs règles d'un coup
  11. $inflector->setFilterRule('script', array(
  12.     'Word_CamelCaseToDash',
  13.     new Zend_Filter_StringToLower()
  14. ));

Paramétrer plusieurs règles en une seule fois

En temps normal il est plus pratique de spécifier ses règles (statiques et/ou filtres) en une seule fois, plutôt qu'en plusieurs étapes. Les méthodes de Zend_Filter_Inflector comme addRules() et setRules() permettent ceci.

Chacune de ces 2 méthodes prend en paramètre un tableau de variable/règle. La règle peut être n'importe quel type accepté (string, objet filtre ou array). Les noms des variables proposent une syntaxe spéciale pour différencier les règles statiques des filtres :

  • ":" prefix : règle à base de filtre.

  • Pas de prefix : règle statique.

Example #2 Paramétrer plusieurs règles en une seule fois

  1. // setRules() accepte la même notation :
  2. $inflector->addRules(array(
  3.     // règles filtres:
  4.     ':controller' => array('CamelCaseToUnderscore','StringToLower'),
  5.     ':action'     => array('CamelCaseToUnderscore','StringToLower'),
  6.  
  7.     // règles statiques :
  8.     'suffix'      => 'phtml'
  9. ));

Autres méthodes utilitaires

Zend_Filter_Inflector possède d'autres méthodes pour changer le plugin loader, manipuler des règles, et contrôler les exceptions.

  • setPluginLoader() peut être utile si vous avez configuré votre propre chargeur de plugins (plugin loader) et que vous voulez l'utiliser avec Zend_Filter_Inflector ; getPluginLoader() retourne cette valeur.

  • setThrowTargetExceptionsOn() accepte un booléen. Ceci spécifie qu'une exception doit être lancée si une variable est toujours présente dans la cible après le passage de l'inflecteur. Par défaut, ça n'est pas le cas. isThrowTargetExceptionsOn() retourne la valeur actuelle.

  • getRules($spec = null) récupère toutes les règles, ou les règles d'une certaine variable.

  • getRule($spec, $index) récupère une règle précise, même dans une chaîne de filtre.$index doit être précisé.

  • clearRules() va effacer toutes les règles fixées préalablement.

Zend_Config avec Zend_Filter_Inflector

Zend_Config peut être utilisé pour spécifier les règles, les préfixes des filtres et d'autres choses dans vos inflecteurs. Passez un objet Zend_Config au constructeur ou à la méthode setConfig() :

  • target définit la cible de l'inflecteur.

  • filterPrefixPath définit le préfixe/chemins des filtres.

  • throwTargetExceptionsOn est un booléen. Ceci spécifie qu'une exception doit être lancée si une variable est toujours présente dans la cible après le passage de l'inflecteur.

  • targetReplacementIdentifier spécifie le caractère à utiliser pour définir les variables de remplacement.

  • rules spécifie un tableau de règles, comme accepté par addRules().

Example #3 Utiliser Zend_Config avec Zend_Filter_Inflector

  1. // Par le constructeur :
  2. $config    = new Zend_Config($options);
  3. $inflector = new Zend_Filter_Inflector($config);
  4.  
  5. // Ou via setConfig() :
  6. $inflector = new Zend_Filter_Inflector();
  7. $inflector->setConfig($config);

Zend_Filter_Input