Zend Framework教程-Zend_Helpers-动作助手-ActionStack,AjaxContext和ContextSwitch

以下是不常见的几个助手,知道其基本用法就可以了,用到的时候在具体查看即可。


ActionStack 用于操作动作堆栈。

ContextSwitch 和 AjaxContext 为你的动作提供替代响应格式;


ActionStack


动作堆栈助手允许把请求压到动作堆栈前端控制器插件,有效地帮助你在请求期间创建一个动作队列来执行。(动作堆栈)助手允许你通过指定新的请求对象或通过“动作/控制器/模块”集合来添加动作。 


Note: 调用动作堆栈助手来初始化动作堆栈插件
调用动作堆栈 助手暗中注册动作堆栈 插件 -- 这就意味着你不需要显性地注册动作堆栈 插件来用这个功能。 

 

Example #1 用动作、控制器和模块名来添加一个任务


经常地,仅仅指定动作、控制器和模块(和可选的参数)最简单,和调用Zend_Controller_Action::_forward()一样: 


class FooController extends Zend_Controller_Action
{
    public function barAction()
    {
        // Add two actions to the stack
        // Add call to /foo/baz/bar/baz
        // (FooController::bazAction() with request var bar == baz)
        $this->_helper->actionStack('baz',
                                    'foo',
                                    'default',
                                    array('bar' => 'baz'));


        // Add call to /bar/bat
        // (BarController::batAction())
        $this->_helper->actionStack('bat', 'bar');
    }
}


        Example #2 使用请求对象添加一个任务


有时候请求对象的OOP本性很有用;你也可以传递这样一个对象给动作堆栈助手。 


class FooController extends Zend_Controller_Action
{
    public function barAction()
    {
        // Add two actions to the stack
        // Add call to /foo/baz/bar/baz
        // (FooController::bazAction() with request var bar == baz)
        $request = clone $this->getRequest();
        // Don't set controller or module; use current values
        $request->setActionName('baz')
                ->setParams(array('bar' => 'baz'));
        $this->_helper->actionStack($request);


        // Add call to /bar/bat
        // (BarController::batAction())
        $request = clone $this->getRequest();
        // don't set module; use current value
        $request->setActionName('bat')
                ->setControllerName('bar');
        $this->_helper->actionStack($request);
    }
}





ContextSwitch and AjaxContext
ContextSwitch 动作助手用来使在请求后返回不同的响应格式变得容易。AjaxContext 助手是 ContextSwitch 的特别版本,用来返回响应到 XmlHttpRequests。 


你必须在关于动作能够响应哪个上下文(context)的控制器中打开(enable)其中一个。如果一个进来的请求对给定的动作指出一个有效的上下文,助手将做: 


如果布局被打开,则关闭它。 


设置一个备用的视图后缀,为上下文有效地请求一个分离的视图脚本。 


为期望的上下文发送适当的响应头。 


可选地调用特别的回调(callback)来设置上下文和/或执行处理后的任务。 


例子,考虑下列的控制器: 


class NewsController extends Zend_Controller_Action
{
    /**
     * Landing page; forwards to listAction()
     */
    public function indexAction()
    {
        $this->_forward('list');
    }


    /**
     * List news items
     */
    public function listAction()
    {
    }


    /**
     * View a news item
     */
    public function viewAction()
    {
    }
}




    我们想让 listAction() 也支持 XML 格式,不用创建一个不同的动作,我们可以提示它可以返回 XML 响应: 


class NewsController extends Zend_Controller_Action
{
    public function init()
    {
        $contextSwitch = $this->_helper->getHelper('contextSwitch');
        $contextSwitch->addActionContext('list', 'xml')
                      ->initContext();
    }


    // ...
}




    这将完成: 


设置 'Content-Type' 响应头为 'text/xml'. 


修改视图后缀为 'xml.phtml' (或者,如果你使用另外的视图后缀,'xml.[你的后缀]')。 


现在你需要创建一个新的视图脚本 'news/list.xml.phtml',它将创建和解析 XML。 


