Zend_Form-QuickStart.xml 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658
  1. <sect1 id="zend.form.quickstart">
  2. <title>Inicio rápido a Zend_Form</title>
  3. <para>
  4. Esta guía rápida pretende cubrir los fundamentos para
  5. crear, validar y presentar formularios usando <code>Zend_Form</code>
  6. </para>
  7. <sect2 id="zend.form.quickstart.create">
  8. <title>Creando un objeto formulario</title>
  9. <para>
  10. Crear un objeto de formulario es muy simple: solo instancíe
  11. <code>Zend_Form</code>
  12. </para>
  13. <programlisting role="php"><![CDATA[
  14. $form = new Zend_Form;
  15. ]]>
  16. </programlisting>
  17. <para>
  18. Para casos de uso avanzados, es posible desee crear una subclase de
  19. <code>Zend_Form</code>, pero para formularios simples, puede
  20. programar la creación de un formulario usando un objeto
  21. <code>Zend_Form</code>
  22. </para>
  23. <para>
  24. Si desea especificar el action y method del formulario (siempre
  25. buenas ideas), puede hacer uso de los accesos
  26. <code>setAction()</code> y <code>setMethod()</code>:
  27. </para>
  28. <programlisting role="php"><![CDATA[
  29. $form->setAction('/resource/process')
  30. ->setMethod('post');
  31. ]]>
  32. </programlisting>
  33. <para>
  34. El código de arriba establece el action del formulario a la URL
  35. parcial "/resource/process" y como method HTTP POST. Esto se
  36. mostrará en la presentación final.
  37. </para>
  38. <para>
  39. Usted puede establecer atributos HTML adicionales para la etiqueta
  40. <code>&lt;form&gt;</code> mediante el uso de los métodos
  41. setAttrib() o setAttribs(). Por ejemplo, si desea especificar el id
  42. establezca el atributo "id":
  43. </para>
  44. <programlisting role="php"><![CDATA[
  45. $form->setAttrib('id', 'login');
  46. ]]>
  47. </programlisting>
  48. </sect2>
  49. <sect2 id="zend.form.quickstart.elements">
  50. <title>Añadir elementos al formulario</title>
  51. <para>
  52. Un formulario no es nada sin sus elementos. <code>Zend_Form</code>
  53. contiene de manera predeterminada algunos elementos que generan
  54. XHTML vía auxiliares <code>Zend_View</code>. Son los
  55. siguientes:
  56. </para>
  57. <itemizedlist>
  58. <listitem><para>
  59. button
  60. </para></listitem>
  61. <listitem><para>
  62. checkbox (o varios checkboxes a la vez con multiCheckbox)
  63. </para></listitem>
  64. <listitem><para>
  65. hidden
  66. </para></listitem>
  67. <listitem><para>
  68. image
  69. </para></listitem>
  70. <listitem><para>
  71. password
  72. </para></listitem>
  73. <listitem><para>
  74. radio
  75. </para></listitem>
  76. <listitem><para>
  77. reset
  78. </para></listitem>
  79. <listitem><para>
  80. select (tanto regulares como de multi-selección)
  81. </para></listitem>
  82. <listitem><para>
  83. submit
  84. </para></listitem>
  85. <listitem><para>
  86. text
  87. </para></listitem>
  88. <listitem><para>
  89. textarea
  90. </para></listitem>
  91. </itemizedlist>
  92. <para>
  93. Tiene dos opciones para añadir elementos a un formulario; puede
  94. instanciar elementos concretos y pasarlos como objetos, o
  95. simplemente puede pasar el tipo de elemento y <code>Zend_Form</code>
  96. instaciará por usted un objeto del tipo correspondiente.
  97. </para>
  98. <para>
  99. Algunos ejemplos:
  100. </para>
  101. <programlisting role="php"><![CDATA[
  102. // Instanciando un elemento y pasandolo al objeto form:
  103. $form->addElement(new Zend_Form_Element_Text('username'));
  104. // Pasando el tipo de elemento del formulario al objeto form:
  105. $form->addElement('text', 'username');
  106. ]]>
  107. </programlisting>
  108. <para>
  109. De manera predeterminada, éstos no tienen validadores o filtros.
  110. Esto significa que tendrá que configurar sus elementos con un
  111. mínimo de validadores, y potencialmente filtros. Puede hacer esto
  112. (a) antes de pasar el elemento al formulario, (b) vía opciones de
  113. configuración pasadas cuando crea un elemento a través de
  114. <code>Zend_Form</code>, o (c) recuperar el elemento del objeto form
  115. y configurándolo posteriormente.
  116. </para>
  117. <para>
  118. Veamos primero la creación de validadores para la instancia de
  119. un elemento concreto. Puede pasar objetos
  120. <code>Zend_Validate_*</code> o el nombre de un validador para utilizar:
  121. </para>
  122. <programlisting role="php"><![CDATA[
  123. $username = new Zend_Form_Element_Text('username');
  124. // Pasando un objeto Zend_Validate_*:
  125. $username->addValidator(new Zend_Validate_Alnum());
  126. // Pasando el nombre de un validador:
  127. $username->addValidator('alnum');
  128. ]]>
  129. </programlisting>
  130. <para>
  131. Cuando se utiliza esta segunda opción, si el constructor del
  132. validador acepta argumentos, se pueden pasar en un array
  133. como tercer parámetro:
  134. </para>
  135. <programlisting role="php"><![CDATA[
  136. // Pasando un patrón
  137. $username->addValidator('regex', false, array('/^[a-z]/i'));
  138. ]]>
  139. </programlisting>
  140. <para>
  141. (El segundo parámetro se utiliza para indicar si el fallo
  142. debería prevenir la ejecución de validadores posteriores o no; por
  143. defecto, el valor es false.)
  144. </para>
  145. <para>
  146. Puede también desear especificar un elemento como requerido. Esto
  147. puede hacerse utilizando un método de acceso o pasando una opción al
  148. crear el elemento. En el primer caso:
  149. </para>
  150. <programlisting role="php"><![CDATA[
  151. // Hace este elemento requerido:
  152. $username->setRequired(true);
  153. ]]>
  154. </programlisting>
  155. <para>
  156. Cuando un elemento es requerido, un validador 'NotEmpty' (NoVacio)
  157. es añadido a la parte superior de la cadena de validaciones,
  158. asegurando que el elemento tenga algún valor cuando sea requerido.
  159. </para>
  160. <para>
  161. Los filtros son registrados básicamente de la misma manera que los
  162. validadores. Para efectos ilustrativos, vamos a agregar un filtro
  163. para poner en minúsculas el valor final:
  164. </para>
  165. <programlisting role="php"><![CDATA[
  166. $username->addFilter('StringtoLower');
  167. ]]>
  168. </programlisting>
  169. <para>
  170. Entonces, la configuración final de nuestro elemento queda así:
  171. </para>
  172. <programlisting role="php"><![CDATA[
  173. $username->addValidator('alnum')
  174. ->addValidator('regex', false, array('/^[a-z]/'))
  175. ->setRequired(true)
  176. ->addFilter('StringToLower');
  177. // o, de manera más compacta:
  178. $username->addValidators(array('alnum',
  179. array('regex', false, '/^[a-z]/i')
  180. ))
  181. ->setRequired(true)
  182. ->addFilters(array('StringToLower'));
  183. ]]>
  184. </programlisting>
  185. <para>
  186. Tan simple como esto, realizarlo para cada uno de los elementos
  187. del formulario puede resultar un poco tedioso. Intentemos la opción
  188. (b) arriba mencionada. Cuando creamos un nuevo elemento utilizando
  189. <code>Zend_Form::addElement()</code> como fábrica, opcionalmente
  190. podemos pasar las opciones de configuración. Éstas pueden incluir
  191. validadores y los filtros que se van a utilizar. Por lo tanto, para hacer todo
  192. lo anterior implícitamente, intente lo siguiente:
  193. </para>
  194. <programlisting role="php"><![CDATA[
  195. $form->addElement('text', 'username', array(
  196. 'validators' => array(
  197. 'alnum',
  198. array('regex', false, '/^[a-z]/i')
  199. ),
  200. 'required' => true,
  201. 'filters' => array('StringToLower'),
  202. ));
  203. ]]>
  204. </programlisting>
  205. <note><para>
  206. Si encuentra que está asignando elementos con las mismas opciones en
  207. varios lugares, podría considerar crear su propia subclase de
  208. <code>Zend_Form_Element</code> y utilizar ésta; a largo plazo le
  209. permitirá escribir menos.
  210. </para></note>
  211. </sect2>
  212. <sect2 id="zend.form.quickstart.render">
  213. <title>Generar un formulario</title>
  214. <para>
  215. Generar un formulario es simple. La mayoría de los elementos
  216. utilizan un auxiliar de <code>Zend_View</code> para generarse a sí
  217. mismos, por lo tanto necesitan un objeto vista con el fin de
  218. generarse. Además, tiene dos opciones: usar el método render()
  219. del formulario, o simplemente mostrarlo con echo.
  220. </para>
  221. <programlisting role="php"><![CDATA[
  222. // Llamando a render() explicitamente, y pasando un objeto vista opcional:
  223. echo $form->render($view);
  224. // Suponiendo un objeto vista ha sido previamente establecido vía setView():
  225. echo $form;
  226. ]]>
  227. </programlisting>
  228. <para>
  229. De manera predeterminada, <code>Zend_Form</code> y
  230. <code>Zend_Form_Element</code> intentarán utilizar el objeto vista
  231. inicializado en el <code>ViewRenderer</code>, lo que significa que
  232. no tendrá que establecer la vista manualmente cuando use el MVC de
  233. Zend Framework. Generar un formulario en un script vista es tan
  234. simple como:
  235. </para>
  236. <programlisting role="php"><![CDATA[
  237. <?php echo $this->form ?>
  238. ]]>
  239. </programlisting>
  240. <para>
  241. Detrás del telón, <code>Zend_Form</code> utiliza "decoradores"
  242. (decorators) para generar la salida. Estos decoradores pueden
  243. reemplazar, añadir o anteponer contenido, y tienen plena
  244. introspección al elemento que les es pasado. Como resultado, puede
  245. combinar múltiples decoradores para lograr efectos personalizados.
  246. Predeterminadamente, <code>Zend_Form_Element</code> actualmente
  247. combina cuatro decoradores para obtener su salida; la configuración
  248. sería como sigue:
  249. </para>
  250. <programlisting role="php"><![CDATA[
  251. $element->addDecorators(array(
  252. 'ViewHelper',
  253. 'Errors',
  254. array('HtmlTag', array('tag' => 'dd')),
  255. array('Label', array('tag' => 'dt')),
  256. ));
  257. ]]>
  258. </programlisting>
  259. <para>
  260. (Donde &lt;HELPERNAME&gt; es el nombre de un view helper que
  261. utilizar, y varía según el elemento)
  262. </para>
  263. <para>
  264. Lo anterior crea una salida como la siguiente:
  265. </para>
  266. <programlisting role="html"><![CDATA[
  267. <dt><label for="username" class="required">Username</dt>
  268. <dd>
  269. <input type="text" name="username" value="123-abc" />
  270. <ul class="errors">
  271. <li>'123-abc' has not only alphabetic and digit characters</li>
  272. <li>'123-abc' does not match against pattern '/^[a-z]/i'</li>
  273. </ul>
  274. </dd>
  275. ]]>
  276. </programlisting>
  277. <para>
  278. (Aunque no con el mismo formato.)
  279. </para>
  280. <para>
  281. Puede cambiar los decoradores usados para un elemento si desea tener
  282. diferente salida; véase la sección sobre decoradores para mayor
  283. información.
  284. </para>
  285. <para>
  286. El propio formulario simplemente itera sobre los elementos y
  287. los cubre en un &lt;form&gt; HTML. El action y method que
  288. proporcionó cuando definió el formulario se pasan a la etiqueta
  289. <code>&lt;form&gt;</code>, como cualquier atributo que establezca
  290. vía <code>setAttribs()</code> y familia.
  291. </para>
  292. <para>
  293. Elementos son desplegados en el orden en el que fueron registrados,
  294. o, si el elemento contienen un atributo de orden, ese orden será
  295. utilizado. Usted puede fijar el orden de un elemento usando:
  296. </para>
  297. <programlisting role="php"><![CDATA[
  298. $element->setOrder(10);
  299. ]]>
  300. </programlisting>
  301. <para>
  302. O, cuando crea un elemento, pasándolo como una opción:
  303. </para>
  304. <programlisting role="php"><![CDATA[
  305. $form->addElement('text', 'username', array('order' => 10));
  306. ]]>
  307. </programlisting>
  308. </sect2>
  309. <sect2 id="zend.form.quickstart.validate">
  310. <title>Comprobar si un formulario es válido</title>
  311. <para>
  312. Después que un formulario es enviado, necesitará comprobar y ver si
  313. pasa las validaciones. Cada elemento es valuado contra los datos
  314. provistos; si una clave no está presente y el campo fue marcado como
  315. requerido, la validación se ejecuta contra un valor nulo.
  316. </para>
  317. <para>
  318. ¿De dónde provienen los datos?. Puede usar <code>$_POST</code> o
  319. <code>$_GET</code>, o cualquier otra fuente de datos que tenga a
  320. mano (solicitud de un servicio web, por ejemplo):
  321. </para>
  322. <programlisting role="php"><![CDATA[
  323. if ($form->isValid($_POST)) {
  324. // ¡Correcto!
  325. } else {
  326. // ¡Fallo!
  327. }
  328. ]]>
  329. </programlisting>
  330. <para>
  331. Con solicitudes AJAX, a veces puede ignorar la validación de un solo
  332. elemento, o grupo de elementos.
  333. <code>isValidPartial()</code> validará parcialmente el formulario.
  334. A diferencia de <code>isValid()</code>, que como sea, si alguna
  335. clave no esta presente, no ejecutará las validaciones para ese
  336. elemento en particular.
  337. </para>
  338. <programlisting role="php"><![CDATA[
  339. if ($form->isValidPartial($_POST)) {
  340. // de los elementos presentes, todos pasaron las validaciones
  341. } else {
  342. // uno u más elementos probados no pasaron las validaciones
  343. }
  344. ]]>
  345. </programlisting>
  346. <para>
  347. Un método adicional, <code>processAjax()</code>, puede también ser
  348. usado para validar formularios parciales. A diferencia de
  349. <code>isValidPartial()</code>, regresa una cadena en formato JSON
  350. conteniendo mensajes de error en caso de fallo.
  351. </para>
  352. <para>
  353. Asumiendo que sus validaciones han pasado, ahora puede obtener los
  354. valores filtrados:
  355. </para>
  356. <programlisting role="php"><![CDATA[
  357. $values = $form->getValues();
  358. ]]>
  359. </programlisting>
  360. <para>
  361. Si necesita los valores sin filtrar en algún punto, utilice:
  362. </para>
  363. <programlisting role="php"><![CDATA[
  364. $unfiltered = $form->getUnfilteredValues();
  365. ]]>
  366. </programlisting>
  367. </sect2>
  368. <sect2 id="zend.form.quickstart.errorstatus">
  369. <title>Obteniendo el estado de error</title>
  370. <para>
  371. Entonces, ¿su formulario no es válido? En la mayoría de los casos,
  372. simplemente puede generar el formulario nuevamente y los errores se
  373. mostrarán cuando se usen los decoradores predeterminados:
  374. </para>
  375. <programlisting role="php"><![CDATA[
  376. if (!$form->isValid($_POST)) {
  377. echo $form;
  378. // o asigne al objeto vista y genere una vista...
  379. $this->view->form = $form;
  380. return $this->render('form');
  381. }
  382. ]]>
  383. </programlisting>
  384. <para>
  385. Si quiere inspeccionar los errores, tiene dos métodos.
  386. <code>getErrors()</code> regresa una matriz asociativa de nombres /
  387. códigos de elementos (donde códigos es una matriz de códigos de
  388. error). <code>getMessages()</code> regresa una matriz asociativa
  389. de nombres / mensajes de elementos (donde mensajes es una matriz
  390. asociativa de pares código de error / mensaje de error). Si un
  391. elemento no tiene ningún error, no será incluido en la matriz.
  392. </para>
  393. </sect2>
  394. <sect2 id="zend.form.quickstart.puttingtogether">
  395. <title>Poniendo todo junto</title>
  396. <para>
  397. Construyamos un simple formulario de login. Necesitaremos
  398. elementos que representen:
  399. </para>
  400. <itemizedlist>
  401. <listitem><para>usuario</para></listitem>
  402. <listitem><para>contraseña</para></listitem>
  403. <listitem><para>Botón de ingreso</para></listitem>
  404. </itemizedlist>
  405. <para>
  406. Para nuestros propósitos, vamos a suponer que un usuario válido
  407. cumplirá con tener solo caracteres alfanuméricos, comenzar con una
  408. letra, tener una longitud mínima de 6 caracteres y una longitud
  409. máxima de 20 caracteres; se normalizarán en minúsculas. Las
  410. contraseñas deben tener un mínimo de 6 caracteres. Cuando se procese
  411. vamos simplemente a mostrar el valor, por lo que puede permanecer
  412. inválido.
  413. </para>
  414. <para>
  415. Usaremos el poder de la opciones de configuración de
  416. <code>Zend_Form</code> para crear el formulario:
  417. </para>
  418. <programlisting role="php"><![CDATA[
  419. $form = new Zend_Form();
  420. $form->setAction('/user/login')
  421. ->setMethod('post');
  422. // Crea un y configura el elemento username
  423. $username = $form->createElement('text', 'username');
  424. $username->addValidator('alnum')
  425. ->addValidator('regex', false, array('/^[a-z]+/'))
  426. ->addValidator('stringLength', false, array(6, 20))
  427. ->setRequired(true)
  428. ->addFilter('StringToLower');
  429. // Crea y configura el elemento password:
  430. $password = $form->createElement('password', 'password');
  431. $password->addValidator('StringLength', false, array(6))
  432. ->setRequired(true);
  433. // Añade los elementos al formulario:
  434. $form->addElement($username)
  435. ->addElement($password)
  436. // uso de addElement() como fábrica para crear el botón 'Login':
  437. ->addElement('submit', 'login', array('label' => 'Login'));
  438. ]]>
  439. </programlisting>
  440. <para>
  441. A continuación, vamos a crear un controlador para manejar esto:
  442. </para>
  443. <programlisting role="php"><![CDATA[
  444. class UserController extends Zend_Controller_Action
  445. {
  446. public function getForm()
  447. {
  448. // crea el formulario como se indicó arriba
  449. return $form;
  450. }
  451. public function indexAction()
  452. {
  453. // genera user/form.phtml
  454. $this->view->form = $this->getForm();
  455. $this->render('form');
  456. }
  457. public function loginAction()
  458. {
  459. if (!$this->getRequest()->isPost()) {
  460. return $this->_forward('index');
  461. }
  462. $form = $this->getForm();
  463. if (!$form->isValid($_POST)) {
  464. // Falla la validación; Se vuelve a mostrar el formulario
  465. $this->view->form = $form;
  466. return $this->render('form');
  467. }
  468. $values = $form->getValues();
  469. // ahora intenta y autentica...
  470. }
  471. }
  472. ]]>
  473. </programlisting>
  474. <para>
  475. Y un script para la vista que muestra el formulario:
  476. </para>
  477. <programlisting role="php"><![CDATA[
  478. <h2>Please login:</h2>
  479. <?= $this->form ?>
  480. ]]>
  481. </programlisting>
  482. <para>
  483. Como notará en el código del controlador, hay más trabajo por hacer:
  484. mientras la información enviada sea válida, necesitará todavía
  485. realizar la autenticación usando <code>Zend_Auth</code>, por
  486. ejemplo.
  487. </para>
  488. </sect2>
  489. <sect2 id="zend.form.quickstart.config">
  490. <title>Usando un objeto Zend_Config</title>
  491. <para>
  492. Todas las clases <code>Zend_Form</code> son configurables mediante
  493. <code>Zend_Config</code>; puede incluso pasar un objeto al
  494. constructor o pasarlo a través de <code>setConfig()</code>. Veamos
  495. cómo podemos crear el formulario anterior usando un archivo INI.
  496. Primero, vamos a seguir las recomendaciones, y colocaremos nuestras
  497. configuraciones dentro de secciones reflejando su objetivo y
  498. y enfocándonos en la sección 'development'. A continuación,
  499. pondremos en una sección de configuración para el controlador dado
  500. ('user'), y una clave para el formulario ('login'):
  501. </para>
  502. <programlisting role="ini"><![CDATA[
  503. [development]
  504. ; metainformación general del formulario
  505. user.login.action = "/user/login"
  506. user.login.method = "post"
  507. ; elemento username
  508. user.login.elements.username.type = "text"
  509. user.login.elements.username.options.validators.alnum.validator = "alnum"
  510. user.login.elements.username.options.validators.regex.validator = "regex"
  511. user.login.elements.username.options.validators.regex.options.pattern = "/^[a-z]/i"
  512. user.login.elements.username.options.validators.strlen.validator = "StringLength"
  513. user.login.elements.username.options.validators.strlen.options.min = "6"
  514. user.login.elements.username.options.validators.strlen.options.max = "20"
  515. user.login.elements.username.options.required = true
  516. user.login.elements.username.options.filters.lower.filter = "StringToLower"
  517. ; elemento password
  518. user.login.elements.password.type = "password"
  519. user.login.elements.password.options.validators.strlen.validator = "StringLength"
  520. user.login.elements.password.options.validators.strlen.options.min = "6"
  521. user.login.elements.password.options.required = true
  522. ; elemento submit
  523. user.login.elements.submit.type = "submit"
  524. ]]>
  525. </programlisting>
  526. <para>
  527. Entonces puede pasarlo al constructor del formulario:
  528. </para>
  529. <programlisting role="php"><![CDATA[
  530. $config = new Zend_Config_Ini($configFile, 'development');
  531. $form = new Zend_Form($config->user->login);
  532. ]]>
  533. </programlisting>
  534. <para>
  535. y el formulario entero será definido.
  536. </para>
  537. </sect2>
  538. <sect2 id="zend.form.quickstart.conclusion">
  539. <title>Conclusión</title>
  540. <para>
  541. Esperamos que después de este pequeño tutorial sea capaz de descubrir
  542. el poder y flexibilidad de <code>Zend_Form</code>. Continúe leyendo
  543. para profundizar más en el tema.
  544. </para>
  545. </sect2>
  546. </sect1>
  547. <!--
  548. vim:se ts=4 sw=4 et:
  549. -->