view-placeholders-standard.xml 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 24249 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="learning.view.placeholders.standard">
  5. <title>Placeholders standards</title>
  6. <para>
  7. Dans la <link linkend="learning.view.placeholders.basics">section précédente</link>, nous
  8. avons vu l'aide de vue <methodname>placeholder()</methodname> et comment l'utiliser pour
  9. aggréger du contenu personnalisable. Dans ce chapitre, nous allons passer en revue quelques
  10. placeholders concrets fournis avec Zend Framework, ainsi que la manière de les utiliser à
  11. votre avantage pour créer des layouts complexes.
  12. </para>
  13. <para>
  14. La plupart des placeholders fournis permettent de gérer le contenur de la section
  15. <emphasis>&lt;head&gt;</emphasis> de la layout -- une zone qui ne peut typiquement pas être
  16. manipulée directement par vos scripts de vue, mais que vous voulez tout de même traiter.
  17. Par exemples: vous voudriez que votre titre se compose d'un certain contenu sur toutes les
  18. pages mais aussi d'une partie dynamique relative au contrôleur/action en cours; aussi vous
  19. voudriez préciser des fichiers <acronym>CSS</acronym> à charger basés sur la section de
  20. l'application en cours; enfin vous pourriez avoir recours à des scripts JavaScript
  21. spécifiques parfois, ou encore changer la déclaration de <emphasis>DocType</emphasis>.
  22. </para>
  23. <para>
  24. Zend Framework est livré avec des implémentations de placeholder pour chacune de ces
  25. situations et encore d'autres.
  26. </para>
  27. <sect2 id="learning.view.placeholders.standard.doctype">
  28. <title>Changer le DocType</title>
  29. <para>
  30. Les déclarations de <emphasis>DocType</emphasis> sont difficiles à mémoriser et souvent
  31. essentielles pour s'assurer que le navigateur rende correctement le contenu. L'aide de
  32. vue <methodname>doctype()</methodname> permet d'utiliser des mnemonics pour spécifier un
  33. <emphasis>DocType</emphasis>; aussi, d'autres aides de vues interrogeront l'aide
  34. <methodname>doctype()</methodname> pour s'assurer que le contenu qu'elles génèrent est
  35. conforme au <emphasis>DocType</emphasis> utilisé.
  36. </para>
  37. <para>
  38. Par exemple si vous souhaitez utiliser la <acronym>DTD</acronym>
  39. <acronym>XHTML1</acronym> Strict, vous pouvez simplement la préciser
  40. comme ceci&#160;:
  41. </para>
  42. <programlisting language="php"><![CDATA[
  43. $this->doctype('XHTML1_STRICT');
  44. ]]></programlisting>
  45. <para>
  46. Voici les autres mnemonics utilisables&#160;:
  47. </para>
  48. <variablelist>
  49. <varlistentry>
  50. <term>XHTML1_STRICT</term>
  51. <listitem>
  52. <para>
  53. <acronym>XHTML</acronym> 1.0 Strict
  54. </para>
  55. </listitem>
  56. </varlistentry>
  57. <varlistentry>
  58. <term>XHTML1_TRANSITIONAL</term>
  59. <listitem>
  60. <para>
  61. <acronym>XHTML</acronym> 1.0 Transitional
  62. </para>
  63. </listitem>
  64. </varlistentry>
  65. <varlistentry>
  66. <term>HTML4_STRICT</term>
  67. <listitem>
  68. <para>
  69. <acronym>HTML</acronym> 4.01 Strict
  70. </para>
  71. </listitem>
  72. </varlistentry>
  73. <varlistentry>
  74. <term>HTML4_Loose</term>
  75. <listitem>
  76. <para>
  77. <acronym>HTML</acronym> 4.01 Loose
  78. </para>
  79. </listitem>
  80. </varlistentry>
  81. <varlistentry>
  82. <term>HTML5</term>
  83. <listitem>
  84. <para>
  85. <acronym>HTML</acronym> 5
  86. </para>
  87. </listitem>
  88. </varlistentry>
  89. </variablelist>
  90. <para>
  91. Vous pouvez changer le type et rendre la déclaration en un seul appel&#160;:
  92. </para>
  93. <programlisting language="php"><![CDATA[
  94. echo $this->doctype('XHTML1_STRICT');
  95. ]]></programlisting>
  96. <para>
  97. Cependant l'approche conseillée est de préciser le type dans le bootstrap et rendre
  98. l'aide de vue dans la layout. Essayez d'ajouter ceci à votre classe de bootstrap&#160;:
  99. </para>
  100. <programlisting language="php"><![CDATA[
  101. class Bootstrap extends Zend_Application_Bootstrap_Bootstrap
  102. {
  103. protected function _initDocType()
  104. {
  105. $this->bootstrap('View');
  106. $view = $this->getResource('View');
  107. $view->doctype('XHTML1_STRICT');
  108. }
  109. }
  110. ]]></programlisting>
  111. <para>
  112. Puis, dans le script de layout, affichez simplement avec
  113. <methodname>echo()</methodname> l'aide en haut du fichier&#160;:
  114. </para>
  115. <programlisting language="php"><![CDATA[
  116. <?php echo $this->doctype() ?>
  117. <html>
  118. <!-- ... -->
  119. ]]></programlisting>
  120. <para>
  121. Ceci permet d'être sûr que les aides de vue diverses utiliseront cette déclaration,
  122. que le docType est précisé avant le rendu du layout et qu'il n'existe qu'un seul
  123. endroit logique pour le changer.
  124. </para>
  125. </sect2>
  126. <sect2 id="learning.view.placeholders.standard.head-title">
  127. <title>Spécifier le titre de la page</title>
  128. <para>
  129. Souvent, le site incluera le nom de la société dans le titre de la page et ajoutera
  130. ensuite des informations basées sur la page en cours de lecture. Par exemple, le site
  131. <filename>zend.com</filename> inclut la chaine "<filename>Zend.com</filename>" sur
  132. toutes les pages et y fait précèder des informations relatives à la page en
  133. cours&#160;: "Zend Server - <filename>Zend.com</filename>". Dans Zend Framework,
  134. l'aide de vue <methodname>headTitle()</methodname> peut vous simplifier cette tâche.
  135. </para>
  136. <para>
  137. Au plus simple, l'aide <methodname>headTitle()</methodname> permet d'aggréger du
  138. contenu pour la balise <emphasis>&lt;title&gt;</emphasis>; lorsque vous l'affichez,
  139. il assemble son contenu dans l'ordre des ajouts. Pour contrôler l'ordre, les méthodes
  140. <methodname>prepend()</methodname> et <methodname>append()</methodname> sont là, pour
  141. changer le séparateur à utiliser entre les segments, utilisez la méthode
  142. <methodname>setSeparator()</methodname>.
  143. </para>
  144. <para>
  145. Typiquement vous devriez renseigner tous les segments communs à toutes les pages en
  146. bootstrap, de la même manière que nous avions agit avec le doctype. Dans ce cas, nous
  147. allons écrire une méthode <methodname>_initPlaceholders()</methodname> pour gérer
  148. tous les placeholders et préciser un titre initial ainsi qu'un séparateur.
  149. </para>
  150. <programlisting language="php"><![CDATA[
  151. class Bootstrap extends Zend_Application_Bootstrap_Bootstrap
  152. {
  153. // ...
  154. protected function _initPlaceholders()
  155. {
  156. $this->bootstrap('View');
  157. $view = $this->getResource('View');
  158. $view->doctype('XHTML1_STRICT');
  159. // Précise le titre initial et le séparateur:
  160. $view->headTitle('My Site')
  161. ->setSeparator(' :: ');
  162. }
  163. // ...
  164. }
  165. ]]></programlisting>
  166. <para>
  167. Dans un script de vue, nous voulons ajouter un nouveau segment&#160;:
  168. </para>
  169. <programlisting language="php"><![CDATA[
  170. <?php $this->headTitle()->append('Some Page'); // placé après les autres segments ?>
  171. <?php $this->headTitle()->prepend('Some Page'); // placé avant ?>
  172. ]]></programlisting>
  173. <para>
  174. Dans notre layout, nous affichons simplement l'aide
  175. <methodname>headTitle()</methodname>&#160;:
  176. </para>
  177. <programlisting language="php"><![CDATA[
  178. <?php echo $this->doctype() ?>
  179. <html>
  180. <?php echo $this->headTitle() ?>
  181. <!-- ... -->
  182. ]]></programlisting>
  183. <para>
  184. Le contenu suivant aura été généré&#160;:
  185. </para>
  186. <programlisting language="html"><![CDATA[
  187. <!-- Si append() a été utilisé: -->
  188. <title>My Site :: Some Page</title>
  189. <!-- Si prepend() a été utilisé: -->
  190. <title>Some Page :: My Site</title>
  191. ]]></programlisting>
  192. </sect2>
  193. <sect2 id="learning.view.placeholders.standard.head-link">
  194. <title>Spécifier des feuilles de style avec HeadLink</title>
  195. <para>
  196. Les bons développeurs <acronym>CSS</acronym> créront souvent une feuille de style
  197. globale et des feuilles individuelles pour les sections spécifiques ou certaines
  198. pages du site puis chargeront celles-ci plus tard conditionnellement afin de réduire
  199. le nombre de données à transférer entre chaque requête. Le placeholder
  200. <methodname>headLink()</methodname> permet de réaliser de telles aggrégations
  201. conditionnelles de feuilles de style.
  202. </para>
  203. <para>
  204. Pour cela, <methodname>headLink()</methodname> definit une certain nombre de méthodes
  205. "virtuelles" (via surcharge) pour simplifier le tout. Celles qui vont nous concernet
  206. sont <methodname>appendStylesheet()</methodname> et
  207. <methodname>prependStylesheet()</methodname>. Chacune peut accepter jusqu'à quatre
  208. arguments, <varname>$href</varname> (chemin relatif vers la feuille de style),
  209. <varname>$media</varname> (le type <acronym>MIME</acronym>, par défaut "text/css"),
  210. <varname>$conditionalStylesheet</varname> (à utiliser pour préciser une "condition"
  211. à évaluer pour la feuille de style), et <varname>$extras</varname> (un tableau
  212. associatif utiliser générallement pour renseigner une clé pour "media"). Dans la
  213. plupart des cas, seul le premier argument suffira, le chemin relatif vers la feuille
  214. de style.
  215. </para>
  216. <para>
  217. Dans notre exemple, nous supposerons que toutes les pages ont besoin de charger une
  218. feuille de style stockée dans "<filename>/styles/site.css</filename>" (relativement
  219. au document root)&#160;; nous allons préciser cela dans notre méthode de bootstrap
  220. <methodname>_initPlaceholders()</methodname>.
  221. </para>
  222. <programlisting language="php"><![CDATA[
  223. class Bootstrap extends Zend_Application_Bootstrap_Bootstrap
  224. {
  225. // ...
  226. protected function _initPlaceholders()
  227. {
  228. $this->bootstrap('View');
  229. $view = $this->getResource('View');
  230. $view->doctype('XHTML1_STRICT');
  231. // Affecte le titre original et le séparateur:
  232. $view->headTitle('My Site')
  233. ->setSeparator(' :: ');
  234. // Affecte la feuille de style originale:
  235. $view->headLink()->prependStylesheet('/styles/site.css');
  236. }
  237. // ...
  238. }
  239. ]]></programlisting>
  240. <para>
  241. Plus tard, dans un contrôleur par exemple, nous pouvons rajouter des feuilles de
  242. style:
  243. </para>
  244. <programlisting language="php"><![CDATA[
  245. <?php $this->headLink()->appendStylesheet('/styles/user-list.css') ?>
  246. ]]></programlisting>
  247. <para>
  248. Dans notre layout, là encore, un simple echo sur le placeholder&#160;:
  249. </para>
  250. <programlisting language="php"><![CDATA[
  251. <?php echo $this->doctype() ?>
  252. <html>
  253. <?php echo $this->headTitle() ?>
  254. <?php echo $this->headLink() ?>
  255. <!-- ... -->
  256. ]]></programlisting>
  257. <para>
  258. Ceci génèrera quelque chose comme:
  259. </para>
  260. <programlisting language="html"><![CDATA[
  261. <link rel="stylesheet" type="text/css" href="/styles/site.css" />
  262. <link rel="stylesheet" type="text/css" href="/styles/user-list.css" />
  263. ]]></programlisting>
  264. </sect2>
  265. <sect2 id="learning.view.placeholders.standard.head-script">
  266. <title>Aggréger des scripts avec HeadScript</title>
  267. <para>
  268. Un autre moyen de ne pas surcharger la page est de ne charger le JavaScript que
  269. lorsque c'est nécessaire. Vous aurez donc besoin de scripts découpés: peut-être un
  270. pour afficher le menu du site progressivement, un autre pour traiter le contenu d'une
  271. page spécifique. Dans ces cas, l'aide <methodname>headScript()</methodname> propose
  272. une solution.
  273. </para>
  274. <para>
  275. Comme l'aide <methodname>headLink()</methodname>, <methodname>headScript()</methodname>
  276. permet d'empiler en début ou fin des scripts entiers et de les afficher d'un coup. Cela
  277. est très flexible pour spécifier des fichiers de scripts entiers à charger, ou encore
  278. du code JavaScript explicite. Vous pouvez aussi capturer le JavaScript via
  279. <methodname>captureStart()</methodname>/<methodname>captureEnd()</methodname>, qui
  280. permettent d'utiliser du code JavaScript inline plutot que de demander un appel serveur
  281. pour charger un fichier.
  282. </para>
  283. <para>
  284. Tout comme <methodname>headLink()</methodname>, <methodname>headScript()</methodname>
  285. propose des mééthodes "virtuelles" via surcharge pour spécifier rapidement des contenus
  286. à aggréger; les méthodes sont <methodname>prependFile()</methodname>,
  287. <methodname>appendFile()</methodname>, <methodname>prependScript()</methodname>, et
  288. <methodname>appendScript()</methodname>. Les deux premières vous permettent de préciser
  289. des fichiers référéncés dans l'attribut <varname>$src</varname> d'une balise
  290. <emphasis>&lt;script&gt;</emphasis>; les deux dernières vont prendre le contenu qu'on
  291. leur passe et le rendre comme du JavaScript dans les balises
  292. <emphasis>&lt;script&gt;</emphasis>.
  293. </para>
  294. <para>
  295. Dans cet exemple, nous allons spécifier qu'un script,
  296. "<filename>/js/site.js</filename>" a besoin d'être chargé sur chaque page&#160;; nous
  297. allons donc mettre à jour notre méthode de bootstap
  298. <methodname>_initPlaceholders()</methodname> pour effectuer cela.
  299. </para>
  300. <programlisting language="php"><![CDATA[
  301. class Bootstrap extends Zend_Application_Bootstrap_Bootstrap
  302. {
  303. // ...
  304. protected function _initPlaceholders()
  305. {
  306. $this->bootstrap('View');
  307. $view = $this->getResource('View');
  308. $view->doctype('XHTML1_STRICT');
  309. // Titre et séparateur d'origine:
  310. $view->headTitle('My Site')
  311. ->setSeparator(' :: ');
  312. // Feuille de style originale:
  313. $view->headLink()->prependStylesheet('/styles/site.css');
  314. // Affecte le JS initial à charger:
  315. $view->headScript()->prependFile('/js/site.js');
  316. }
  317. // ...
  318. }
  319. ]]></programlisting>
  320. <para>
  321. Dans un script de vue, nous voulons ajouter un script ou capturer du contenu
  322. JavaScript à inclure dans le document.
  323. </para>
  324. <programlisting language="php"><![CDATA[
  325. <?php $this->headScript()->appendFile('/js/user-list.js') ?>
  326. <?php $this->headScript()->captureStart() ?>
  327. site = {
  328. baseUrl: "<?php echo $this->baseUrl() ?>"
  329. };
  330. <?php $this->headScript()->captureEnd() ?>
  331. ]]></programlisting>
  332. <para>
  333. Dans notre script de layout, nous affichons simplement le placeholder, tout comme
  334. nous avions fait pour les autres précédemment&#160;:
  335. </para>
  336. <programlisting language="php"><![CDATA[
  337. <?php echo $this->doctype() ?>
  338. <html>
  339. <?php echo $this->headTitle() ?>
  340. <?php echo $this->headLink() ?>
  341. <?php echo $this->headScript() ?>
  342. <!-- ... -->
  343. ]]></programlisting>
  344. <para>
  345. Le contenu suivant sera généré:
  346. </para>
  347. <programlisting language="html"><![CDATA[
  348. <script type="text/javascript" src="/js/site.js"></script>
  349. <script type="text/javascript" src="/js/user-list.js"></script>
  350. <script type="text/javascript">
  351. site = {
  352. baseUrl: "<?php echo $this->baseUrl() ?>"
  353. };
  354. </script>
  355. ]]></programlisting>
  356. <note>
  357. <title>Variante InlineScript</title>
  358. <para>
  359. La plupart des navigateur bloquent l'affichage tant que tous les scritps et les
  360. feuilles de style référencés dans la section <emphasis>&lt;head&gt;</emphasis>
  361. ne sont pas chargés. Ces règles permettent un meilleur feeling au niveau du rendu
  362. de la page et permettent à l'utilisateur de voir le contenu de la page plus tôt.
  363. </para>
  364. <para>
  365. Pour cela, vous pouvez par exemple écrire vos tags
  366. <emphasis>&lt;script&gt;</emphasis> après avoir fermé
  367. <emphasis>&lt;body&gt;</emphasis>. (C'est une pratique recommandée
  368. par <ulink url="http://developer.yahoo.com/yslow/">Y!Slow project</ulink>.)
  369. </para>
  370. <para>
  371. Zend Framework supporte cela de deux manières différentes&#160;:
  372. </para>
  373. <itemizedlist>
  374. <listitem>
  375. <para>
  376. Vous pouvez rendre <methodname>headScript()</methodname> où vous voulez
  377. dans votre layout; ce n'est pas parce que la méthode commence par "head"
  378. que vous devez l'appeler pour cette section du HTML.
  379. </para>
  380. </listitem>
  381. <listitem>
  382. <para>
  383. Aussi, vous pourriez utiliser l'aide de vue
  384. <methodname>inlineScript()</methodname>, qui est simplement une variante
  385. de <methodname>headScript()</methodname> avec le même
  386. comportement mais un registre séparé.
  387. </para>
  388. </listitem>
  389. </itemizedlist>
  390. </note>
  391. </sect2>
  392. </sect1>