为决定一个请求是否应该初始化一个上下文开关(context switch),这个助手检查在请求对象理的令牌。缺省地,它寻找一个 'format' 参数,尽管这是可配置的。这意味着,在大多数情况下,为触发一个上下文开关,你可以添加一个 'format' 参数给你的请求: 


通过 URL 参数:/news/list/format/xml (回忆一下,缺省路由模式(schema)允许在动作后跟随任意的键/值对) 


通过 GET 参数:/news/list?format=xml 


ContextSwitch 允许指定任意的上下文,包括需要修改后缀的、任何应该被发送的响应头和任意的用来初始化和善后的回调(callback)。 


缺省可用的上下文 
缺省地,ContextSwitch 助手的两个上下文可用:json 和 xml。 


JSON. JSON 上下文设置 'Content-Type' 响应头为 'application/json',设置视图脚本后缀为 'json.phtml'。 


缺省地,不需要视图脚本,它将系列化所有的视图变量,立即发出 JSON 响应。 


通过关闭 auto-JSON serialization 可禁止这个行为: 


$this->_helper->contextSwitch()->setAutoJsonSerialization(false);


                XML. XML 上下文设置 'Content-Type' 响应头为 'text/xml',设置视图脚本后缀为 'xml.phtml'。你需要为这个上下文创建新的视图脚本。 


创建定制的上下文 
有时候,缺省的上下文不够用,例如你需要返回 YAML,或系列化 PHP、RSS 或 ATOM feed 等等,ContextSwitch 正是你需要的东西。 


最容易的添加新的上下文的办法是通过 addContext() 方法。这个方法带有两个参数:上下文的名称和一个规范(specification)数组,规范应当包含下列中的一个或多个: 


suffix: 当在视图解析器里注册时,预先准备的缺省的视图后缀。 

headers: 作为响应的一部分发送的头/值(header/value)对的数组。

callbacks: 数组,包含一个或更多键 'init' 或 'post',指向有效的可用于上下文和善后处理的 PHP 回调(callback)。


回调的初始化发生在当 ContextSwitch 检测到上下文时,你可以用它来执行任意的应该发生的逻辑。作为例子,当 auto-JSON serialization 是 on 的时候,JSON 上下文使用回调来关闭视图解析器(ViewRenderer)。 


善后处理(post processing)发生在动作的 postDispatch() 程序期间,可用来执行任意的逻辑。作为例子,JSON 上下文使用回调(callback)来决定是否 auto-JSON serialization 是 on;如果是,它系列化视图变量到 JSON 和 发送响应,如果不是,它重新打开(re-enable)视图解析器(ViewRenderer)。 


和上下文交互作用的方法: 


addContext($context, array $spec): 添加新的上下文,如果上下文存在,抛出一个异常。 

setContext($context, array $spec): 添加新的上下文或重写一个已存在的上下文,和 addContext() 使用相同的规范(specification)。 

addContexts(array $contexts): 一次添加多个上下文。$contexts 应当是上下文/规范(context/specification)对的数组。如果任何一个上下文存在,就抛出异常。 

setContexts(array $contexts): 添加和重写存在的上下文(多于一个),和 addContexts() 使用相同的规范。 

hasContext($context): 如果上下文存在,返回 true,否则返回 false。 

getContext($context): 通过名称来获取一个单个的上下文,返回一个遵循用于 addContext() 的规范的数组。 

getContexts(): 获取所有的上下文,返回上下文/规范对的数组。 

removeContext($context): 通过名称来清除一个单个的上下文,成功返回 true,如果没有发现上下文返回 false。 

clearContexts(): 清除所有上下文。 


为每个动作设置上下文 
有两个设置可用的上下文的机制:或者在控制器里手工创建数组,或者使用在 ContextSwitch 里的方法来装配它们。 


添加动作/上下文关系的基本方法是 addActionContext() 。它有两个参数:上下文被添加到的动作和上下文的名称或上下文数组的其中之一。作为例子,考虑下列的控制器类: 


class FooController extends Zend_Controller_Action
{
    public function listAction()
    {
    } 
    public function viewAction()
    {
    } 
    public function commentsAction()
    {
    } 
    public function updateAction()
    {
    }
}




        假如我们想添加 XML 上下文到 'list'动作、XML 和 JSON 上下文到 'comments' 动作,可以在 init() 方法中完成: 


