Zend_Controller-ActionHelpers-AutoComplete.xml 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <!-- EN-Revision: 15617 -->
  3. <!-- Reviewed: no -->
  4. <sect3 id="zend.controller.actionhelpers.autocomplete">
  5. <title>AutoComplete</title>
  6. <para>
  7. Beaucoup de librairies javascript AJAX propose une fonctionnalité dite
  8. d'auto-complétion. Une liste de résultats possibles est chargée par AJAX au fur et à mesure
  9. que l'utilisateur saisit. L'aide <code>AutoComplete</code> est destinée à simplifier le
  10. retour de ces valeurs vers la librairie Javascript.
  11. </para>
  12. <para>
  13. Toutes les librairies JS n'implémentant pas l'auto-complétion de la même manière,
  14. l'aide <code>AutoComplete</code> propose une solution abstraite, ainsi que des
  15. implémentations concrètes pour certaines librairies. Les types de valeur de retour sont en
  16. général des tableaux de chaînes JSON, des tableaux de tableaux JSON, ou du HTML.
  17. </para>
  18. <para>L'utilisation basique ressemble à ceci :</para>
  19. <programlisting language="php"><![CDATA[
  20. class FooController extends Zend_Controller_Action
  21. {
  22. public function barAction()
  23. {
  24. // Ici du travail ....
  25. // Encode et envoie la réponse
  26. $this->_helper->autoCompleteDojo($data);
  27. // Ou :
  28. $response = $this->_helper
  29. ->autoCompleteDojo
  30. ->sendAutoCompletion($data);
  31. // Ou alors prépare simplement les données :
  32. $response = $this->_helper
  33. ->autoCompleteDojo
  34. ->prepareAutoCompletion($data);
  35. }
  36. }
  37. ]]></programlisting>
  38. <para>Par défaut, l'auto-complétion :</para>
  39. <itemizedlist>
  40. <listitem>
  41. <para>Désactive les layouts et le ViewRenderer.</para>
  42. </listitem>
  43. <listitem>
  44. <para>Affecte des en-têtes de réponse appropriés.</para>
  45. </listitem>
  46. <listitem>
  47. <para>
  48. Remplit le corps de la réponse avec les données d'auto-complétion
  49. encodées/formatées.
  50. </para>
  51. </listitem>
  52. <listitem>
  53. <para>Envoie la réponse.</para>
  54. </listitem>
  55. </itemizedlist>
  56. <para>Les méthodes disponibles sont :</para>
  57. <itemizedlist>
  58. <listitem>
  59. <para>
  60. <code>disableLayouts()</code> est utilisée pour désactiver les layouts et le
  61. ViewRenderer. Cette méthode est appelées par
  62. <code>prepareAutoCompletion()</code>.
  63. </para>
  64. </listitem>
  65. <listitem>
  66. <para>
  67. <code>encodeJson($data, $keepLayouts = false)</code> va encoder les données
  68. en JSON. Cette méthode est appelées par
  69. <code>prepareAutoCompletion()</code>.
  70. </para>
  71. </listitem>
  72. <listitem>
  73. <para>
  74. <code>prepareAutoCompletion($data, $keepLayouts = false)</code> prépare les
  75. données dans le format de réponse nécessaire à une implémentation concrète. La
  76. valeur de retour va changer en fonction de l'implémentation (de la librairie AJAX
  77. utilisée).
  78. </para>
  79. </listitem>
  80. <listitem>
  81. <para>
  82. <code>sendAutoCompletion($data, $keepLayouts = false)</code> Va appeler
  83. <code>prepareAutoCompletion()</code>, puis envoyer la réponse.
  84. </para>
  85. </listitem>
  86. <listitem>
  87. <para>
  88. <code>direct($data, $sendNow = true, $keepLayouts = false)</code> est une
  89. méthode utilisée par le gestionnaire d'aides (helper broker). La valeur de
  90. <code>$sendNow</code> va déterminer si c'est <code>sendAutoCompletion()</code> ou
  91. <code>prepareAutoCompletion()</code>, qui doit être appelée.
  92. </para>
  93. </listitem>
  94. </itemizedlist>
  95. <para>
  96. Actuellement, <code>AutoComplete</code> supporte les librairies AJAX Dojo et
  97. Scriptaculous.
  98. </para>
  99. <sect4 id="zend.controller.actionhelpers.autocomplete.dojo">
  100. <title>AutoCompletion avec Dojo</title>
  101. <para>
  102. Dojo n'a pas une fonctionnalité d'auto-complétion, mais deux :
  103. <code>ComboBox</code> et <code>FilteringSelect</code>. Dans les deux cas, elle demande
  104. une structure de données qui implémente <code>QueryReadStore</code> ; voyez la
  105. documentation de
  106. <ulink url="http://dojotoolkit.org/book/dojo-book-0-9/part-3-programmatic-dijit-and-dojo/data-retrieval-dojo-data-0">
  107. dojo.data</ulink>
  108. </para>
  109. <para>
  110. Dans Zend Framework, vous pouvez passer un simple tableau indexé à l'aide
  111. <code>AutoCompleteDojo</code>, elle retournera une réponse JSON compatible avec la
  112. structure de données Dojo :
  113. </para>
  114. <programlisting language="php"><![CDATA[
  115. // à l'intérieur d'une action de contrôleur :
  116. $this->_helper->autoCompleteDojo($data);
  117. ]]></programlisting>
  118. <example id="zend.controller.actionhelpers.autocomplete.dojo.example1">
  119. <title>AutoCompletion avec Dojo en utilisant MVC</title>
  120. <para>
  121. L'auto-complétion avec Dojo via MVC requière plusieurs choses: générer un
  122. objet formulaire sur le <code>ComboBox</code> sur lequel vous voulez de
  123. l'auto-complétion, un contrôleur avec une action pour servir les résultats, la
  124. création d'un <code>QueryReadStore</code> à connecter à l'action et la génération
  125. du javascript à utiliser pour initialiser l'auto-complétion coté serveur.
  126. </para>
  127. <para>
  128. Voyons le javascript nécessaire. Dojo est une librairie qui propose des
  129. objets haut niveau pour l'auto-complétion (entre autres), un peu comme Zend
  130. Framework pour PHP. Il est possible de créer des pseudo-namespaces en utilisant
  131. l'arborescence des répertoires. Nous allons créer un répertoire "custom" au même
  132. niveau que le répertoire Dojo. A l'intérieur, nous allons créer un fichier
  133. javascript, <filename>TestNameReadStore.js</filename>, avec le contenu
  134. suivant&#160;:
  135. </para>
  136. <programlisting language="javascript"><![CDATA[
  137. dojo.provide("custom.TestNameReadStore");
  138. dojo.declare("custom.TestNameReadStore",
  139. dojox.data.QueryReadStore,
  140. {
  141. fetch:function (request) {
  142. request.serverQuery = { test:request.query.name };
  143. return this.inherited("fetch", arguments);
  144. }
  145. });
  146. ]]></programlisting>
  147. <para>
  148. Cette classe est une simple extension de <code>QueryReadStore</code>, qui est
  149. une classe abstraite. Nous définissons simplement une méthode de requête, et on lui
  150. assigne notre élément "test".
  151. </para>
  152. <para>
  153. Ensuite, créons le formulaire sur lequel nous souhaitons une auto-complétion&#160;:
  154. </para>
  155. <programlisting language="php"><![CDATA[
  156. class TestController extends Zend_Controller_Action
  157. {
  158. protected $_form;
  159. public function getForm()
  160. {
  161. if (null === $this->_form) {
  162. require_once 'Zend/Form.php';
  163. $this->_form = new Zend_Form();
  164. $this->_form->setMethod('get')
  165. ->setAction($this->getRequest()->getBaseUrl()
  166. . '/test/process')
  167. ->addElements(array(
  168. 'test' => array('type' => 'text', 'options' => array(
  169. 'filters' => array('StringTrim'),
  170. 'dojoType' => array('dijit.form.ComboBox'),
  171. 'store' => 'testStore',
  172. 'autoComplete' => 'false',
  173. 'hasDownArrow' => 'true',
  174. 'label' => 'Your input:',
  175. )),
  176. 'go' => array('type' => 'submit',
  177. 'options' => array('label' => 'Go!'))
  178. ));
  179. }
  180. return $this->_form;
  181. }
  182. }
  183. ]]></programlisting>
  184. <para>
  185. Ici, nous créons simplement un formulaire avec des méthodes "test" et "go".
  186. La méthode "test" ajoute plusieurs attributs Dojo spéciaux : <code>dojoType</code>,
  187. <code>store</code>, <code>autoComplete</code>, et <code>hasDownArrow</code>.
  188. <code>dojoType</code> est utilisé pour indiquer la création d'une
  189. <code>ComboBox</code>, et nous allons la relier au conteneur de données
  190. ("<code>store</code>") de "<code>testStore</code>". Mettre
  191. "<code>autoComplete</code>" à <code>false</code> dit à Dojo de ne pas sélectionner
  192. automatiquement la première valeur, mais de plutôt montrer une liste de valeurs
  193. possibles. Enfin, "<code>hasDownArrow</code>" crée une flèche bas comme sur les
  194. select box.
  195. </para>
  196. <para>
  197. Ajoutons une méthode pour afficher le formulaire, et une entrée pour traiter
  198. l'auto-complétion&#160;:
  199. </para>
  200. <programlisting language="php"><![CDATA[
  201. class TestController extends Zend_Controller_Action
  202. {
  203. // ...
  204. /**
  205. * Landing page
  206. */
  207. public function indexAction()
  208. {
  209. $this->view->form = $this->getForm();
  210. }
  211. public function autocompleteAction()
  212. {
  213. if ('ajax' != $this->_getParam('format', false)) {
  214. return $this->_helper->redirector('index');
  215. }
  216. if ($this->getRequest()->isPost()) {
  217. return $this->_helper->redirector('index');
  218. }
  219. $match = trim($this->getRequest()->getQuery('test', ''));
  220. $matches = array();
  221. foreach ($this->getData() as $datum) {
  222. if (0 === strpos($datum, $match)) {
  223. $matches[] = $datum;
  224. }
  225. }
  226. $this->_helper->autoCompleteDojo($matches);
  227. }
  228. }
  229. ]]></programlisting>
  230. <para>
  231. Dans <code>autocompleteAction()</code>, nous vérifions que nous avons bien
  232. une requête post, et un paramètre "<code>format</code>" avec la valeur
  233. "<code>ajax</code>". Ensuite, nous vérifions la présence d'un paramètre
  234. "<code>test</code>", et le comparons avec nos données. (<code>getData()</code>
  235. retourne des données quelconques.).Enfin, nous envoyons nos résultats à notre aide
  236. <code>AutoCompletion</code>.
  237. </para>
  238. <para>
  239. Voyons maintenant notre script de vue. Nous devons configurer notre entrepôt
  240. de données, puis rendre le formulaire, et s'assurer que les librairies Dojo
  241. appropriées sont bien chargées (ainsi que notre entrepôt). Voici le script de vue
  242. :
  243. </para>
  244. <programlisting language="php"><![CDATA[
  245. <?php // configuration de l'entrepôt de données : ?>
  246. <div dojoType="custom.TestNameReadStore" jsId="testStore"
  247. url="<?php echo $this->baseUrl() ?>/unit-test/autocomplete/format/ajax"
  248. requestMethod="get"></div>
  249. <?php // rendu du formulaire : ?>
  250. <?php echo $this->form ?>
  251. <?php // configuration des CSS de Dojo dans le head HTML : ?>
  252. <?php $this->headStyle()->captureStart() ?>
  253. @import "<?php echo $this->baseUrl()
  254. ?>/javascript/dijit/themes/tundra/tundra.css";
  255. @import "<?php echo $this->baseUrl() ?>/javascript/dojo/resources/dojo.css";
  256. <?php $this->headStyle()->captureEnd() ?>
  257. <?php // configuration de javascript pour charger
  258. // les librairies Dojo dans le head HTML : ?>
  259. <?php $this->headScript()
  260. ->setAllowArbitraryAttributes(true)
  261. ->appendFile($this->baseUrl() . '/javascript/dojo/dojo.js',
  262. 'text/javascript',
  263. array('djConfig' => 'parseOnLoad: true'))
  264. ->captureStart() ?>
  265. djConfig.usePlainJson=true;
  266. dojo.registerModulePath("custom","../custom");
  267. dojo.require("dojo.parser");
  268. dojo.require("dojox.data.QueryReadStore");
  269. dojo.require("dijit.form.ComboBox");
  270. dojo.require("custom.TestNameReadStore");
  271. <?php $this->headScript()->captureEnd() ?>
  272. ]]></programlisting>
  273. <para>
  274. Notez les appels aux aides de vue comme <code>headStyle</code> et
  275. <code>headScript</code>.
  276. </para>
  277. <para>Nous pouvons dès lors faire fonctionner l'auto-complétion Dojo.</para>
  278. </example>
  279. </sect4>
  280. <sect4 id="zend.controller.actionhelpers.autocomplete.scriptaculous">
  281. <title>AutoCompletion avec Scriptaculous</title>
  282. <para>
  283. <ulink url="http://wiki.script.aculo.us/scriptaculous/show/Ajax.Autocompleter">
  284. Scriptaculous</ulink>attend une réponse HTML dans un format spécifique.
  285. </para>
  286. <para>
  287. Utilisez l'aide "<code>AutoCompleteScriptaculous</code>". Passez lui un tableau
  288. de données et l'aide créera une réponse HTML compatible avec
  289. <code>Ajax.Autocompleter</code>.
  290. </para>
  291. </sect4>
  292. </sect3>