Legacy templates

This article covers legacy PHP templates. For modern templates, follow the Twig documentation .

A template is mainly composed of HTML and PHP code. It is used to structure a piece of content of a module or a content element, etc. For example, the template news_full.html5 displays the full content of a news item whereas the template news_short.html5 displays only a portion of this content.

The templates are located in their own modules. For example, news_full.html5 is located under vendor/contao/news-bundle/src/Resources/contao/templates/news.

A template can be a part of the structure of a module, a content element, a form, etc. and that is why they are prefixed. They can be easily grouped, ordered and recognized. For example: the prefix ce_ means “Content Element” and mod_ means “Module”.

Template Instantiation and Rendering

Contao provides two template classes: Contao\FrontendTemplate and Contao\BackendTemplate, which both inherit from the abstract Contao\Template class and use the Contao\TemplateInheritance trait. The FrontendTemplate class should be used whenever you want to render a template in the front end, while the BackendTemplate can be used to render a template in the back end. All ce_*, mod_*, news and event templates etc. are always front end templates, whereas most back end templates start with the prefix be_.

The following example will instantiate a front end template object and render the template as a string:

use Contao\FrontendTemplate;

$template = new FrontendTemplate('my_front_end_template');
$template->someData = 'foobar';
$buffer = $template->parse();

The instantiation references a template named my_front_end_template. This will refer to a template file named my_front_end_template.html5 which will then be searched for by Contao in different locations.

Template Folders

When a template instance is created, Contao will search for the appropriate template in the following locations and the following order:

Folder Description
templates/<THEME>/ A subfolder within templates/ as defined in the front end theme. This folder will contain overridden and extended templates that will only apply for page layouts of a specific theme in the front end.
templates/ This folder can contain template overrides and extensions, as well as custom templates.
contao/templates/ Here you can organize your application specific templates, e.g. the templates for custom content elements or front end modules etc. of your application. These templates can then still be overridden and extended via the previous folders.
<BUNDLE>/contao/templates/ Each Contao extension (i.e. contao-bundle) can provide templates for their own content elements or front end modules etc. These templates can then be overridden and extended via templates in the templates/ directory (including the theme’s subdirectory).

Template Groups

In order for a new template to show up in the template selection in the back end for a certain content element or front end module, the template’s name must be prefixed with either ce_ or mod_ and then its type plus another underscore: ce_<element-type>_ or mod_<module-type>_. For example, if you want to create a new template for the Text content element, whose defined type in the DCA is text, then the template name needs to be ce_text_ipsum.html5. Or if you want to create an additional template for the HTML module, whose defined type in the DCA is html, then the template name must be mod_html_foobar.html5.

The same applies to form field templates of the form generator. For a new Textarea template, the template’s name must be form_textarea_custom.html5. For sub-item templates, like news templates, navigation item templates or event templates, there will only be one required prefix, i.e. news_, nav_ and event_ respectively.

Template Inheritance

The inheritance allow you to create a template based on a second template. This means that a template (child) inherits the content of a second template (parent).

In order that the content of a parent template may be modified or completed in the child template, it must be surrounded by an element named block.

A block is built as follows:

<?php $this->block('name_of_the_block'); ?>

  // Block content

<?php $this->endblock(); ?>

The example below shows a parent template with a block surrounding the content of the head tag.

Template fe_page.html5:

<!DOCTYPE html>
<html>
<head>
    <?php $this->block('head'); ?>
        <title><?= $this->title; ?></title>
        <link rel="stylesheet" href="style.css">
    <?php $this->endblock(); ?>
</head>
<body>
    […]
</body>
</html>

In the child template fe_custom.html5, a style sheet is added in the head tag in addition to the inherited content of the parent template fe_page.html5.

Template fe_custom.html5:

<?php $this->extend('fe_page'); ?>

<?php $this->block('head'); ?>
    <?php $this->parent(); ?>
  
    <link rel="stylesheet" href="style_2.css">
<?php $this->endblock(); ?>
  • The extend() function specifies the template name whose it inherits the content.
  • The parent() function allows to complete a block without replacing the inherited content.

The output of the fe_custom.html5 template will be:

<!DOCTYPE html>
<html>
<head>
    <title>A title</title>
    <link rel="stylesheet" href="style.css">
    <link rel="stylesheet" href="style_2.css">
</head>
<body>
    […]
</body>
</html>

Contao also provides default base templates for content elements and modules, called block_searchable and block_unsearchable. These templates provide a basic wrapper for your element or module, including attributes for CSS classes and IDs as well as other attributes, if defined. It also automatically displays a headline and provides an inheritance block for the headline section and the content section. This is the content of the block_searchable template for example:

<div class="<?= $this->class ?> block"<?= $this->cssID ?><?php if ($this->style): ?> style="<?= $this->style ?>"<?php endif; ?>>

  <?php $this->block('headline'); ?>
    <?php if ($this->headline): ?>
      <<?= $this->hl ?>><?= $this->headline ?></<?= $this->hl ?>>
    <?php endif; ?>
  <?php $this->endblock(); ?>

  <?php $this->block('content'); ?>
  <?php $this->endblock(); ?>