class FooController extends Zend_Controller_Action
{
    public function init()
    {
        $this->_helper->contextSwitch()
             ->addActionContext('list', 'xml')
             ->addActionContext('comments', array('xml', 'json'))
             ->initContext();
    }
}




        另外,还可以定义数组属性 $contexts: 


class FooController extends Zend_Controller_Action
{
    public $contexts = array(
        'list'     => array('xml'),
        'comments' => array('xml', 'json')
    );


    public function init()
    {
        $this->_helper->contextSwitch()->initContext();
    }
}




        上述不但缺少周密考虑,而且有潜在的错误。 


下面的方法可用来构造上下文映射: 


addActionContext($action, $context): 标记一个或多个可用的上下文给一个动作,如果映射已经存在,就追加到那些映射中。$context 可以是一个单个的上下文,或者是一个上下文数组。 


一个上下文的 true 值将标记所有可用的上下文给动作。 

一个 $context 的空值将关闭(disable)所有给定动作的上下文。 

setActionContext($action, $context): 标记一个或多个上下文对动作可用,如果映射已经存在,它就替换它们。$context 可以是一个单个的上下文,或者是一个上下文数组。 

addActionContexts(array $contexts): 一次性添加若干动作/上下文对,$contexts 是动作/上下文对的关联数组。它代理 addActionContext(),意味着如果那些动作/上下文对存在,就追加之。 

setActionContexts(array $contexts): 和 addActionContexts() 一样,但重写已存在的动作/上下文对。 

hasActionContext($action, $context): 决定一个特定的动作是否有一个给定的上下文。 

getActionContexts($action = null): 返回或者给定动作的所有上下文,或者所有动作/上下文对。 

removeActionContext($action, $context): 从给定动作中清除一个或多个上下文。$context 可以是一个单个的上下文,或者是一个上下文数组。 

clearActionContexts($action = null): 从给定动作或者从有上下文的动作中清除所有上下文。 


初始化上下文开关 
为初始化上下文开关,需要在动作控制器中调用 initContext(): 


class NewsController extends Zend_Controller_Action
{
    public function init()
    {
        $this->_helper->contextSwitch()->initContext();
    }
}




        在某些情况下,你想强制使用上下文,例如,如果上下文开关是激活状态,你只想用 XML 上下文,可以通过传递上下文给 initContext() 来完成: 


$contextSwitch->initContext('xml');


        另外的功能 
用来改变 ContextSwitch 助手行为的方法包括: 


setAutoJsonSerialization($flag): 缺省地,JSON 上下文将系列化任何视图变量给 JSON 符号并把它作为响应返回。如果想创建自己的响应,你需要关闭它,这需要在调用 initContext() 之前来完成。 


$contextSwitch->setAutoJsonSerialization(false);
$contextSwitch->initContext();


                用 getAutoJsonSerialization() 来获取 flag 的值。 


setSuffix($context, $suffix, $prependViewRendererSuffix): 用这个方法,你可以为给定的上下文指定一个不同的后缀。第三个参数用来指示是否用新的后缀预先准备当前视图解析器,这个 flag 缺省为打开。 


传递空值给后缀将导致只有视图解析器的后缀被使用。 


addHeader($context, $header, $content): 为给定的上下文添加一个响应头,$header 是头的名称,$content 是为这个头传递的值。 


每个上下文可以有多个头,addHeader() 把另外的头添加到头的堆栈。 


如果为上下文指定的 $header 已经存在,就抛出一个异常。 


setHeader($context, $header, $content): setHeader() 和 addHeader() 一样,但它允许重写已经存在的上下文的头。 


addHeaders($context, array $headers): 一次性添加多个头到给定的上下文,代理 addHeader(),如果头已经存在,将抛出异常。$headers 是一个头/上下文对的数组。 


setHeaders($context, array $headers.): 象 addHeaders() 一样,但代理 setHeader(),允许重写已存在的头。 


getHeader($context, $header): 获取给定上下文的头的值,如果没有发现返回 null。 


removeHeader($context, $header): 清除一个单个的给定上下文的头。 


clearHeaders($context, $header): 清除所有给定上下文的头。 


