Using Zend_View Placeholders to Your Advantage
Somebody asked for some examples of how I use the headLink()
, headScript()
,
and other placeholder helpers, so I thought I'd take a crack at that today.
First off, let's look at what these helpers do. Each are concrete instances of a placeholder. In Zend Framework, placeholders are used for a number of purposes:
- Doctype awareness
- Aggregation and formatting of aggregated content
- Capturing content
- Persistence of content between view scripts and layout scripts
Let's look at these in detail.
Doctype Hinting
The HTML specification encourages you to use a DocType declaration in your HTML documents — and XHTML actually requires one. Simply put, the DocType helps tell your browser what is considered valid syntax, as well as provides some hints to how it should render.
Now, if you're like me, these are a pain to remember; the syntax is somewhat
arcane, very long, and not something I want to type very often. Fortunately, the
new doctype()
helper allows you to use mnemonics such as
XHTML1_TRANSITIONAL
or HTML4_STRICT
to invoke the appropriate doctype:
<?= $this->doctype('XHTML1_TRANSITIONAL') ?>
However, a doctype isn't just a hint to the browser; it's a contract that you need to follow. If you select a particular doctype, you're agreeing to write markup that follows the specification for it.
The doctype()
helper is actually used internally in many of the placeholder
helpers (as well as the form*()
helpers) to ensure that the markup they
generate — if any — adheres the the given doctype. However, for this to work,
you need to specify your doctype early. I recommend doing it either in your
bootstrap or in a plugin that runs before any output is emitted; typically, I
will pull the view from the ViewRenderer
in order to do so:
$viewRenderer = Zend_Controller_Action_HelperBroker::getStaticHelper('ViewRenderer');
$viewRenderer->initView();
$viewRenderer->view->doctype('XHTML1_TRANSITIONAL');
Since this sets the doctype helper's state, you can then simply echo the return value of the doctype helper later in your layout script:
<?= $this->doctype() ?>
Content Aggregation
Placeholders aggregate and store content across view instances. By aggregate, I
mean that they store the data provided in an ArrayObject
, allowing you to
collect related data for later display. Since placeholders imlement
__toString()
, and can be collections, we've added accessors to allow you to
set arbitrary text to prefix, append, and separate the items in the collection.
The various concrete placeholders — primarily the head*()
helpers — make use
of this particular feature, storing each entry as a separate item in the
collection, and then decorating them when called on to render.
Additionally, the concrete instances each contain some custom logic. In the case
of headLink()
and headScript
helpers, we perform checks to ensure that when
specifying files, duplicate entries are ignored. Why is this a good idea? Well,
since you can _forward()
to other actions, or even call the action()
view
helper, you could potentially have multiple view scripts loading the same
stylesheets or javascript; we help protect against such situations.
So, as an example:
<? // /foo/bar view script: ?>
<?
$this->headLink()->appendStylesheet('/css/foo.css');
$this->headScript()->appendFile('/js/foo.js');
echo $this->action('baz', 'foo');
?>
<? // /foo/baz view script; ?>
<?
$this->headLink()->appendStylesheet('/css/foo.css');
$this->headScript()->appendFile('/js/foo.js');
?>
FOO BAZ!
It's a contrived example, for sure, but it shows the problem quite well: if two view scripts are rendered during creation of the same content, then you have the potential for duplicate content in your placeholders. However, in this case, the duplicate content will not occur, as the helpers detect the duplicate entries when they're added, and skip them.
Capturing Content
One way in which placeholders aggregate content is by capturing content. The
base placeholder class defines both a captureStart()
and captureEnd()
method, allowing you to create content in your view scripts that you then
capture for use later.
This is particularly useful for the headScript()
helper, as it allows you to
create javascript directly in your view that will be executed in the HTML head
(or, if you use the inlineScript()
) helper, you can have it executed at the
end of your document, which is what Y!Slow recommends). The same goes for the
headStyle()
helper; you can define custom stylesheets to include directly in
your document directly with the view that needs them.
As an example, Dojo ships with some custom stylesheets for rendering its various widgits, and also has the ability to load custom classes and widgets dynamically. Let's say we want to present a Dojo ComboBox in our page: we'll need a couple of stylesheets, as well as a few Dojo resources:
First, let's tackle the stylesheets:
<? $this->headStyle()->captureStart() ?>
@import \"/js/dijit/themes/tundra/tundra.css\";
@import \"/js/dojo/resources/dojo.css\";
<? $this->headStyle()->captureEnd() ?>
These are now aggregated in our headStyle()
view helper, and we can render
them later; they will not appear inline in the page as they do here in the view
script.
Now, let's tackle the javascript. We need to load the main dojo.js
file as a
script, and then create an inline script to load our various widgets. Dojo often
uses its own custom HTML attributes, and the head*()
helpers typically don't
like this (they like to stick to those attributes defined in the specs), so
we'll need to tell the helper that this is okay so that Dojo will parse the page
when it finishes loading (to decorate our widget with the appropriate, requested
functionality).
<? $this->headScript()
->setAllowArbitraryAttributes(true)
->appendFile('/js/dojo/dojo.js', 'text/javascript', array('djConfig' => 'parseOnLoad: true'))
->captureStart() ?>
djConfig.usePlainJson=true;
dojo.require(\"dojo.parser\");
dojo.require(\"dojox.data.QueryReadStore\");
dojo.require(\"dijit.form.ComboBox\");
<? $this->headScript()->captureEnd() ?>
What's the benefit to doing this? It allows you to keep the JS and CSS functionality that's related to the specific view script at hand with that view script — you have everything in one place. If you need to change what JS or CSS is loaded, or modify the inline JS you're going to utilize, you can find it with the rest of the content to which it applies.
Putting it Together: the Layout
I keep talking about "when you render it later" in this narrative. "Later" refers to your layout script. I'm not going to go into how you initialize or define your layouts here, as it's been covered in other places. However, let's look at how we can pull in our doctype and head helpers into our layout:
<?= $this->doctype() ?>
<html>
<head>
<? // headTitle() is another concrete placeholder ?>
<?= $this->headLink() ?>
<?= $this->headStyle() ?>
<?= $this->headScript() ?>
</head>
...
Sure, you may want to put more in there than that — if you have stylesheets or scripts that load on every page, you may want to define them statically in the layout… in addition to calling the placeholder helpers. But adding the placeholder helpers gives you some definite benefits: increased separation of code, more maintainable code (as the CSS and JS specific to a view is kept with the view), simpler logic in your layouts, and the ability to prevent duplicate file inclusions.
All this functionality is now standard with Zend Framework 1.5.0; if you haven't given it a try, download it today.
Note: my colleague, Ralph Schindler — the original proposal author of
Zend_Layout
and a substantial contributor to the various placeholder helpers —
is giving a webinar on Zend_Layout and Zend_View
tomorrow, 18 March 2008; if you're interested in this topic, you should check it
out.
Updated: fixed links to layout articles.