AutoComplete
Beaucoup de librairies javascript AJAX propose une fonctionnalité dite
d'auto-complétion. Une liste de résultats possibles est chargée au fur et à mesure que
l'utilisateur saisit. L'aide AutoComplete est destinée à simplifier
le retour de ces valeurs vers la librairie Javascript.
Toutes les librairies JS n'implémentant pas l'auto-complétion de la même manière,
l'aide AutoComplete propose une solution abstraite, ainsi que des
implémentations concrètes pour certaines librairies. Les types de valeur de retour sont en
général des tableaux de chaînes JSON, des tableaux de tableaux
JSON (avec chaque membre étant un tableau associatif de métadonnées
utilisées pour créer la liste de sélection), ou du HTML.
L'utilisation basique ressemble à ceci :
_helper->autoCompleteDojo($data);
// Ou :
$response = $this->_helper
->autoCompleteDojo
->sendAutoCompletion($data);
// Ou alors prépare simplement les données :
$response = $this->_helper
->autoCompleteDojo
->prepareAutoCompletion($data);
}
}
]]>
Par défaut, l'auto-complétion :
Désactive les layouts et le ViewRenderer.
Affecte des en-têtes de réponse appropriés.
Remplit le corps de la réponse avec les données d'auto-complétion
encodées/formatées.
Envoie la réponse.
Les méthodes disponibles sont :
disableLayouts() est utilisée pour désactiver les layouts
et le ViewRenderer. Cette méthode est appelées par
prepareAutoCompletion().
encodeJson($data, $keepLayouts = false) va encoder les
données en JSON. Cette méthode est appelées par
prepareAutoCompletion().
prepareAutoCompletion($data, $keepLayouts = false)
prépare les données dans le format de réponse nécessaire à une implémentation
concrète. La valeur de retour va changer en fonction de l'implémentation
(de la librairie utilisée).
sendAutoCompletion($data, $keepLayouts = false) Va appeler
prepareAutoCompletion(), puis envoyer la réponse.
direct($data, $sendNow = true, $keepLayouts = false) est
une méthode utilisée par le gestionnaire d'aides (helper broker). La valeur de
$sendNow va déterminer si c'est
sendAutoCompletion() ou
prepareAutoCompletion(), qui doit être appelée.
Actuellement, AutoComplete supporte les librairies
AJAX Dojo et Scriptaculous.
AutoCompletion avec Dojo
Dojo n'a pas une fonctionnalité d'auto-complétion, mais deux :
ComboBox et FilteringSelect. Dans les deux cas, elle demande
une structure de données qui implémente QueryReadStore ; voyez la
documentation de dojo.data
Dans Zend Framework, vous pouvez passer un simple tableau indexé à l'aide
AutoCompleteDojo, elle retournera une réponse JSON
compatible avec la structure de données Dojo :
_helper->autoCompleteDojo($data);
]]>
AutoCompletion avec Dojo en utilisant MVC
L'auto-complétion avec Dojo via MVC requière plusieurs choses :
générer un objet formulaire sur le ComboBox sur lequel vous voulez de
l'auto-complétion, un contrôleur avec une action pour servir les résultats, la
création d'un QueryReadStore à connecter à l'action et la génération
du javascript à utiliser pour initialiser l'auto-complétion coté serveur.
Voyons le javascript nécessaire. Dojo est une librairie complète pour la création
de javascript OO, un peu comme Zend Framework pour
PHP. Il est possible de créer des pseudo-namespaces en utilisant
l'arborescence des répertoires. Nous allons créer un répertoire "custom" au même
niveau que le répertoire Dojo. A l'intérieur, nous allons créer un fichier
javascript, TestNameReadStore.js, avec le contenu
suivant :
Cette classe est une simple extension de QueryReadStore, qui est
une classe abstraite. Nous définissons simplement une méthode de requête, et on lui
assigne notre élément "test".
Ensuite, créons le formulaire sur lequel nous souhaitons une auto-complétion :
_form) {
require_once 'Zend/Form.php';
$this->_form = new Zend_Form();
$this->_form->setMethod('get')
->setAction($this->getRequest()->getBaseUrl()
. '/test/process')
->addElements(array(
'test' => array('type' => 'text', 'options' => array(
'filters' => array('StringTrim'),
'dojoType' => array('dijit.form.ComboBox'),
'store' => 'testStore',
'autoComplete' => 'false',
'hasDownArrow' => 'true',
'label' => 'Your input:',
)),
'go' => array('type' => 'submit',
'options' => array('label' => 'Go!'))
));
}
return $this->_form;
}
}
]]>
Ici, nous créons simplement un formulaire avec des méthodes "test" et "go".
La méthode "test" ajoute plusieurs attributs Dojo spéciaux : dojoType,
store, autoComplete, et hasDownArrow.
dojoType est utilisé pour indiquer la création d'une
ComboBox, et nous allons la relier au conteneur de données
("store") de "testStore". Mettre
"autoComplete" à FALSE dit à Dojo de ne pas sélectionner
automatiquement la première valeur, mais de plutôt montrer une liste de valeurs
possibles. Enfin, "hasDownArrow" crée une flèche bas comme sur les
select box.
Ajoutons une méthode pour afficher le formulaire, et une entrée pour traiter
l'auto-complétion :
view->form = $this->getForm();
}
public function autocompleteAction()
{
if ('ajax' != $this->_getParam('format', false)) {
return $this->_helper->redirector('index');
}
if ($this->getRequest()->isPost()) {
return $this->_helper->redirector('index');
}
$match = trim($this->getRequest()->getQuery('test', ''));
$matches = array();
foreach ($this->getData() as $datum) {
if (0 === strpos($datum, $match)) {
$matches[] = $datum;
}
}
$this->_helper->autoCompleteDojo($matches);
}
}
]]>
Dans autocompleteAction(), nous vérifions que nous avons
bien une requête post, et un paramètre "format" avec la valeur
"ajax". Ensuite, nous vérifions la présence d'un paramètre
"test", et le comparons avec nos données. (getData()
retourne des données quelconques). Enfin, nous envoyons nos résultats à notre aide
AutoCompletion.
Voyons maintenant notre script de vue. Nous devons configurer notre entrepôt
de données, puis rendre le formulaire, et s'assurer que les librairies Dojo
appropriées sont bien chargées (ainsi que notre entrepôt). Voici le script de vue
:
form ?>
headStyle()->captureStart() ?>
@import "baseUrl()
?>/javascript/dijit/themes/tundra/tundra.css";
@import "baseUrl() ?>/javascript/dojo/resources/dojo.css";
headStyle()->captureEnd() ?>
headScript()
->setAllowArbitraryAttributes(true)
->appendFile($this->baseUrl() . '/javascript/dojo/dojo.js',
'text/javascript',
array('djConfig' => 'parseOnLoad: true'))
->captureStart() ?>
djConfig.usePlainJson=true;
dojo.registerModulePath("custom","../custom");
dojo.require("dojo.parser");
dojo.require("dojox.data.QueryReadStore");
dojo.require("dijit.form.ComboBox");
dojo.require("custom.TestNameReadStore");
headScript()->captureEnd() ?>
]]>
Notez les appels aux aides de vue comme headStyle et headScript ; celles-ci
sont des emplacements réservés, que nous pouvons ensuite utiliser pour effectuer
le rendu dans la section "head" du HTML de votre script de
layout.
Nous pouvons dès lors faire fonctionner l'auto-complétion Dojo.
AutoCompletion avec Scriptaculous
Scriptaculous
attend une réponse HTML dans un format spécifique.
Utilisez l'aide "AutoCompleteScriptaculous". Passez lui un tableau
de données et l'aide créera une réponse HTML compatible avec
"Ajax.Autocompleter".