Les aides de vues Dojo
Zend Framework fournit les aides de vues spécifiques à Dojo suivantes :
-
dojo(): paramètre l'environnement Dojo de votre page,
incluant les valeurs de configuration dojo, les chemins de modules personnalisés,
les appels de chargement des modules requis, les feuilles de styles des thêmes,
l'utilisation ou non d'un CDN, et d'autres encore.
Example #1 Utilisation des aides de vues Dojo
Pour utiliser les aides de vues Dojo, vous devrez informer votre objet de vue de
leur localisation. Vous pouvez faire ceci en appelant
addHelperPath() :
$view->addHelperPath('Zend/Dojo/View/Helper/',
'Zend_Dojo_View_Helper');
De manière alternative, vous pouvez utiliser la méthode enableView()
de Zend_Dojo qui le fait pour vous :
Zend_Dojo::enableView($view);
dojo() View Helper
The dojo() view helper is intended to simplify setting up
the Dojo environment, including the following responsibilities:
-
Specifying either a CDN or a local path to a Dojo install.
Specifying paths to custom Dojo modules.
Specifying dojo.require statements.
Specifying dijit stylesheet themes to use.
-
Specifying dojo.addOnLoad() events.
The dojo() view helper implementation is an example of a
placeholder implementation. The data set in it persists between view
objects and may be directly echoed from your layout script.
Example #2 dojo() View Helper Usage Example
For this example, let's assume the developer will be using Dojo from
a local path, will require several dijits, and will be
utilizing the Tundra dijit theme.
On many pages, the developer may not utilize Dojo at all. So, we
will first focus on a view script where Dojo is needed and then on the
layout script, where we will setup some of the Dojo environment and
then render it.
First, we need to tell our view object to use the Dojo view helper
paths. This can be done in your bootstrap or an early-running
plugin; simply grab your view object and execute the following:
$view->addHelperPath('Zend/Dojo/View/Helper/', 'Zend_Dojo_View_Helper');
Next, the view script. In this case, we're going to specify
that we will be using a FilteringSelect -- which will consume a
custom store based on QueryReadStore, which we'll call
'PairedStore' and store in our 'custom' module.
<?php // setup data store for FilteringSelect ?>
<div dojoType="custom.PairedStore" jsId="stateStore"
url="/data/autocomplete/type/state/format/ajax"
requestMethod="get"></div>
<?php // Input element: ?>
State: <input id="state" dojoType="dijit.form.FilteringSelect"
store="stateStore" pageSize="5" />
<?php // setup required dojo elements:
$this->dojo()->enable()
->setDjConfigOption('parseOnLoad', true)
->registerModulePath('custom', '../custom/')
->requireModule('dijit.form.FilteringSelect')
->requireModule('custom.PairedStore'); ?>
In our layout script, we'll then check to see if Dojo is enabled,
and, if so, we'll do some more general configuration and assemble
it:
<?php echo $this-> doctype() ?>
<html>
<head>
<?php echo $this-> headTitle() ?>
<?php echo $this-> headMeta() ?>
<?php echo $this-> headLink() ?>
<?php echo $this-> headStyle() ?>
<?php if ($this->dojo()->isEnabled()){
$this->dojo()->setLocalPath('/js/dojo/dojo.js')
->addStyleSheetModule('dijit.themes.tundra');
}
?>
<?php echo $this-> headScript() ?>
</head>
<body class="tundra">
<?php echo $this-> layout()-> content ?>
<?php echo $this-> inlineScript() ?>
</body>
</html>
At this point, you only need to ensure that your files are in the
correct locations and that you've created the end point action for
your FilteringSelect!
Note: UTF-8 encoding used by default
By default, Zend Framework uses UTF-8 as its default encoding, and,
specific to this case, Zend_View does as well. Character encoding
can be set differently on the view object itself using the
setEncoding() method (or the encoding
instantiation parameter). However, since Zend_View_Interface does
not define accessors for encoding, it's possible that if you are using a custom view
implementation with the Dojo view helper, you will not have a
getEncoding() method, which is what the view helper uses
internally for determining the character set in which to encode.
If you do not want to utilize UTF-8 in such a situation, you will
need to implement a getEncoding() method in your custom view
implementation.
Programmatic and Declarative Usage of Dojo
Dojo allows both declarative and
programmatic usage of many of its features.
Declarative usage uses standard HTML elements
with non-standard attributes that are parsed when the page is
loaded. While this is a powerful and simple syntax to utilize, for
many developers this can cause issues with page validation.
Programmatic usage allows the developer to
decorate existing elements by pulling them by ID or CSS selectors
and passing them to the appropriate object constructors in Dojo.
Because no non-standard HTML attributes are used, pages continue to
validate.
In practice, both use cases allow for graceful degradation when
javascript is disabled or the various Dojo script resources are
unreachable. To promote standards and document validation,
Zend Framework uses programmatic usage by default; the various view
helpers will generate javascript and push it to the
dojo() view helper for inclusion when rendered.
Developers using this technique may also wish to explore the option
of writing their own programmatic decoration of the page. One
benefit would be the ability to specify handlers for dijit events.
To allow this, as well as the ability to use declarative syntax,
there are a number of static methods available to set this behavior
globally.
Example #3 Specifying Declarative and Programmatic Dojo Usage
To specify declarative usage, simply call the static
setUseDeclarative() method:
Zend_Dojo_View_Helper_Dojo::setUseDeclarative();
If you decide instead to use programmatic usage, call the static
setUseProgrammatic() method:
Zend_Dojo_View_Helper_Dojo::setUseProgrammatic();
Finally, if you want to create your own programmatic rules, you
should specify programmatic usage, but pass in the value '-1';
in this situation, no javascript for decorating any dijits used
will be created.
Zend_Dojo_View_Helper_Dojo::setUseProgrammatic(-1);
Themes
Dojo allows the creation of themes for its dijits (widgets). You may
select one by passing in a module path:
$view->dojo()->addStylesheetModule('dijit.themes.tundra');
The module path is discovered by using the character '.' as a
directory separator and using the last value in the list as the name
of the CSS file in that theme directory to use; in the example
above, Dojo will look for the theme in
'dijit/themes/tundra/tundra.css'.
When using a theme, it is important to remember to pass the theme
class to, at the least, a container surrounding any dijits you are
using; the most common use case is to pass it in the body:
Using Layers (Custom Builds)
By default, when you use a dojo.require statement, dojo will make a
request back to the server to grab the appropriate javascript file.
If you have many dijits in place, this results in many requests to
the server -- which is not optimal.
Dojo's answer to this is to provide the ability to create
custom builds. Builds do several things:
-
Groups required files into layers; a layer
lumps all required files into a single JS file. (Hence the name
of this section.)
-
"Interns" non-javascript files used by dijits (typically,
template files). These are also grouped in the same JS file as
the layer.
-
Passes the file through ShrinkSafe, which strips whitespace and
comments, as well as shortens variable names.
Some files can not be layered, but the build process will create a
special release directory with the layer file and all other files.
This allows you to have a slimmed-down distribution customized for
your site or application needs.
To use a layer, the dojo() view helper has a
addLayer() method for adding paths to required layers:
$view->dojo()->addLayer('/js/foo/foo.js');
For more information on creating custom builds, please » refer
to the Dojo build documentation.
Methods Available
The dojo() view helper always returns an instance of
the dojo placeholder container. That container object has the
following methods available:
-
setView(Zend_View_Interface $view): set a view instance
in the container.
-
enable(): explicitly enable Dojo integration.
-
disable(): disable Dojo integration.
-
isEnabled(): determine whether or not Dojo integration
is enabled.
-
requireModule($module): setup a
dojo.require statement.
-
getModules(): determine what modules have been
required.
-
registerModulePath($module, $path):
register a custom Dojo module path.
-
getModulePaths(): get list of registered module paths.
-
addLayer($path): add a layer (custom build) path to
use.
-
getLayers(): get a list of all registered layer paths
(custom builds).
-
removeLayer($path): remove the layer
that matches $path from the list of registered
layers (custom builds).
-
setCdnBase($url): set the base URL
for a CDN; typically, one of the
Zend_Dojo::CDN_BASE_AOL or
Zend_Dojo::CDN_BASE_GOOGLE, but it only needs
to be the URL string prior to the version number.
-
getCdnBase(): retrieve the base CDN
url to utilize.
-
setCdnVersion($version = null): set
which version of Dojo to utilize from the CDN.
-
getCdnVersion(): retrieve what
version of Dojo from the CDN will be used.
-
setCdnDojoPath($path): set the relative
path to the dojo.js or dojo.xd.js
file on a CDN; typically, one of the
Zend_Dojo::CDN_DOJO_PATH_AOL or
Zend_Dojo::CDN_DOJO_PATH_GOOGLE, but it only
needs to be the path string following the version number.
-
getCdnDojoPath(): retrieve the last path segment of the
CDN url pointing to the dojo.js file.
-
useCdn(): tell the container to
utilize the CDN; implicitly enables integration.
-
setLocalPath($path): tell the container
the path to a local Dojo install (should be a path relative
to the server, and contain the dojo.js file itself);
implicitly enables integration.
-
getLocalPath(): determine what local
path to Dojo is being used.
-
useLocalPath(): is the integration
utilizing a Dojo local path?
-
setDjConfig(array $config): set
dojo or dijit configuration values (expects assoc array).
-
setDjConfigOption($option, $value): set
a single dojo or dijit configuration value.
-
getDjConfig(): get all dojo or dijit configuration
values.
-
getDjConfigOption($option, $default = null): get a
single dojo or dijit configuration value.
-
addStylesheetModule($module): add a
stylesheet based on a module theme.
-
getStylesheetModules(): get stylesheets
registered as module themes.
-
addStylesheet($path): add a local
stylesheet for use with Dojo.
-
getStylesheets(): get local Dojo stylesheets.
-
addOnLoad($spec, $function = null): add
a lambda for dojo.onLoad to call. If one argument is passed,
it is assumed to be either a function name or a javascript
closure. If two arguments are passed, the first is assumed
to be the name of an object instance variable and the second
either a method name in that object or a closure to utilize
with that object.
-
prependOnLoad($spec, $function = null):
exactly like addOnLoad(), excepts prepends to
beginning of onLoad stack.
-
getOnLoadActions(): retrieve all
dojo.onLoad actions registered with the container. This will
be an array of arrays.
-
onLoadCaptureStart($obj = null):
capture data to be used as a lambda for dojo.onLoad().
If $obj is provided, the captured JS code will be considered
a closure to use with that Javascript object.
-
onLoadCaptureEnd($obj = null): finish
capturing data for use with dojo.onLoad().
-
javascriptCaptureStart():
capture arbitrary javascript to be included with Dojo JS
(onLoad, require, etc. statements).
-
javascriptCaptureEnd(): finish capturing javascript.
-
__toString(): cast the container to a
string; renders all HTML style and script elements.
Dijit-Specific View Helpers
From the Dojo manual: "Dijit is a widget system layered on top of
dojo." Dijit includes a variety of layout and form widgets designed to
provide accessibility features, localization, and standardized (and
themeable) look-and-feel.
Zend Framework ships a variety of view helpers that allow you to render
and utilize dijits within your view scripts. There are three basic
types:
-
Layout Containers: these are designed to be
used within your view scripts or consumed by form decorators
for forms, sub forms, and display groups. They wrap the various
classes offerred in dijit.layout. Each dijit layout view helper
expects the following arguments:
-
$id: the container name or DOM ID.
-
$content: the content to wrap in the
layout container.
-
$params (optional): dijit-specific
parameters. Basically, any non-HTML attribute that can
be used to configure the dijit layout container.
-
$attribs (optional): any additional
HTML attributes that should be used to render the
container div. If the key 'id' is passed in this array, it will
be used for the form element DOM id, and
$id will be used for its name.
If you pass no arguments to a dijit layout view helper, the
helper itself will be returned. This allows you to capture
content, which is often an easier way to pass content to the
layout container. Examples of this functionality will be shown
later in this section.
-
Form Dijit: the dijit.form.Form dijit, while
not completely necessary for use with dijit form elements, will
ensure that if an attempt is made to submit a form that does
not validate against client-side validations, submission will
be halted and validation error messages raised. The form dijit
view helper expects the following arguments:
-
$id: the container name or DOM ID.
-
$attribs (optional): any additional
HTML attributes that should be used to render the
container div.
-
$content (optional): the content to wrap
in the form. If none is passed, an empty string will be
used.
The argument order varies from the other dijits in order to
keep compatibility with the standard form() view
helper.
-
Form Elements: these are designed to be
consumed with Zend_Form, but can be used
standalone within view scripts as well. Each dijit element view
helper expects the following arguments:
-
$id: the element name or DOM ID.
-
$value (optional): the current value of the element.
-
$params (optional): dijit-specific
parameters. Basically, any non-HTML attribute that can
be used to configure a dijit.
-
$attribs (optional): any additional
HTML attributes that should be used to render the dijit.
If the key 'id' is passed in this array, it will be used
for the form element DOM id, and $id
will be used for its name.
Some elements require more arguments; these will be noted with
the individual element helper descriptions.
In order to utilize these view helpers, you need to register the path
to the dojo view helpers with your view object.
Example #4 Registering the Dojo View Helper Prefix Path
$view->addHelperPath('Zend/Dojo/View/Helper', 'Zend_Dojo_View_Helper');
Dijit Layout Elements
The dijit.layout family of elements are for creating custom,
predictable layouts for your site. For any questions on general
usage, » read
more about them in the Dojo manual.
All dijit layout elements have the signature string ($id = null, $content = '',
array $params = array(), array $attribs = array()).
In all caess, if you pass no arguments, the helper object itself will be returned. This
gives you access to the captureStart() and
captureEnd() methods, which allow you to capture
content instead of passing it to the layout container.
-
AccordionContainer:
dijit.layout.AccordionContainer. Stack all panes together
vertically; clicking on a pane titlebar will expand and
display that particular pane.
<?php echo $view-> accordionContainer(
'foo',
$content,
'duration' => 200,
),
'style' => 'width: 200px; height: 300px;',
),
); ?>
-
AccordionPane:
dijit.layout.AccordionPane. For use within
AccordionContainer.
<?php echo $view-> accordionPane(
'foo',
$content,
'title' => 'Pane Title',
),
'style' => 'background-color: lightgray;',
),
); ?>
-
BorderContainer:
dijit.layout.BorderContainer. Achieve layouts with
optionally resizable panes such as you might see in a
traditional application.
<?php echo $view-> borderContainer(
'foo',
$content,
'design' => 'headline',
),
'style' => 'width: 100%; height: 100%',
),
); ?>
-
ContentPane: dijit.layout.ContentPane.
Use inside any container except AccordionContainer.
<?php echo $view-> contentPane(
'foo',
$content,
'title' => 'Pane Title',
'region' => 'left',
),
'style' => 'width: 120px; background-color: lightgray;',
),
); ?>
-
SplitContainer:
dijit.layout.SplitContainer. Allows resizable content
panes; deprecated in Dojo in favor of BorderContainer.
<?php echo $view-> splitContainer(
'foo',
$content,
'orientation' => 'horizontal',
'sizerWidth' => 7,
'activeSizing' => true,
),
'style' => 'width: 400px; height: 500px;',
),
); ?>
-
StackContainer:
dijit.layout.StackContainer. All panes within a
StackContainer are placed in a stack; build buttons or
functionality to reveal one at a time.
<?php echo $view-> stackContainer(
'foo',
$content,
'style' => 'width: 400px; height: 500px; border: 1px;',
),
); ?>
-
TabContainer:
dijit.layout.TabContainer. All panes within a
TabContainer are placed in a stack, with tabs positioned on
one side for switching between them.
<?php echo $view-> tabContainer(
'foo',
$content,
'style' => 'width: 400px; height: 500px; border: 1px;',
),
); ?>
The following capture methods are available for all layout
containers:
-
captureStart($id, array $params = array(), array $attribs =
array());: begin capturing content to include in a container.
$params refers to the dijit params to use with
the container, while $attribs refer to any
general HTML attributes to use.
Containers may be nested when capturing, so long
as no ids are duplicated.
-
captureEnd($id):
finish capturing content to include in a container.
$id should refer to an id previously used with
a captureStart() call. Returns a string
representing the container and its contents, just as if
you'd simply passed content to the helper itself.
Example #5 BorderContainer layout dijit example
BorderContainers, particularly when coupled with the ability to
capture content, are especially useful for achieving complex
layout effects.
$view->borderContainer()->captureStart('masterLayout',
array('design' => 'headline'));
'menuPane',
'This is the menu pane',
array('region' => 'top'),
array('style' => 'background-color: darkblue;')
);
'navPane',
'This is the navigation pane',
array('region' => 'left'),
array('style' => 'width: 200px; background-color: lightblue;')
);
'mainPane',
'This is the main content pane area',
array('region' => 'center'),
array('style' => 'background-color: white;')
);
'statusPane',
'Status area',
array('region' => 'bottom'),
array('style' => 'background-color: lightgray;')
);
echo $view-> borderContainer()-> captureEnd('masterLayout');
Custom Dijits
If you delve into Dojo much at all, you'll find yourself writing
custom dijits, or using experimental dijits from Dojox. While Zend
Framework cannot support every dijit directly, it does provide some
rudimentary support for arbitrary dijit types via the
CustomDijit view helper.
The CustomDijit view helper's API is exactly
that of any other dijit, with one major difference: the third "params"
argument must contain the attribute "dojotype".
The value of this attribute should be the Dijit class you plan to
use.
CustomDijit extends the base
DijitContainer view helper, which also allows it to
capture content (using the
captureStart()/ captureEnd() pair of
methods). captureStart() also expects that you pass the
"dojoType" attribute to its "params" argument.
Example #6 Using CustomDijit to render a dojox.layout.ContentPane
dojox.layout.ContentPane is a next-generation
iteration of dijit.layout.ContentPane, and provides
a superset of that class's capabilities. Until it's
functionality stabilizes, it will continue to live in Dojox.
However, if you want to use it in Zend Framework today, you can,
using the CustomDijit view helper.
At its most basic, you can do the following:
<?php echo $this-> customDijit(
'foo',
$content,
'dojoType' => 'dojox.layout.ContentPane',
'title' => 'Custom pane',
'region' => 'center'
)
); ?>
If you wanted to capture content instead, simply use the
captureStart() method, and pass the "dojoType" to
the "params" argument:
<?php $this->customDijit()->captureStart(
'foo',
'dojoType' => 'dojox.layout.ContentPane',
'title' => 'Custom pane',
'region' => 'center'
)
); ?>
This is the content of the pane
<?php echo $this-> customDijit()-> captureEnd('foo'); ?>
You can also extend CustomDijit easily to create
support for your own custom dijits. As an example, if you
extended dijit.layout.ContentPane to create your
own foo.ContentPane class, you could create the
following helper to support it:
class My_View_Helper_FooContentPane
extends Zend_Dojo_View_Helper_CustomDijit
{
protected $_defaultDojoType = 'foo.ContentPane';
public function fooContentPane(
$id = null, $value = null,
) {
return $this->customDijit($id, $value, $params, $attribs);
}
}
As long as your custom dijit follows the same basic API as
official dijits, using or extending CustomDijit
should work correctly.
|
|