Zend_Controller-ActionController.xml 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614
  1. <sect1 id="zend.controller.action">
  2. <title>Action Controllers</title>
  3. <sect2 id="zend.controller.action.introduction">
  4. <title>Wprowadzenie</title>
  5. <para>
  6. <code>Zend_Controller_Action</code> jest klasą abstrakcyjną, której
  7. możesz użyć do implementacji kontrolerów akcji, których wraz z
  8. kontrolerem frontowym użyjesz do budowania aplikacji opartej na
  9. wzorcu Model-View-Controller (MVC).
  10. </para>
  11. <para>
  12. Aby użyć klasy <code>Zend_Controller_Action</code>, powinieneś ją
  13. rozszerzyć w swoich klasach kontrolerów akcji (lub rozszerzyć ją aby
  14. utworzyć swoją własną bazową klasę dla kontrolerów akcji).
  15. Najbardziej podstawową operacją jest rozszerzenie tej klasy oraz
  16. utworzenie metod akcji, które odpowiadają różnym akcjom jakie ma
  17. obsługiwać kontroler na twojej stronie. Obsługa routingu i
  18. uruchamiania w Zend_Controller automatycznie przeszuka wszystkie
  19. metody twojej klasy, których nazwa zakończona jest wyrazem 'Action',
  20. aby znaleźć odpowiednią akcję kontrolera.
  21. </para>
  22. <para>
  23. Na przykład, załóżmy, że twoja klasa jest zdefiniowana w ten sposób:
  24. </para>
  25. <programlisting role="php"><![CDATA[
  26. class FooController extends Zend_Controller_Action
  27. {
  28. public function barAction()
  29. {
  30. // zrób coś
  31. }
  32. public function bazAction()
  33. {
  34. // zrób coś
  35. }
  36. }
  37. ]]>
  38. </programlisting>
  39. <para>
  40. Powyższa klasa <code>FooController</code> (kontroler <code>foo</code>)
  41. definiuje dwie akcje, <code>bar</code> oraz <code>baz</code>.
  42. </para>
  43. <para>
  44. Można tu osiągnąć dużo więcej, na przykład: utworzyć własne akcje
  45. inicjalizacyjne, utworzyć domyślne akcje do wywołania gdy nie ma
  46. określonej akcji (lub określona jest nieprawidłowa), użyć metod
  47. pre- oraz post-dispatch oraz użyć wielu różnych metod pomocników.
  48. Ten rozdział jest rozeznaniem w funkcjonalnościach kontrolera akcji.
  49. </para>
  50. <note>
  51. <title>Domyślne zachowanie</title>
  52. <para>
  53. Domyślnie <link linkend="zend.controller.front">kontroler
  54. frontowy</link> włącza klasę pomocniczą akcji <link
  55. linkend="zend.controller.actionhelpers.viewrenderer">ViewRenderer</link>.
  56. Ta klasa zajmuje się przekazywaniem widoku do kontrolera, a
  57. także automatycznym renderowaniem widoków. Możesz to wyłączyć
  58. w swoim kontrolerze akcji używając jednej z poniższych metod:
  59. </para>
  60. <programlisting role="php"><![CDATA[
  61. class FooController extends Zend_Controller_Action
  62. {
  63. public function init()
  64. {
  65. // Lokalnie, tylko dla tego kontrolera; affects all actions, as loaded in init:
  66. $this->_helper->viewRenderer->setNoRender(true);
  67. // Globalnie:
  68. $this->_helper->removeHelper('viewRenderer');
  69. // Także globalnie, but would need to be in conjunction with the local
  70. // version in order to propagate for this controller:
  71. Zend_Controller_Front::getInstance()->setParam('noViewRenderer', true);
  72. }
  73. }
  74. ]]>
  75. </programlisting>
  76. <para>
  77. <code>initView()</code>, <code>getViewScript()</code>,
  78. <code>render()</code>, and <code>renderScript()</code> each
  79. proxy to the <code>ViewRenderer</code> unless the helper is not
  80. in the helper broker or the <code>noViewRenderer</code> flag has
  81. been set.
  82. </para>
  83. <para>
  84. Możesz także w prosty sposób wyłączyć renderowanie dla
  85. konkretnego widoku ustawiając flagę <code>noRender</code> w
  86. klasie <code>ViewRenderer</code>:
  87. </para>
  88. <programlisting role="php"><![CDATA[
  89. class FooController extends Zend_Controller_Action
  90. {
  91. public function barAction()
  92. {
  93. // wyłączamy automatyczne renderowanie tylko dla tej akcji:
  94. $this->_helper->viewRenderer->setNoRender();
  95. }
  96. }
  97. ]]>
  98. </programlisting>
  99. <para>
  100. Możesz chcieć wyłączyć klasę <code>ViewRenderer</code>
  101. jeśli nie potrzebujesz obiektu widoku lub jeśli nie chcesz
  102. renderować skryptów widoku. (na przykład jeśli używasz
  103. kontrolera akcji aby obsługiwać żądania do webserwisu takie jak
  104. SOAP, XML-RPC czy REST). W większości przypadków nie będziesz
  105. potrzebować wyłączać globalnie renderowania klasy
  106. <code>ViewRenderer</code>, wystarczy selektywne wyłączenie
  107. wewnątrz pojedynczego kontrolera lub akcji.
  108. </para>
  109. </note>
  110. </sect2>
  111. <sect2 id="zend.controller.action.initialization">
  112. <title>Inicjalizacja obiektu</title>
  113. <para>
  114. O ile zawsze możesz nadpisać konstruktor kontrolera akcji, nie
  115. zalecamy tego. Zend_Controller_Action::__construct() przeprowadza
  116. kilka ważnych zadań, takich jak zarejestrowanie obiektów żądania i
  117. odpowiedzi, oraz przekazanie argumentów wywołania przez kontroler
  118. frontowy. Jeśli musisz nadpisać konstruktor, upewnij się że
  119. wywołasz metodę <code>parent::__construct($request, $response,
  120. $invokeArgs)</code>.
  121. </para>
  122. <para>
  123. Bardziej odpowiednim sposobem skonfigurowania instancji jest użycie
  124. metody <code>init()</code>, która jest wywoływana jako ostatnie
  125. zadanie konstruktora <code>__construct()</code>. Na przykład jeśli
  126. chcesz połączyć się z bazą danych:
  127. </para>
  128. <programlisting role="php"><![CDATA[
  129. class FooController extends Zend_Controller_Action
  130. {
  131. public function init()
  132. {
  133. $this->db = Zend_Db::factory('Pdo_Mysql', array(
  134. 'host' => 'myhost',
  135. 'username' => 'user',
  136. 'password' => 'XXXXXXX',
  137. 'dbname' => 'website'
  138. ));
  139. }
  140. }
  141. ]]>
  142. </programlisting>
  143. </sect2>
  144. <sect2 id="zend.controller.action.prepostdispatch">
  145. <title>Metody Pre-Dispatch oraz Post-Dispatch</title>
  146. <para>
  147. Klasa <code>Zend_Controller_Action</code> definiuje dwie metody,
  148. <code>preDispatch()</code> oraz <code>postDispatch()</code>, które
  149. mogą być wywołane przed i po wywołaniu akcji. Mogą one być użyteczne
  150. w wielu sytuacjach: weryfikowanie autentykacji oraz kontroli dostępu
  151. ACL odnośnie uruchamianej akcji, (przez wywołanie metody
  152. <code>_forward()</code> w metodzie <code>preDispatch()</code>,
  153. dzięki czemu akcja może być pominięta), lub na przykład umieszczenie
  154. wygenerowanej zawartości w głównym szablonie
  155. (<code>postDispatch()</code>).
  156. </para>
  157. </sect2>
  158. <sect2 id="zend.controller.action.accessors">
  159. <title>Metody dostępowe</title>
  160. <para>
  161. W obiekcie zarejestrowanych jest wiele obiektów oraz zmiennych i
  162. wszystkie mają metody dostępowe..
  163. </para>
  164. <itemizedlist>
  165. <listitem><para>
  166. <emphasis>Obiekt żądania</emphasis>: metoda
  167. <code>getRequest()</code> może być użyta do odebrania obiektu
  168. żądania używanego do wywoływania akcji.
  169. </para></listitem>
  170. <listitem>
  171. <para>
  172. <emphasis>Obiekt odpowiedzi</emphasis>: metoda
  173. <code>getResponse()</code> może być użyta do odebrania
  174. obiektu odpowiedzi przechowującego finalną odpowiedź.
  175. Niektóre typowe wywołania mogą wyglądać tak:
  176. </para>
  177. <programlisting role="php"><![CDATA[
  178. $this->getResponse()->setHeader('Content-Type', 'text/xml');
  179. $this->getResponse()->appendBody($content);
  180. ]]>
  181. </programlisting>
  182. </listitem>
  183. <listitem>
  184. <para>
  185. <emphasis>Argumenty wywołania</emphasis>: kontroler frontowy
  186. może przekazać parametry do routera, obiektu uruchamiającego
  187. oraz do kontrolera akcji. Aby je odebrać użyj metody
  188. <code>getInvokeArg($key)</code>; alternatywnie pobierz całą
  189. listę używając metody <code>getInvokeArgs()</code>.
  190. </para>
  191. </listitem>
  192. <listitem>
  193. <para>
  194. <emphasis>Parametry żądania</emphasis>: Obiekt żądania
  195. przechowuje parametry żądania takie jak dowolne parametry z
  196. tablic _GET lub _POST oraz parametry użytkownika zdefiniowane
  197. w ścieżce adresu URL. Aby je odebrać, użyj metody
  198. <code>_getParam($key)</code> lub <code>_getAllParams()</code>.
  199. Możesz także ustawić parametry żądania używając metody
  200. <code>_setParam()</code>; jest to użyteczne gdy przenosimy
  201. do innych akcji.
  202. </para>
  203. <para>
  204. Aby sprawdzić czy parametr istnieje czy nie (co jest
  205. użyteczne przy wywołaniach logicznych), użyj
  206. <code>_hasParam($key)</code>.
  207. </para>
  208. <note>
  209. <para>
  210. Metoda <code>_getParam()</code> może pobierać opcjonalny
  211. drugi argument zawierający domyślną wartość, ktora
  212. zostanie użyta, jeśli parametr nie został zdefiniowany
  213. lub jeśli jest pusty. Użycie drugiego parametru powoduje,
  214. że wywołanie metody <code>_hasParam()</code> przed
  215. odebraniem parametru nie jest konieczne:
  216. </para>
  217. <programlisting role="php"><![CDATA[
  218. // Użyj domyślnej wartości 1 jeśli parametr id jest pusty
  219. $id = $this->_getParam('id', 1);
  220. // Zamiast:
  221. if ($this->_hasParam('id') {
  222. $id = $this->_getParam('id');
  223. } else {
  224. $id = 1;
  225. }
  226. ]]>
  227. </programlisting>
  228. </note>
  229. </listitem>
  230. </itemizedlist>
  231. </sect2>
  232. <sect2 id="zend.controller.action.viewintegration">
  233. <title>Integracja z widokiem</title>
  234. <para>
  235. <code>Zend_Controller_Action</code> provides a rudimentary and
  236. flexible mechanism for view integration.
  237. Odpowiadają za to dwie metody, <code>initView()</code> oraz
  238. <code>render()</code>;
  239. the former method lazy-loads the <code>$view</code> public property,
  240. and the latter renders a view based on the current requested action,
  241. using the directory hierarchy to determine the script path.
  242. </para>
  243. <sect3 id="zend.controller.action.viewintegration.initview">
  244. <title>Inicjowanie obiektu widoku</title>
  245. <para>
  246. Metoda <code>initView()</code> inicjuje obiekt widoku. Metoda
  247. <code>render()</code> wywołuje <code>initView()</code> w celu
  248. odebrania obiektu widoku, ale może on być zainicjowany w
  249. dowolnym momencie; domyślnie przypisuje ona do właściwości
  250. <code>$view</code> obiekt klasy <code>Zend_View</code>, ale może
  251. być użyta dowolna klasa implementująca interfejs
  252. <code>Zend_View_Interface</code>. Jeśli obiekt
  253. <code>$view</code> jest już zainicjowany, metoda po prostu
  254. zwróci ten obiekt.
  255. </para>
  256. <para>
  257. Domyślna implementacja zakłada taką strukturę katalogów:
  258. </para>
  259. <programlisting role="php"><![CDATA[
  260. applicationOrModule/
  261. controllers/
  262. IndexController.php
  263. views/
  264. scripts/
  265. index/
  266. index.phtml
  267. helpers/
  268. filters/
  269. ]]>
  270. </programlisting>
  271. <para>
  272. Innymi słowy, założone jest, że skrypty widoków znajdują się w
  273. podkatalogu <code>views/scripts/</code>, a podkatalog
  274. <code>views</code> zawiera poboczne funkcjonalności (klasy
  275. pomocnicze, filtry). Gdy określana jest nazwa skryptu oraz
  276. ścieżka, katalog <code>views/scripts/</code> jest używany jako
  277. katalog bazowy. Zawiera on katalogi o nazwach pochodzących od
  278. kontrolerów, co zapewnia hierarchię skryptów widoków.
  279. </para>
  280. </sect3>
  281. <sect3 id="zend.controller.action.viewintegration.render">
  282. <title>Renderowanie widoków</title>
  283. <para>
  284. Metoda <code>render()</code> ma taką sygnaturę:
  285. </para>
  286. <programlisting role="php"><![CDATA[
  287. string render(string $action = null, string $name = null, bool $noController = false);
  288. ]]>
  289. </programlisting>
  290. <para>
  291. <code>render()</code> renderuje skrypt widoku. Jeśli nie
  292. przekazano argumentów, zakładane jest, że ścieżka skryptu to
  293. <code>[kontroler]/[akcja].phtml</code> (gdzie
  294. <code>.phtml</code> jest wartością właściwości
  295. <code>$viewSuffix</code>). Przekazanie wartości parametru
  296. <code>$action</code> spowoduje zrenderowanie tego szablonu z
  297. podkatalogu <code>[kontroler]</code>. Aby zrezygnować z użycia
  298. podkatalogu <code>[kontroler]</code>, przekaż logiczną wartość
  299. true dla <code>$noController</code>. Na koniec szablony są
  300. renderowane i przekazywane do obiektu odpowiedzi; jeśli chcesz
  301. zrenderować do konkretnego <link
  302. linkend="zend.controller.response.namedsegments">nazwanego
  303. segmentu</link> w obiekcie odpowiedzi, przekaż wartość dla
  304. parametru <code>$name</code>.
  305. </para>
  306. <note><para>
  307. Z tego względu, że nazwy kontrolera i akcji mogą zawierać
  308. takie rozgraniczające znaki jak '_', '.', oraz '-', metoda
  309. render() zamienia je wszystkie na '-' gdy określa nazwę
  310. skryptu. Wewnętrznie, do przeprowadzenia tej operacji
  311. używane są znaki rozgraniczające słowa oraz ścieżki z
  312. obiektu uruchamiającego. Dlatego żądanie do
  313. <code>/foo.bar/baz-bat</code> zrenderuje skrypt
  314. <code>foo-bar/baz-bat.phtml</code>. Jeśli nazwa metody akcji
  315. jest w postaci camelCasing, zapamiętaj, że spowoduje to
  316. rozdzieleniem słów za pomocą znaku '-' podczas określania
  317. nazwy pliku skryptu widoku.
  318. </para></note>
  319. <para>
  320. Kilka przykładów:
  321. </para>
  322. <programlisting role="php"><![CDATA[
  323. class MyController extends Zend_Controller_Action
  324. {
  325. public function fooAction()
  326. {
  327. // Renderuje my/foo.phtml
  328. $this->render();
  329. // Renderuje my/bar.phtml
  330. $this->render('bar');
  331. // Renderuje baz.phtml
  332. $this->render('baz', null, true);
  333. // Renderuje my/login.phtml w segmencie 'form' obiektu odpowiedzi
  334. $this->render('login', 'form');
  335. // Renderuje site.phtml w segmencie 'page' obiektu odpowiedzi;
  336. // nie używa podkatalogu 'my/'
  337. $this->render('site', 'page', true);
  338. }
  339. public function bazBatAction()
  340. {
  341. // Renderuje my/baz-bat.phtml
  342. $this->render();
  343. }
  344. }
  345. ]]>
  346. </programlisting>
  347. </sect3>
  348. </sect2>
  349. <sect2 id="zend.controller.action.utilmethods">
  350. <title>Metody narzędziowe</title>
  351. <para>
  352. Oprócz metod dostępowych i metod integracji z widokiem, klasa
  353. <code>Zend_Controller_Action</code> posiada kilka metod
  354. narzędziowych używanych do przeprowadzania ważnych zadań wewnątrz
  355. twoich metod akcji (lub wewnątrz metod pre-/post-dispatch).
  356. </para>
  357. <itemizedlist>
  358. <listitem>
  359. <para>
  360. <code>_forward($action, $controller = null, $module =
  361. null, array $params = null)</code>:
  362. wykonuje inną akcję. Jeśli zostanie wywołana w metodzie
  363. <code>preDispatch()</code>, obecnie zażądana akcja zostanie
  364. pominięta, na rzecz nowej akcji. W przeciwnym wypadku, po
  365. wykonaniu obecnej akcji, będzie wywołana akcja zażądana w
  366. metodzie _forward().
  367. </para>
  368. </listitem>
  369. <listitem>
  370. <para>
  371. <code>_redirect($url, array $options =
  372. array())</code>:
  373. przekierowuje do innej lokacji. Ta metoda przyjmuje w
  374. parametrze URL oraz opcjonalny zestaw opcji. Domyślnie
  375. przeprowadzane jest przekierowanie HTTP 302.
  376. </para>
  377. <para>
  378. Zestaw opcji może zawierać jeden lub więcej z poniższych
  379. kluczy:
  380. </para>
  381. <itemizedlist>
  382. <listitem>
  383. <para>
  384. <emphasis>exit:</emphasis> określa czy skrypt ma
  385. zakończyć działanie od razu po przekierowaniu. Jeśli
  386. tak, to skrypt zamknie wszystkie otwarte sesje i
  387. przeprowadzi przekierowanie.
  388. </para>
  389. <para>
  390. Możesz ustawić tę opcję globalnie wewnątrz
  391. kontrolera używając metody dostępowej
  392. <code>setRedirectExit()</code>.
  393. </para>
  394. </listitem>
  395. <listitem>
  396. <para>
  397. <emphasis>prependBase:</emphasis> określa czy bazowy
  398. adres URL zarejestrowany w obiekcie żądania ma być
  399. dołączony do adresu URL przekierowania.
  400. </para>
  401. <para>
  402. Możesz ustawić tę opcję globalnie wewnątrz
  403. kontrolera używając metody dostępowej
  404. <code>setRedirectPrependBase()</code>.
  405. </para>
  406. </listitem>
  407. <listitem>
  408. <para>
  409. <emphasis>code:</emphasis> kod HTTP do użycia
  410. podczas przekierowania. Domyślnie użyty jest kod
  411. HTTP 302; może być użyty dowolny kod pomiędzy 301
  412. a 306.
  413. </para>
  414. <para>
  415. Możesz ustawić tę opcję globalnie wewnątrz
  416. kontrolera używając metody dostępowej
  417. <code>setRedirectCode()</code>.
  418. </para>
  419. </listitem>
  420. </itemizedlist>
  421. </listitem>
  422. </itemizedlist>
  423. </sect2>
  424. <sect2 id="zend.controller.action.subclassing">
  425. <title>Rozszerzanie klasy kontrolera akcji</title>
  426. <para>
  427. By design, <code>Zend_Controller_Action</code> must be subclassed
  428. in order to create an action controller. At the minimum, you will
  429. need to define action methods that the controller may call.
  430. </para>
  431. <para>
  432. Besides creating useful functionality for your web applications, you
  433. may also find that you're repeating much of the same setup or
  434. utility methods in your various controllers; if so, creating a
  435. common base controller class that extends
  436. <code>Zend_Controller_Action</code> could solve such redundancy.
  437. </para>
  438. <example id="zend.controller.action.subclassing.example-call">
  439. <title>Jak obsługiwać nieistniejące akcje</title>
  440. <para>
  441. Jeśli zażądamy nieistniejącej akcji kontrolera, wywołana zostanie
  442. metoda <code>Zend_Controller_Action::__call()</code>. Metoda
  443. <code>__call()</code> jest oczywiście magiczną metodą PHP
  444. służącą do przeładowania metod.
  445. </para>
  446. <para>
  447. Domyślnie, ta metoda wyrzuca wyjątek
  448. <code>Zend_Controller_Action_Exception</code> oznaczający, że
  449. zażądana metoda nie została znaleziona w kontrolerze. Jeśli nazwa
  450. zażądanej metody zakończona jest słowem 'Action', zakładane jest,
  451. że zażądana została akcja i że ona nie istnieje; taki błąd
  452. powoduje wyrzucenie wyjątku z kodem 404. Próby wywołania
  453. wszystkich innych metod powodują wyrzucenie wyjątku z kodem 500.
  454. Pozwala to na rozróżnienie błędu nie znalezionej strony od
  455. innych błędów aplikacji.
  456. </para>
  457. <para>
  458. Na przykład, jeśli chcesz wyświetlić informacje o błędzie,
  459. możesz zrobić to w taki sposób:
  460. </para>
  461. <programlisting role="php"><![CDATA[
  462. class MyController extends Zend_Controller_Action
  463. {
  464. public function __call($method, $args)
  465. {
  466. if ('Action' == substr($method, -6)) {
  467. // Jeśli metoda akcji nie została znaleziona, renderuje szablon informujący o błędzie
  468. return $this->render('error');
  469. }
  470. // wszystkie inne metody wyrzucają wyjątek
  471. throw new Exception('Invalid method "' . $method . '" called', 500);
  472. }
  473. }
  474. ]]>
  475. </programlisting>
  476. <para>
  477. Inną możliwością jest przeniesienie do domyślnego kontrolera:
  478. </para>
  479. <programlisting role="php"><![CDATA[
  480. class MyController extends Zend_Controller_Action
  481. {
  482. public function indexAction()
  483. {
  484. $this->render();
  485. }
  486. public function __call($method, $args)
  487. {
  488. if ('Action' == substr($method, -6)) {
  489. // Jeśli metoda akcji nie została znaleziona, przenieś do akcji index
  490. return $this->_forward('index');
  491. }
  492. // wszystkie inne metody wyrzucają wyjątek
  493. throw new Exception('Invalid method "' . $method . '" called', 500);
  494. }
  495. }
  496. ]]>
  497. </programlisting>
  498. </example>
  499. <para>
  500. Besides overriding <code>__call()</code>, each of the
  501. initialization, utility, accessor, view, and dispatch hook methods
  502. mentioned previously in this chapter may be overridden in order to
  503. customize your controllers. As an example, if you are storing your
  504. view object in a registry, you may want to modify your
  505. <code>initView()</code> method with code resembling the following:
  506. </para>
  507. <programlisting role="php"><![CDATA[
  508. abstract class My_Base_Controller extends Zend_Controller_Action
  509. {
  510. public function initView()
  511. {
  512. if (null === $this->view) {
  513. if (Zend_Registry::isRegistered('view')) {
  514. $this->view = Zend_Registry::get('view');
  515. } else {
  516. $this->view = new Zend_View();
  517. $this->view->setBasePath(dirname(__FILE__) . '/../views');
  518. }
  519. }
  520. return $this->view;
  521. }
  522. }
  523. ]]>
  524. </programlisting>
  525. <para>
  526. Hopefully, from the information in this chapter, you can see the
  527. flexibility of this particular component and how you can shape it to
  528. your application's or site's needs.
  529. </para>
  530. </sect2>
  531. </sect1>