BLOCKS pragma - bobthecow/mustache.php GitHub Wiki
Make sure you're using Mustache.php v2.7.0 or higher, so that you can use the BLOCKS pragma.
{
"require": {
"mustache/mustache": "~2.7"
}
}
Install the package with composer install
or composer update
.
Let's adapt the hello world example from the repo README to use blocks.
<?php
require 'vendor/autoload.php';
$m = new Mustache_Engine([
'partials' => [
'parent' => '{{% BLOCKS }}Hello {{$ planet }}planet{{/ planet }}'
],
]);
echo $m->render('{{% BLOCKS }}{{< parent }}{{$ planet }}world!{{/ planet }}{{/ parent }}', []);
This returns:
Hello world!
If you know you are going to be using the BLOCKS pragma throughout your entire
application you can also enable globally when constructing the Mustache_Engine
:
$m = new Mustache_Engine([
'pragmas' => [Mustache_Engine::PRAGMA_BLOCKS],
'partials' => [
'parent' => 'Hello {{$ planet }}planet{{/ planet }}'
],
]);
echo $m->render('{{< parent }}{{$ planet }}World!{{/ planet }}{{/ parent }}', []);
Say we have a template named layout.mustache
. This would be our generic site
layout template, consisting of all the sorts of things you'd expect in a layout
template. Many of these values in layout.mustache
would really be dependent on
what is being rendered into it. For example each page probably specifies
additional JavaScript files that should be loaded. Beyond that, it is nice to
have a convenient way for pages to share common markup, without having to split
the common markup into {{> top }}
and {{> bottom }}
partials.
In other templating languages this behavior is achieved using a feature called template inheritance. Content templates are said "inherit" from layout templates. We have chosen to call this feature "blocks", as inheritance is an overloaded term.
Let's take a look at a content template that inherits from a layout template.
<!-- layout.mustache -->
{{% BLOCKS }}
<!DOCTYPE html>
<html>
<head>
<title>{{$ title }}My Awesome Site{{/ title }}</title>
{{$ stylesheets }}
<link rel="stylesheet" href="/assets/css/default.css">
{{/ stylesheets }}
</head>
<body>
<header>
{{$ header }}
<h1>Welcome to My Awesome Site</h1>
{{/ header }}
</header>
<div id="content">
{{$ content }}
<p>Hello, World!</p>
{{/ content }}
</div>
{{$ scripts }}
<script src="/assets/js/default.js"></script>
{{/ scripts }}
</body>
</html>
<!-- article.mustache -->
{{% BLOCKS }}
{{< layout }}
{{$ title }}{{ article.title }}{{/ title }}
{{$ stylesheets }}
<link rel="stylesheet" href="/assets/css/default.css">
<link rel="stylesheet" href="/assets/css/article.css">
{{/ stylesheets }}
{{$ content }}
<p>{{ article.body }}</p>
{{/ content }}
{{$ scripts }}
<script src="/assets/js/default.js"></script>
<script src="/assets/js/article.js"></script>
{{/ scripts }}
{{/ layout }}
Rendering article.mustache with this context:
[
'article' => [
'title' => 'How to do x, y and z',
'body' => '<p>First do x, then do y, then do z.</p>'
],
]
would result in this rendered output:
<!-- layout.mustache -->
<!DOCTYPE html>
<html>
<head>
<title>How to do x, y and z</title>
<link rel="stylesheet" href="/assets/css/default.css">
<link rel="stylesheet" href="/assets/css/article.css">
</head>
<body>
<header>
<h1>Welcome to My Awesome Site</h1>
</header>
<section id="content">
<p>First do x, then do y, then do z.</p>
</section>
<script src="/assets/js/default.js"></script>
<script src="/assets/js/article.js"></script>
</body>
</html>
Many tokens are not allowed directly inside the parent ({{< }}
) tags including sections
({{# }}
), inverted sections ({{^ }}
), partials ({{> }}
) and tokenizer
directives ({{= }}
). For example, all of the following templates will result
in compiler errors:
{{< foo }}{{# bar }}{{$ baz }}{{/ baz }}{{/ bar }}{{/ foo }}
{{< foo }}{{^ bar }}{{$ baz }}set by template{{/ baz }}{{/ bar }}{{/ foo }}
{{< foo }}{{> qux }}{{$ baz }}set by template{{/ baz }}{{/ foo }}
{{< foo }}{{=<% %>=}}<%={{ }}=%>{{/ foo }}
Extensive template inheritance tests are available here.
Ongoing discussion on the semantics of template inheritance as it relates to the mustache spec can be found at this mustache spec pull request.
Thanks to @sayrer for the template inheritance feature in hogan.js, which we stole, tests and all. Much thanks to my coworkers @dtb, @mdg and @benburry at @Etsy. And thanks to @bobthecow for a lot of feedback and help with this implementation.