Zend_Controller-Plugins.xml 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <sect1 id="zend.controller.plugins" xmlns:xi="http://www.w3.org/2001/XInclude">
  4. <title>Плагины</title>
  5. <sect2 id="zend.controller.plugins.introduction">
  6. <title>Введение</title>
  7. <para>
  8. Архитектура контроллеров включает в себя систему плагинов, которая
  9. позволяет добавлять свой код, который будет вызываться при
  10. определенных событиях в процессе жизни контроллера. Фронт-контроллер
  11. использует брокер плагинов (plugin broker) в качестве реестра
  12. пользовательских плагинов, брокер плагинов также обеспечивает вызов
  13. методов событий в каждом плагине, зарегистрированном через
  14. фронт-контроллер.
  15. </para>
  16. <para>
  17. Методы событий определены в абстрактном классе
  18. <classname>Zend_Controller_Plugin_Abstract</classname>, от которого должны
  19. наследовать все пользовательские плагины:
  20. </para>
  21. <itemizedlist>
  22. <listitem>
  23. <para>
  24. <code>routeStartup()</code> вызывается до того, как
  25. <classname>Zend_Controller_Front</classname> вызовет
  26. <link linkend="zend.controller.router">маршрутизатор</link>
  27. для сопоставления запроса с зарегистрированными маршрутами.
  28. </para>
  29. </listitem>
  30. <listitem>
  31. <para>
  32. <code>routeShutdown()</code> вызывается после того, как
  33. <link linkend="zend.controller.router">маршрутизатор</link>
  34. завершит обработку запроса.
  35. </para>
  36. </listitem>
  37. <listitem>
  38. <para>
  39. <code>dispatchLoopStartup()</code> вызывается до того,
  40. как <classname>Zend_Controller_Front</classname> войдет в цикл
  41. диспетчеризации.
  42. </para>
  43. </listitem>
  44. <listitem>
  45. <para>
  46. <code>preDispatch()</code> вызывается до того, как
  47. <link linkend="zend.controller.dispatcher">диспетчером</link>
  48. будет вызвано действие. Этот обратный
  49. вызов (callback) позволяет реализовать поведение посредника
  50. или фильтра. Через изменение запроса и сброс его
  51. флага диспетчеризации (методом
  52. <code>Zend_Controller_Request_Abstract::setDispatched(false)</code>)
  53. текущее действие может быть пропущено и/или заменено на
  54. другое.
  55. </para>
  56. </listitem>
  57. <listitem>
  58. <para>
  59. <code>postDispatch()</code> вызывается после того, как
  60. действие было вызвано
  61. <link linkend="zend.controller.dispatcher">диспетчером</link>.
  62. Этот обратный вызов позволяет реализовать поведение фильтра
  63. или посредника. Через изменение запроса и сброс его
  64. флага диспетчеризации (методом
  65. <code>Zend_Controller_Request_Abstract::setDispatched(false)</code>)
  66. может быть определено новое действие для
  67. диспетчеризации.
  68. </para>
  69. </listitem>
  70. <listitem>
  71. <para>
  72. <code>dispatchLoopShutdown()</code> вызывается после выхода
  73. <classname>Zend_Controller_Front</classname> из его цикла
  74. диспетчеризации.
  75. </para>
  76. </listitem>
  77. </itemizedlist>
  78. </sect2>
  79. <sect2 id="zend.controller.plugins.writing">
  80. <title>Написание плагинов</title>
  81. <para>
  82. Для того, чтобы написать класс плагина, просто включите и расширьте
  83. абстрактный класс <classname>Zend_Controller_Plugin_Abstract</classname>:
  84. </para>
  85. <programlisting language="php"><![CDATA[
  86. class MyPlugin extends Zend_Controller_Plugin_Abstract
  87. {
  88. // ...
  89. }
  90. ]]></programlisting>
  91. <para>
  92. Ни один из методов класса
  93. <classname>Zend_Controller_Plugin_Abstract</classname> не является
  94. абстрактным, поэтому классы плагинов не обязательно должны
  95. реализовывать все из перечисленных выше методов событий.
  96. Разработчики плагинов могут реализовывать только те методы, которые
  97. требуются для их конкретных нужд.
  98. </para>
  99. <para>
  100. <classname>Zend_Controller_Plugin_Abstract</classname> также делает объекты
  101. запроса и ответа доступными плагинам контроллеров через методы
  102. <code>getRequest()</code> и <code>getResponse()</code>,
  103. соответственно.
  104. </para>
  105. </sect2>
  106. <sect2 id="zend.controller.plugins.using">
  107. <title>Использование плагинов</title>
  108. <para>
  109. Классы плагинов регистрируются через
  110. <code>Zend_Controller_Front::registerPlugin()</code>, их можно
  111. регистрировать в любой момент времени. Следующий пример
  112. демонстрирует использование плагина в цепочке контроллеров:
  113. </para>
  114. <programlisting language="php"><![CDATA[
  115. class MyPlugin extends Zend_Controller_Plugin_Abstract
  116. {
  117. public function routeStartup(Zend_Controller_Request_Abstract $request)
  118. {
  119. $this->getResponse()
  120. ->appendBody("<p>routeStartup() called</p>\n");
  121. }
  122. public function routeShutdown(Zend_Controller_Request_Abstract $request)
  123. {
  124. $this->getResponse()
  125. ->appendBody("<p>routeShutdown() called</p>\n");
  126. }
  127. public function dispatchLoopStartup(Zend_Controller_Request_Abstract $request)
  128. {
  129. $this->getResponse()
  130. ->appendBody("<p>dispatchLoopStartup() called</p>\n");
  131. }
  132. public function preDispatch(Zend_Controller_Request_Abstract $request)
  133. {
  134. $this->getResponse()
  135. ->appendBody("<p>preDispatch() called</p>\n");
  136. }
  137. public function postDispatch(Zend_Controller_Request_Abstract $request)
  138. {
  139. $this->getResponse()
  140. ->appendBody("<p>postDispatch() called</p>\n");
  141. }
  142. public function dispatchLoopShutdown()
  143. {
  144. $this->getResponse()
  145. ->appendBody("<p>dispatchLoopShutdown() called</p>\n");
  146. }
  147. }
  148. $front = Zend_Controller_Front::getInstance();
  149. $front->setControllerDirectory('/path/to/controllers')
  150. ->setRouter(new Zend_Controller_Router_Rewrite())
  151. ->registerPlugin(new MyPlugin());
  152. $front->dispatch();
  153. ]]></programlisting>
  154. <para>
  155. При условии, что вызываемые действия не производят вывод, и
  156. что вызвано только одно действие, с плагином выше должен получиться
  157. следующий вывод:
  158. </para>
  159. <programlisting language="php"><![CDATA[
  160. <p>routeStartup() called</p>
  161. <p>routeShutdown() called</p>
  162. <p>dispatchLoopStartup() called</p>
  163. <p>preDispatch() called</p>
  164. <p>postDispatch() called</p>
  165. <p>dispatchLoopShutdown() called</p>
  166. ]]></programlisting>
  167. <note>
  168. <para>
  169. Плагины могут регистрироваться в любой точке выполнения
  170. фронт-контроллера. Однако, если событие, для которого
  171. плагин имеет зарегистрированный метод события, уже произошло, то
  172. этот метод не будет запущен.
  173. </para>
  174. </note>
  175. </sect2>
  176. <sect2 id="zend.controller.plugins.manipulating">
  177. <title>Извлечение и работа с плагинами</title>
  178. <para>
  179. Иногда может понадобиться отменить регистрацию плагина или извлечь
  180. его. Следующие методы фронт-контроллера позволяют сделать это:
  181. </para>
  182. <itemizedlist>
  183. <listitem><para>
  184. <code>getPlugin($class)</code> позволяет извлекать плагин по
  185. имени класса. Если не найден соответствующий плагин, то
  186. возвращается false. Если зарегистрировано более одного плагина
  187. этого класса, то будет возвращен массив.
  188. </para></listitem>
  189. <listitem><para>
  190. <code>getPlugins()</code> возвращает весь стек плагинов.
  191. </para></listitem>
  192. <listitem><para>
  193. <code>unregisterPlugin($plugin)</code> производит удаление
  194. плагина из стека. Вы можете передавать объект плагина или имя
  195. класса плагина, регистрацию которого вы хотите отменить. Если вы
  196. передаете имя класса, то будут удалены все плагины этого класса.
  197. </para></listitem>
  198. </itemizedlist>
  199. </sect2>
  200. <sect2 id="zend.controller.plugins.standard">
  201. <title>Плагины, включенные в стандартную поставку</title>
  202. <para>
  203. Zend Framework в его стандартной поставке включает в себя плагин для
  204. обработки ошибок.
  205. </para>
  206. <xi:include href="Zend_Controller-Plugins-ActionStack.xml" />
  207. <xi:include href="Zend_Controller-Plugins-ErrorHandler.xml" />
  208. </sect2>
  209. </sect1>
  210. <!--
  211. vim:se ts=4 sw=4 et:
  212. -->