Zend_Controller_Action
jest klasą abstrakcyjną, której
możesz użyć do implementacji kontrolerów akcji, których wraz z
kontrolerem frontowym użyjesz do budowania aplikacji opartej na
wzorcu Model-View-Controller (MVC).
Aby użyć klasy Zend_Controller_Action
, powinieneś ją
rozszerzyć w swoich klasach kontrolerów akcji (lub rozszerzyć ją aby
utworzyć swoją własną bazową klasę dla kontrolerów akcji).
Najbardziej podstawową operacją jest rozszerzenie tej klasy oraz
utworzenie metod akcji, które odpowiadają różnym akcjom jakie ma
obsługiwać kontroler na twojej stronie. Obsługa routingu i
uruchamiania w Zend_Controller automatycznie przeszuka wszystkie
metody twojej klasy, których nazwa zakończona jest wyrazem 'Action',
aby znaleźć odpowiednią akcję kontrolera.
Na przykład, załóżmy, że twoja klasa jest zdefiniowana w ten sposób:
class FooController extends Zend_Controller_Action { public function barAction() { // zrób coś } public function bazAction() { // zrób coś } }
Powyższa klasa FooController
(kontroler foo
)
definiuje dwie akcje, bar
oraz baz
.
Można tu osiągnąć dużo więcej, na przykład: utworzyć własne akcje inicjalizacyjne, utworzyć domyślne akcje do wywołania gdy nie ma określonej akcji (lub określona jest nieprawidłowa), użyć metod pre- oraz post-dispatch oraz użyć wielu różnych metod pomocników. Ten rozdział jest rozeznaniem w funkcjonalnościach kontrolera akcji.
Domyślne zachowanie | |
---|---|
Domyślnie kontroler frontowy włącza klasę pomocniczą akcji ViewRenderer. Ta klasa zajmuje się przekazywaniem widoku do kontrolera, a także automatycznym renderowaniem widoków. Możesz to wyłączyć w swoim kontrolerze akcji używając jednej z poniższych metod: <?php class FooController extends Zend_Controller_Action { public function init() { // Lokalnie, tylko dla tego kontrolera: $this->_invokeArgs['noViewRenderer'] = true; // Globalnie: $this->_helper->removeHelper('viewRenderer'); // Also globally, but would need to be in conjunction with the local // version in order to propagate for this controller: Zend_Controller_Front::getInstance()->setParam('noViewRenderer', true); } }
Możesz także w prosty sposób wyłączyć renderowanie dla
konkretnego widoku ustawiając flagę <?php class FooController extends Zend_Controller_Action { public function barAction() { // wyłączamy automatyczne renderowanie dla tej akcji: $this->_helper->viewRenderer->setNoRender(); } }
The primary reasons to disable the |
O ile zawsze możesz nadpisać konstruktor kontrolera akcji, nie
zalecamy tego. Zend_Controller_Action::__construct() przeprowadza
kilka ważnych zadań, takich jak zarejestrowanie obiektów żądania i
odpowiedzi, oraz przekazanie argumentów wywołania przez kontroler
frontowy. Jeśli musisz nadpisać konstruktor, upewnij się że
wywołasz metodę parent::__construct($request, $response,
$invokeArgs)
.
Bardziej odpowiednim sposobem skonfigurowania instancji jest użycie
metody init()
, która jest wywoływana jako ostatnie
zadanie konstruktora __construct()
. Na przykład jeśli
chcesz połączyć się z bazą danych:
class FooController extends Zend_Controller_Action { public function init() { $this->db = Zend_Db::factory('Pdo_Mysql', array( 'host' => 'myhost', 'username' => 'user', 'password' => 'XXXXXXX', 'dbname' => 'website' )); } }
Klasa Zend_Controller_Action
definiuje dwie metody,
preDispatch()
oraz postDispatch()
, które
mogą być wywołane przed i po wywołaniu akcji. Mogą one być użyteczne
w wielu sytuacjach: weryfikowanie autentykacji oraz kontroli dostępu
ACL odnośnie uruchamianej akcji, (przez wywołanie metody
_forward()
w metodzie preDispatch()
,
dzięki czemu akcja może być pominięta), lub na przykład umieszczenie
wygenerowanej zawartości w głównym szablonie
(postDispatch()
).
W obiekcie zarejestrowanych jest wiele obiektów oraz zmiennych i wszystkie mają metody dostępowe..
Obiekt żądania: metoda
getRequest()
może być użyta do odebrania obiektu
żądania używanego do wywoływania akcji.
Obiekt odpowiedzi: metoda
getResponse()
może być użyta do odebrania
obiektu odpowiedzi przechowującego finalną odpowiedź.
Niektóre typowe wywołania mogą wyglądać tak:
$this->getResponse()->setHeader('Content-Type', 'text/xml'); $this->getResponse()->appendBody($content);
Argumenty wywołania: kontroler frontowy
może przekazać parametry do routera, obiektu uruchamiającego
oraz do kontrolera akcji. Aby je odebrać użyj metody
getInvokeArg($key)
; alternatywnie pobierz całą
listę używając metody getInvokeArgs()
.
Parametry żądania: Obiekt żądania
przechowuje parametry żądania takie jak dowolne parametry z
tablic _GET lub _POST oraz parametry użytkownika zdefiniowane
w ścieżce adresu URL. Aby je odebrać, użyj metody
_getParam($key)
lub _getAllParams()
.
Możesz także ustawić parametry żądania używając metody
_setParam()
; jest to użyteczne gdy przenosimy
do innych akcji.
Aby sprawdzić czy parametr istnieje czy nie (co jest
użyteczne przy wywołaniach logicznych), użyj
_hasParam($key)
.
Notatka | |
---|---|
Metoda <?php // Użyj domyślnej wartości 1 jeśli parametr id jest pusty $id = $this->_getParam('id', 1); // Zamiast: if ($this->_hasParam('id') { $id = $this->_getParam('id'); } else { $id = 1; } ?> |
Zend_Controller_Action
provides a rudimentary and
flexible mechanism for view integration.
Odpowiadają za to dwie metody, initView()
oraz
render()
;
the former method lazy-loads the $view
public property,
and the latter renders a view based on the current requested action,
using the directory hierarchy to determine the script path.
Metoda initView()
inicjuje obiekt widoku. Metoda
render()
wywołuje initView()
w celu
odebrania obiektu widoku, ale może on być zainicjowany w
dowolnym momencie; domyślnie przypisuje ona do właściwości
$view
obiekt klasy Zend_View
, ale może
być użyta dowolna klasa implementująca interfejs
Zend_View_Interface
. Jeśli obiekt
$view
jest już zainicjowany, metoda po prostu
zwróci ten obiekt.
Domyślna implementacja zakłada taką strukturę katalogów:
applicationOrModule/ controllers/ IndexController.php views/ scripts/ index/ index.phtml helpers/ filters/
Innymi słowy, założone jest, że skrypty widoków znajdują się w
podkatalogu views/scripts/
, a podkatalog
views
zawiera poboczne funkcjonalności (klasy
pomocnicze, filtry). Gdy określana jest nazwa skryptu oraz
ścieżka, katalog views/scripts/
jest używany jako
katalog bazowy. Zawiera on katalogi o nazwach pochodzących od
kontrolerów, co zapewnia hierarchię skryptów widoków.
Metoda render()
ma taką sygnaturę:
<?php string render(string $action = null, string $name = null, bool $noController = false); ?>
render()
renderuje skrypt widoku. Jeśli nie
przekazano argumentów, zakładane jest, że ścieżka skryptu to
[kontroler]/[akcja].phtml
(gdzie
.phtml
jest wartością właściwości
$viewSuffix
). Przekazanie wartości parametru
$action
spowoduje zrenderowanie tego szablonu z
podkatalogu [kontroler]
. Aby zrezygnować z użycia
podkatalogu [kontroler]
, przekaż logiczną wartość
true dla $noController
. Na koniec szablony są
renderowane i przekazywane do obiektu odpowiedzi; jeśli chcesz
zrenderować do konkretnego nazwanego
segmentu w obiekcie odpowiedzi, przekaż wartość dla
parametru $name
.
Notatka | |
---|---|
Z tego względu, że nazwy kontrolera i akcji mogą zawierać
takie rozgraniczające znaki jak '_', '.', oraz '-', metoda
render() zamienia je wszystkie na '-' gdy określa nazwę
skryptu. Wewnętrznie, do przeprowadzenia tej operacji
używane są znaki rozgraniczające słowa oraz ścieżki z
obiektu uruchamiającego. Dlatego żądanie do
|
Kilka przykładów:
<?php class MyController extends Zend_Controller_Action { public function fooAction() { // Renderuje my/foo.phtml $this->render(); // Renderuje my/bar.phtml $this->render('bar'); // Renderuje baz.phtml $this->render('baz', null, true); // Renderuje my/login.phtml w segmencie 'form' obiektu odpowiedzi $this->render('login', 'form'); // Renderuje site.phtml w segmencie 'page' obiektu odpowiedzi; // nie używa podkatalogu 'my/' $this->render('site', 'page', true); } public function bazBatAction() { // Renderuje my/baz-bat.phtml $this->render(); } }
Oprócz metod dostępowych i metod integracji z widokiem, klasa
Zend_Controller_Action
posiada kilka metod
narzędziowych używanych do przeprowadzania ważnych zadań wewnątrz
twoich metod akcji (lub wewnątrz metod pre-/post-dispatch).
_forward($action, $controller = null, $module =
null, array $params = null)
:
wykonuje inną akcję. Jeśli zostanie wywołana w metodzie
preDispatch()
, obecnie zażądana akcja zostanie
pominięta, na rzecz nowej akcji. W przeciwnym wypadku, po
wykonaniu obecnej akcji, będzie wywołana akcja zażądana w
metodzie _forward().
_redirect($url, array $options =
array())
:
przekierowuje do innej lokacji. Ta metoda przyjmuje w
parametrze URL oraz opcjonalny zestaw opcji. Domyślnie
przeprowadzane jest przekierowanie HTTP 302.
Zestaw opcji może zawierać jeden lub więcej z poniższych kluczy:
exit: określa czy skrypt ma zakończyć działanie od razu po przekierowaniu. Jeśli tak, to skrypt zamknie wszystkie otwarte sesje i przeprowadzi przekierowanie.
Możesz ustawić tę opcję globalnie wewnątrz
kontrolera używając metody dostępowej
setRedirectExit()
.
prependBase: określa czy bazowy adres URL zarejestrowany w obiekcie żądania ma być dołączony do adresu URL przekierowania.
Możesz ustawić tę opcję globalnie wewnątrz
kontrolera używając metody dostępowej
setRedirectPrependBase()
.
code: kod HTTP do użycia podczas przekierowania. Domyślnie użyty jest kod HTTP 302; może być użyty dowolny kod pomiędzy 301 a 306.
Możesz ustawić tę opcję globalnie wewnątrz
kontrolera używając metody dostępowej
setRedirectCode()
.
By design, Zend_Controller_Action
must be subclassed
in order to create an action controller. At the minimum, you will
need to define action methods that the controller may call.
Besides creating useful functionality for your web applications, you
may also find that you're repeating much of the same setup or
utility methods in your various controllers; if so, creating a
common base controller class that extends
Zend_Controller_Action
could solve such redundancy.
If a request to a controller is made that includes an undefined
action method, Zend_Controller_Action::__call()
will be invoked. __call()
is, of course, PHP's
magic method for method overloading.
Domyślnie, ta metoda wyrzuca wyjątek
Zend_Controller_Action_Exception
oznaczający, że
zażądana akcja nie została znaleziona w kontrolerze. Powinieneś
nadpisać tę funkcjonalność jeśli chcesz przeprowadzić inne
operacje.
Na przykład, jeśli chcesz wyświetlić informacje o błędzie, możesz zrobić to w taki sposób:
<?php class MyController extends Zend_Controller_Action { public function __call($method, $args) { if ('Action' == substr($method, -6)) { // Jeśli metoda akcji nie została znaleziona, renderuje szablon informujący o błędzie return $this->render('error'); } // wszystkie inne metody wyrzucają wyjątek throw new Exception('Invalid method "' . $method . '" called'); } } ?>
Inną możliwością jest przeniesienie do domyślnego kontrolera:
<?php class MyController extends Zend_Controller_Action { public function indexAction() { $this->render(); } public function __call($method, $args) { if ('Action' == substr($method, -6)) { // Jeśli metoda akcji nie została znaleziona, przenieś do akcji index return $this->_forward('index'); } // wszystkie inne metody wyrzucają wyjątek throw new Exception('Invalid method "' . $method . '" called'); } } ?>
Besides overriding __call()
, each of the
initialization, utility, accessor, view, and dispatch hook methods
mentioned previously in this chapter may be overridden in order to
customize your controllers. As an example, if you are storing your
view object in a registry, you may want to modify your
initView()
method with code resembling the following:
<?php abstract class My_Base_Controller extends Zend_Controller_Action { public function initView() { if (null === $this->view) { if (Zend_Registry::isRegistered('view')) { $this->view = Zend_Registry::get('view'); } else { $this->view = new Zend_View(); $this->view->setBasePath(dirname(__FILE__) . '/../views'); } } return $this->view; } } ?>
Hopefully, from the information in this chapter, you can see the flexibility of this particular component and how you can shape it to your application's or site's needs.