</div>

The content of the block_unsearchable template is the same, but it is additionally wrapped with <!-- indexer::stop --> and <!-- indexer::continue -->, meaning its contents will not be indexed by Contao’s search indexer.

Most of the templates for Contao’s content elements and front end modules extend from these templates. This is the content of the newslist module’s template mod_newslist for example:

<?php $this->extend('block_unsearchable'); ?>

<?php $this->block('content'); ?>

  <?php if (empty($this->articles)): ?>
    <p class="empty"><?= $this->empty ?></p>
  <?php else: ?>
    <?= implode('', $this->articles) ?>
    <?= $this->pagination ?>
  <?php endif; ?>

<?php $this->endblock(); ?>

It extends from block_unsearchable, as the news article teasers should not be indexed by Contao’s search indexer by default (only the detail pages) and only overrides the parent’s content block, where the news articles (and pagination) will be shown.

It is also possible to override parent blocks with empty content. For example, if you want to show a text content element’s headline not in its default location, but somewhere else entirely, you could extend that template, override the headline block with empty content and instead show the headline in a custom location within the content block.

<?php $this->extends('ce_text'); ?>

<?php $this->block('headline'); ?>
<?php $this->endblock(); ?>

<?php $this->block('content'); ?>

  <?php if ($this->addImage): ?>
    <?php $this->insert('image', $this->arrData); ?>
  <?php endif; ?>

  <div class="my-text-wrapper">
    <?php if ($this->headline): ?>
      <<?= $this->hl ?>><?= $this->headline ?></<?= $this->hl ?>>
    <?php endif; ?>
    <?= $this->text ?>
  </div>

<?php $this->endblock(); ?>

Any template within the templates/ folder can override and extend existing templates from the contao/templates/ folder or from any Contao extension. The templates/ folder also allows you to override and extend from the same template at the same time. For instance, you can create a templates/fe_page.html5 and still use $this->extend('fe_page') as this will extend from the Contao Core Bundle’s fe_page. However, you cannot do the same within Contao extensions. You can only extend from other templates there using a different template name. The same applies to the contao/templates/ folder as this can be considered your application’s extension.

Template Insertion

A template can be inserted into another template thanks to the insert() function.

<?php $this->insert('template_name'); ?>

The insert() function also accepts the assignment of variables as second parameter.

<?php $this->insert('template_name', ['key' => 'value']); ?>

// This passes all variables from the current template
<?php $this->insert('template_name', $this->getData()); ?>

In the example below, we would like to insert the template image-copyright.html5 in the template image.html5.

The template image.html5 contains an img tag and the insert() function.

Template image.html5:

<img src="<?= $this->src; ?>" alt="<?= $this->alt; ?>" />
<?php $this->insert('image-copyright', ['name' => 'Donna Evans', 'license' => 'Creative Commons']); ?>

The template image-copyright.html5 contains a small tag that will be inserted below the img tag in the template image.html5. The variables name and license will be replaced with the values determined in the insert() function.

Template image-copyright.html5:

<small>Photograph by <?= $this->name; ?>, licensed under <?= $this->license; ?></small>

The output of the image.html5 template will be:

<img src="files/images/house.jpg" alt="A small house in England" />
<small>Photograph by Donna Evans, licensed under Creative Commons</small>

Template Data

The available template data varies depending on the source of the template. Typically, content element templates or element templates of modules will have their complete data record available in the template. For example, in any content element template, the complete data set of the database record of the element will be available in the template via $this->…. The content of some fields might have been changed by the content element or module controller though.

To set template data you can use one of two ways. On the one hand the Template class implements the __set method so that you can directly set a template variable as if it was an object variable:

use Contao\FrontendTemplate;

$template = new FrontendTemplate('my_template');
$template->foobar = 'foobar';

This variable can then be accessed from within the template:

<!-- contao/templates/my_template.html5 -->
<?= $this->foobar ?>

You can also set and override the template data via an associative array:

$template->setData([
    'myVariable' => 'foobar',
    'myOtherVariable' => 'Lorem Ipsum',
]);

You can also assign a callable to a template variable:

$template->getHash = function(string $string): string {
    return substr(md5($string), 0, 8);
};

This function can then be called from within the template, as Contao’s Template class implements __call and checks whether a template variable of this name is available as a callable:

<?= $this->getHash('foobar') ?>

You can also inspect the available template data by using either

<?php $this->dumpTemplateVars() ?>

or

<?php dump($this) ?>

within the template. Both of these statements will use the Symfony VarDumper component to display the template’s data.

If your template uses template inheritance, the template variable dump will only be visible either in debug mode or if the dump is in between the $this->block(…) and $this->endblock() statements.