7.7. Action Controllers

7.7.1. Wprowadzenie

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.

[Notatka] 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);
    }
}

initView(), getViewScript(), render(), and renderScript() each proxy to the ViewRenderer unless the helper is not in the helper broker or the noViewRenderer flag has been set.

Możesz także w prosty sposób wyłączyć renderowanie dla konkretnego widoku ustawiając flagę noRender w klasie ViewRenderer:

<?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 ViewRenderer are if you simply do not need a view object or if you are not rendering via view scripts (for instance, when using an action controller to serve web service protocols such as SOAP, XML-RPC, or REST). In most cases, you will never need to globally disable the ViewRenderer, only selectively within individual controllers or actions.

7.7.2. Inicjalizacja obiektu

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'
        ));
    }
}

7.7.3. Metody Pre-Dispatch oraz Post-Dispatch

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()).

7.7.4. Metody dostępowe

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] Notatka

    Metoda _getParam() może pobierać opcjonalny drugi argument zawierający domyślną wartość, ktora zostanie użyta, jeśli parametr nie został zdefiniowany lub jeśli jest pusty. Użycie drugiego parametru powoduje, że wywołanie metody _hasParam() przed odebraniem parametru nie jest konieczne:

    <?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;
    }
    ?>

7.7.5. Integracja z widokiem

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.

7.7.5.1. Inicjowanie obiektu widoku

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.

7.7.5.2. Renderowanie 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] 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 /foo.bar/baz-bat zrenderuje skrypt foo-bar/baz-bat.phtml. Jeśli nazwa metody akcji jest w postaci camelCasing, zapamiętaj, że spowoduje to rozdzieleniem słów za pomocą znaku '-' podczas określania nazwy pliku skryptu widoku.

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();
    }
}

7.7.6. Metody narzędziowe

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().

7.7.7. Rozszerzanie klasy kontrolera akcji

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.

7.7.7.1. Jak obsługiwać nieistniejące akcje

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.