setCallback($context, $trigger, $callback): 为给定的上下文在给定的触发器设置回调(callback),触发器或者是 'init',或者是 'post'(指明回调将被在上下文初始化时或者派遣后(postDispatch)调用)。 $callback 应当是一个有效的 PHP 回调。 


setCallbacks($context, array $callbacks): 为给定的上下文设置多个回调,$callbacks 应当是触发器/回调对。事实上,最常用的可注册的回调有两个:一个为初始化用的,一个是做善后处理。 


getCallback($context, $trigger): 在给定的上下文中从给定的触发器中获取一个回调。 


getCallbacks($context): 从给定的上下文获取所有的回调,返回一个触发器/回调对数组。 


removeCallback($context, $trigger): 从给定的触发器和上下文清除一个回调。 


clearCallbacks($context): 为给定的上下文清除所有的回调。 


setContextParam($name): 设置请求参数来检查什么时候决定上下文开关是否已经请求,缺省值为 'format',但这个访问器可以用来设置一个备用的值。 


getContextParam() 用来获取当前的值。 


setAutoDisableLayout($flag): 缺省地,当上下文开关出现,布局是关闭的;这是因为一般布局将只用来返回正常的相应,对备用的(alternate)上下文没有意义。然而,如果想使用布局(也许你对新的上下文有个布局),你可以通过传递一个 false 的值给 setAutoDisableLayout() 来改变它的行为。这应当 在 调用 initContext() 之前 来做。 


用访问器 getAutoDisableLayout() 来获取这个 flag 的值。 


getCurrentContext() 可用来确定什么上下文被检测到,如果有的话。如果没有上下文开关发生,或者在调用 initContext() 之前调用它,则返回 null。 


AjaxContext 函数 
AjaxContext 助手继承 ContextSwitch,所有 ContextSwitch 函数列表对它也有效,只是有些小小的不同。 


首先,它为确定上下文使用不同的动作控制器属性 - $ajaxable。这样你就可以对 AJAX 和 普通的 HTTP 请求使用不同的上下文。AjaxContext 的各种各样 *ActionContext*() 方法将写到这个属性。 


其次,由请求对象的 isXmlHttpRequest() 方法来确定,它只有在 XmlHttpRequest 出现后才触发。这样,如果上下文参数 ('format')在请求中传递,但请求没有做成 XmlHttpRequest,不会触发上下文开关。 


第三,AjaxContext 添加另外的上下文 - HTML。在这个上下文中,为了区别这个上下文和普通的请求,它设置后缀为 'ajax.phtml'。没有另外的头返回。 


Example #4 允许动作响应 Ajax 的请求 


在下面的例子中,我们允许对动作 'view'、 'form' 和 'process' 的请求响应 AJAX 的请求。在头两个例子 'view' 和 'form',返回不更新页面的 HTML 片段;在最后的例子,返回 JSON。 


class CommentController extends Zend_Controller_Action
{
    public function init()
    {
        $ajaxContext = $this->_helper->getHelper('AjaxContext');
        $ajaxContext->addActionContext('view', 'html')
                    ->addActionContext('form', 'html')
                    ->addActionContext('process', 'json')
                    ->initContext();
    }


    public function viewAction()
    {
        // Pull a single comment to view.
        // When AjaxContext detected, uses the comment/view.ajax.phtml
        // view script.
    }


    public function formAction()
    {
        // Render the "add new comment" form.
        // When AjaxContext detected, uses the comment/form.ajax.phtml
        // view script.
    }


    public function processAction()
    {
        // Process a new comment
        // Return the results as JSON; simply assign the results as
        // view variables, and JSON will be returned.
    }
}



            在客户端,AJAX 库将请求终点 '/comment/view'、 '/comment/form' 和 '/comment/process',并传递 'format' 参数:'/comment/view/format/html'、'/comment/form/format/html' 和 '/comment/process/format/json'。(或者你可以通过查询字符串( query string) 传递参数,如:"?format=json") 


假定你的库传递 'X-Requested-With:XmlHttpRequest'头,这些动作将返回适当的响应格式。 



你可能感兴趣的:(ZendFramework,Zend,Framework)