In der vorherigen Sektion, haben
wir den placeholder()
View Helfer kennengelernt, und wie er für das
Sammeln eigener Inhalte verwendet werden kann. In dieser Sektion sehen wir uns einige
konkrete Platzhalter an die mit Zend Framework ausgeliefert werden, und wie Sie für den
eigenen Vorteil verwendet werden können wenn komplexe kombinierte Layouts erstellt werden.
Die meisten mitgelieferten Platzhalter sind für die Erstellung von Inhalten der <head> Sektion des Layout Inhalts -- ein Areal das man normalerweise nicht direkt über Anwendungs View Skripte manipuliert, aber eines das man beeinflussen will. Als Beispiel: Wenn man will das der Titel auf jeder Seite bestimmte Inhalte enthält, aber spezifische Inhalte auf dem Controller und/oder der Aktion basieren; wenn man CSS Dateien spezifizieren will um Sie basieren auf der Sektion in der man sich befindet zu Laden; wenn man spezifische JavaScript Skripts zu unterschiedlichen Zeiten laden will; oder wenn man die DocType Deklaration setzen will.
Zend Framework wird mit einer Platzhalter Implementation für jede dieser Situationen ausgeliefert, und verschiedenen mehr.
Die DocType Deklaration ist problematisch zu merken, und oft ist es
essentiell Sie in die Dokumente einzubinden um sicherzustellen das der Browser den
Inhalt richtig darstellt. Der doctype()
View Helfer erlaubt es
einfache String Merkmale zu verwenden um den gewünschten DocType zu
spezifizieren; zusätzlich fragen andere Helfer den doctype()
ab, um sicherzustellen dass die erzeugte Ausgabe mit dem angefragten
DocType übereinstimmt.
Wenn man als Beispiel XHTML1 Strict DTD verwenden will kann man einfach folgendes spezifizieren:
$this->doctype('XHTML1_STRICT');
Neben anderen vorhandenen Gedächnishilfen sind die folgenden Typen zu finden:
- XHTML1_STRICT
-
XHTML 1.0 Strict
- XHTML1_TRANSITIONAL
-
XHTML 1.0 Transitional
- HTML4_STRICT
-
HTML 4.01 Strict
- HTML4_Loose
-
HTML 4.01 Loose
- HTML5
-
HTML 5
Man kann den Typ zuordnen und die Deklaration in einem einzelnen Aufruf darstellen:
echo $this->doctype('XHTML1_STRICT');
Trotzdem ist der bessere Ansatz den Typ in der Bootstrap zuzuordnen, und Ihn dann im Layout darzustellen. Man könnte versuchen das folgende in der Bootstrap Klasse hinzuzufügen:
class Bootstrap extends Zend_Application_Bootstrap_Bootstrap { protected function _initDocType() { $this->bootstrap('View'); $view = $this->getResource('View'); $view->doctype('XHTML1_STRICT'); } }
Und dann im Layout Skript einfach auf dem Helfer am Beginn der Datei
echo()
ausrufen:
<?php echo $this->doctype() ?> <html> <!-- ... -->
Das stellt sicher dass DocType-beachtende View Helfer das richtige Markup darstellen, das der Typ richtig gesetzt ist bevor das Layout dargestellt wird, und bietet einen einzelnen Ort an dem der DocType geändert werden kann.
Oft will eine Site den Namen der Site oder der Firma als Teil des Seitentitels einfügen,
und dann zusätzliche Informationen basieren auf dem Ort in dieser Site einfügen. Als
Beispiel enthält die Website zend.com
den String
"Zend.com
" auf allen Seiten und fügt Informationen basierend auf
der Seite voran: "Zend Server - Zend.com
". Im Zend Framework
kann der headTitle()
View Helfer helfen diese Aufgabe zu
vereinfachen.
Am einfachsten erlaubt es der headTitle()
Helfer den Inhalt zu
für das <title> Tag zu sammeln; wenn man es ausgibt, wird es
basierend auf der Reihenfolge mit der es hinzugefügt wurde zusammengefügt. Man kann die
Reihenfolge kontrollieren indem prepend()
und
append()
verwendet werden, und einen Separator angegeben
welcher zwischen den Segmenten zu verwenden ist, indem die Methode
setSeparator()
verwendet wird.
Typischerweise sollten jene Segmente die in allen Seiten gemeinsam sind in der Bootstrap
spezifiziert werden, ähnlich wie wir den DocType definiert haben. In diesem Fall
definieren wir eine _initPlaceholders()
Methode um auf den
verschiedenen Platzhaltern zu arbeiten, und einen initialen Titel sowie einen Separator
zu spezifizieren.
class Bootstrap extends Zend_Application_Bootstrap_Bootstrap { // ... protected function _initPlaceholders() { $this->bootstrap('View'); $view = $this->getResource('View'); $view->doctype('XHTML1_STRICT'); // Setzt den initialen Titel und Separator: $view->headTitle('Meine Site') ->setSeparator(' :: '); } // ... }
Im View Skript könnten wir ein weiteres Segment hinzufügen:
<?php $this->headTitle()->append('Eine Seite'); // Nach anderen Segmenten platzieren ?> <?php $this->headTitle()->prepend('Eine Seite'); // Davor platzieren ?>
In unserem Layout geben wie den headTitle()
Helfer einfach aus:
<?php echo $this->doctype() ?> <html> <?php echo $this->headTitle() ?> <!-- ... -->
Das erzeugt die folgende Ausgabe:
<!-- Wenn append() verwendet wurde: --> <title>Meine Site :: Eine Seite</title> <!-- Wenn prepend() verwendet wurde: --> <title>Eine Seite :: Meine Site</title>
Gute CSS Entwickler erstellen oft ein generisches Stylesheet für
Siteweite Stile, und individuelle Stylesheets für spezifische Sektionen oder Seite der
Website, und laden die zweiteren über Konditionen um die Menge der Daten zu verringern
die bei jeder Anfrage übertragen werden müssen. Der headLink()
Platzhalter macht die Sammlung von solchen konditionellen Stylesheets in der Anwendung
trivial.
Um das zu ermöglichen definiert headLink()
eine Anzahl von
"virtuellen" Methoden (durch Überladen) welche den Prozess trivial machen. Jene mit
denen wir uns befassen sind appendStylesheet()
und
prependStylesheet()
. Jede nimmt bis zu vier Argumente,
$href
(den relativen Pfad zum Stylesheet), $media
(den MIME Typ, der standardmäßig "text/css" ist),
$conditionalStylesheet
(kann verwendet werden um eine "Kondition" zu
spezifizieren bei dem das Stylesheet evaluiert wird), und $extras
(ein assoziatives Array von Schlüssel und Werte Paare, üblicherweise verwendet um einen
Schlüssel für "media" zu definieren). In den meisten Fällen muss man nur das erste
Argument spezifizieren, den relativen Pfad zum Stylesheet.
In unserem Beispiel nehmen wir an das alle Seiten das Stylesheet laden mussen welches in
"/styles/site.css
" vorhanden ist (relativ zum Dokument Root); wir
spezifizieren dass in unserer Bootstrap Methode
_initPlaceholders()
.
class Bootstrap extends Zend_Application_Bootstrap_Bootstrap { // ... protected function _initPlaceholders() { $this->bootstrap('View'); $view = $this->getResource('View'); $view->doctype('XHTML1_STRICT'); // Setzt den initialen Titel und Separator: $view->headTitle('Meine Site') ->setSeparator(' :: '); // Setzt das initiale Stylesheet: $view->headLink()->prependStylesheet('/styles/site.css'); } // ... }
Später, im Controller oder einem Action-spezifischen View Skript, können wir weitere Stylesheets hinzufügen:
<?php $this->headLink()->appendStylesheet('/styles/user-list.css') ?>
In unserem Layout View Skript geben wir den Platzhalter einfach wieder aus:
<?php echo $this->doctype() ?> <html> <?php echo $this->headTitle() ?> <?php echo $this->headLink() ?> <!-- ... -->
Das erzeugt die folgende Ausgabe:
<link rel="stylesheet" type="text/css" href="/styles/site.css" /> <link rel="stylesheet" type="text/css" href="/styles/user-list.css" />
Eine andere gemeinsame Taktik um lange Ladezeiten bei Seiten zu verhindern besteht darin
JavaScript nur dann zu Laden wenn es notwendig ist. So gesehen benötigt man verschiedene
Layer von Skripten: möglicherweise einen für die fortlaufende Verbesserung der Menüs der
Site, und einen weiteren für Seiten-spezifische Inhalte. In diesen Situationen bietet
der headScript()
Helfer eine Lösung.
Ähnlich wie der headLink()
Helfer bietet
headScript()
die Möglichkeit Skripte der Sammlung anzuhängen
oder voranzustellen, und dann das komplette Set auszugeben. Es bietet die Flexibilität
Skriptdateien zu spezifizieren damit diese selbst geladen werden, als auch explizit
JavaScript. Man hat auch die Option JavaScript über
captureStart()
/captureEnd()
einzufangen, was es erlaubt JavaScript einfach im Code zu haben statt notwendigerweise
einen zusätzlichen Aufruf zum Server zu machen.
So wie headLink()
bietet headScript()
"virtuelle" Methoden durch Überladen als Bequemlichkeit wenn Elemente spezifiziert
werden um Sie zu sammeln; übliche Methoden sind prependFile()
,
appendFile()
, prependScript()
, und
appendScript()
. Die ersten zwei erlauben es Dateien zu
spezifizieren auf die im $src
Attribut des
<script> Tags referenziert wird; die letzteren zwei nehmen
den angegebenen Inhalt und stellen Ihn als literales JavaScript im
<script> Tag dar.
In diesem Beispiel spezifizieren wir ein Skript, "/js/site.js
"
muss bei jeder Seite geladen werden; wir aktualisieren die
_initPlaceholders()
Methode in der Bootstrap um das zu tun.
class Bootstrap extends Zend_Application_Bootstrap_Bootstrap { // ... protected function _initPlaceholders() { $this->bootstrap('View'); $view = $this->getResource('View'); $view->doctype('XHTML1_STRICT'); // Setzt den initialen Titel und Separator: $view->headTitle('My Site') ->setSeparator(' :: '); // Setzt das initiale Stylesheet: $view->headLink()->prependStylesheet('/styles/site.css'); // Setzt das initiale JS das geladen werden soll: $view->headScript()->prependFile('/js/site.js'); } // ... }
In einem View Skript können wir dann eine extra Skript Datei der Quelle hinzufügen um etwas JavaScript zu sammeln und es in unserem Dokument einzufügen.
<?php $this->headScript()->appendFile('/js/user-list.js') ?> <?php $this->headScript()->captureStart() ?> site = { baseUrl: "<?php echo $this->baseUrl() ?>" }; <?php $this->headScript()->captureEnd() ?>
In unserem Layout Skript wird der Platzhalter dann einfach ausgegeben, wie wir es bereits bei den anderen gemacht haben:
<?php echo $this->doctype() ?> <html> <?php echo $this->headTitle() ?> <?php echo $this->headLink() ?> <?php echo $this->headScript() ?> <!-- ... -->
Das erstellt die folgende Ausgabe:
<script type="text/javascript" src="/js/site.js"></script> <script type="text/javascript" src="/js/user-list.js"></script> <script type="text/javascript"> site = { baseUrl: "<?php echo $this->baseUrl() ?>" }; </script>
InlineScript Variante
Viele Browser blockieren oft die Anzeige von Seiten bis alle Skripte und Stylesheets geladen wurden auf die in der <head> Sektion referenziert wird. Wenn man eine Anzahl solcher Direktiven hat, kann das Einfluß darauf haben wie bald jemand damit beginnen kann sich die Seite anzuschauen:
Ein Weg darum zu kommen besteht darin die <script> Tags einfach nach dem schließenden <body> Tag des Dokuments auszugeben. (Das ist eine Praxis die speziell vom Y! Slow Projekt empfohlen wird)
Zend Framework unterstützt das auf zwei unterschiedlichen Wegen:
-
Man kann das
headScript()
Tag im Layout Skript überall wo man will darstellen; nur weil der Titel auf "head" referenziert heißt das nicht dass er an dieser Position dargestellt werden muss. -
Alternativ kann der
inlineScript()
Helfer verwendet werden der einfach eine Variante vonheadScript()
ist und das selbe Verhalten hat, aber eine eigene Registry verwendet.