An open API service indexing awesome lists of open source software.

https://github.com/wol-soft/php-micro-template

A minimalistic templating engine for PHP
https://github.com/wol-soft/php-micro-template

lightweight php php7 php8 php81 php82 php83 php84 regex template-engine templating

Last synced: 3 months ago
JSON representation

A minimalistic templating engine for PHP

Awesome Lists containing this project

README

          

[![Latest Version](https://img.shields.io/packagist/v/wol-soft/php-micro-template.svg)](https://packagist.org/packages/wol-soft/php-micro-template)
[![Maintainability](https://api.codeclimate.com/v1/badges/9e3c565c528edb3d58d5/maintainability)](https://codeclimate.com/github/wol-soft/php-micro-template/maintainability)
[![Build Status](https://github.com/wol-soft/php-micro-template/actions/workflows/main.yml/badge.svg)](https://github.com/wol-soft/php-micro-template/actions/workflows/main.yml)
[![Coverage Status](https://coveralls.io/repos/github/wol-soft/php-micro-template/badge.svg?branch=master)](https://coveralls.io/github/wol-soft/php-micro-template?branch=master)
[![MIT License](https://img.shields.io/packagist/l/wol-soft/php-micro-template.svg)](https://github.com/wol-soft/php-micro-template/blob/master/LICENSE)

# php-micro-template
A minimalistic, lightweight templating engine for PHP based on regular expressions.

## Features ##

- Replace variables inside a template
- Iterate over an array or iterable object
- Conditional sections
- Pass objects
- call functions

## Requirements ##

- Requires at least PHP 7.1

## Installation ##

The recommended way to install php-micro-template is through [Composer](http://getcomposer.org):
```
$ composer require wol-soft/php-micro-template
```

## Examples ##

First create your template file:

```html

{{ pageTitle }}


    {% foreach products as product %}
    {% if product.isVisible() %}

  • {{ productHead }}
    {{ product.getTitle() }}

  • {% endif %}
    {% endforeach %}

{% if showVersion %}

1.0.1

{% endif %}

```

Afterwards create a new instance of the Render class and render your template:

```php
renderTemplate(
'productList.template',
[
'pageTitle' => $translator->translate('availableProducts'),
'productHead' => $translator->translate('product'),
'products' => $products,
'showVersion' => true
]
);

/* ... */
```

Instead of saving your templates into files you can also prepare a string which contains the template:

```php

{{ productHead }}
{{ product.getTitle() }}

{% endif %}
{% endforeach %}
';

$render = new Render();

$result = $render->renderTemplateString(
$myPartialTemplate,
[
'productHead' => $translator->translate('product'),
'products' => $products
]
);

/* ... */
```

### Replacement of variables

Values which are assigned to the template and used directly will be casted to string.
For assigned objects you can call methods which return a value.
Afterwards the returned value will be casted to string.
As constant values integer numbers, strings in single quotes and booleans (true, false) are supported.

```html
{{ simpleValue }}
{{ myObject.getProperty() }}
{{ 'Hello World' }}
{{ 12345 }}
```

Your provided data may be a nested array which can be resolved in the template:

```php
$render->renderTemplateString(
'{{ render.productRender.renderProductName(product.details.name) }}',
[
'product' => [
'details' => [
'name' => 'MyProduct',
],
],
'render' => [
'productRender' => new ProductRender(),
]
]
);
```

Also, public properties of objects may be accessed from the template:

```php
$person = new stdClass();
$person->name = 'Hans';

$render->renderTemplateString(
'{{ person.name }}',
[
'person' => $person,
]
);
```

By default, a used variable which is not provided will result in an `UndefinedSymbolException`. You can register a callback function via `onResolveError` to handle unresolved variable errors. The callback function must implement the signature `function (string $unresolvedVariable): string`. The provided `$unresolvedVariable` will contain the whole expression which failed to resolve (eg. `myUnresolvedVariable`, `myUnresolvedObject.render(var1, var2)`).

```php
$render->onResolveError(function (string $var): string {
return 'Undefined';
});

// will result in "Person name: Undefined"
$result = $render->renderTemplateString('Person name: {{ name }}');
```

### Loops

If you assign an array or an iterable object you can use the *foreach* loop to iterate.

```html
{% foreach products as product %}
{{ product.getTitle() }}
{% endforeach %}
```

All variables of the parent scope are available inside the loop as well as the current item of the loop. Multiple foreach loops can be nested (compare tests). You can also provide a function which returns an array or an iterable object:

```html
{% foreach product.getIngredients() as ingredient %}
{{ ingredient.getTitle() }}
{% endforeach %}
```

Loops support the usage of key value pairs:

```html
{% foreach products as bestSellerNumber, product %}
Bestseller Nr. {{ bestSellerNumber }}:{{ product.getTitle() }}

{% endforeach %}
```

### Conditional sections

With the *if* statement you can create conditional sections. As a condition you can pass either a value which will be casted to bool or call a method on an object. In this case the return value of the function will be casted to bool.
Neither multiple values in a single condition combined by operators nor calculations or similar additional functions are provided. For advanced conditions compare the section *function calls* with a ViewHelper-Object.

```html
{% if showProducts %}
{% if product.isVisible() %}
{{ product.getTitle() }}
{% else %}
Product {{ product.getTitle() }} currently not available
{% endif %}
{% endif %}
```

Multiple if statements can be nested. To invert an if condition the keyword *not* can be used:

```html
{% if not product.isVisible() %}
Product {{ product.getTitle() }} currently not available
{% endif %}
```

### function calls

The methods which are called can take parameters.
Allowed parameters are variables taken out of the current scope or another function call on an object available in the current scope as well as the supported constant values integer numbers, strings in single quotes and booleans (true, false).
As an example a ViewHelper-Object can be assigned to the render process and methods of the ViewHelper can be used in the template for advanced logic inside the template.

```php
%s', $weight, $label);
}
}

/* ... */

$render = new Render(__DIR__ . '/Templates/');

$result = $render->renderTemplate(
'functionExample.template',
[
'viewHelper' => new ViewHelper(),
'currencyFormatter' => new CurrencyFormatter(),
'basePrice' => 3.00,
'products' => $products
]
);

/* ... */

```

```html

Products: {{ viewHelper.count(products) }}


    {% foreach products as product %}

  • {{ viewHelper.weightFont(product.getTitle(), 600) }}
    Price: {{
    currencyFormatter.format(
    viewHelper.sum(
    product.getPrice(),
    basePrice
    )
    )
    }}


  • {% endforeach %}

```

Additionally, PHP global functions can be used directly in the template as well as assigned callback methods:

```php
renderTemplate(
'functionExample.template',
[
'customCallback' => function(string $in): string {
return trim(strtoupper($in));
},
]
);

/* ... */

```

```html

{{ customCallback('products') }}


{{ strtolower('UNDER CONSTRUCTION') }}

```

### Whitespace tolerance

The templating syntax is whitespace tolerant so a template like the one below would be perfectly fine:

```html
{%if
product.getCategories()
%}

Categories:



    {%foreach
    product.getCategories()
    as
    category
    %}
  • {{product.getTitle()} [{{ category }}]

  • {%endforeach%}

{%endif%}
```