PHP Views is a Composer package designed to simplify templating in PHP. It introduces a model-driven approach, supports multiple namespaces, and includes a custom Blade implementation as the default template engine.
1. Introduction
2. Installation and minimal usage
3. Model-driven approach
4. Views Manager
5. View Renderer
6. Contribution
7. Credits
While many PHP frameworks come with their own solutions for views, there are countless PHP projects without a templating system (including CMS, like WordPress). Writing pure PHP templates can be cumbersome and error-prone. PHP Views aims to make templating simple, flexible, and accessible to anyone.
This package includes a custom Blade compiler and uses it as the default template engine. However, flexibility is key. You can integrate Twig, or any other template engine, with just a few lines of code while still enjoying the benefits of the model-driven approach and other features that PHP Views brings to the table.
- Blazing fast: Outperforms the original Laravel Blade (see the benchmark below).
- Zero Dependencies: Lightweight and easy to integrate into any project.
- Wide Compatibility: PHP 7.4+, 8.0+
- SOLID architecture: Designed with flexibility in mind, allowing you to easily override modules to meet your specific requirements
- Namespace Support: Seamlessly manage multiple templates under a unified, structured approach.
- Reliable: Thoroughly tested with Pest and checked by PHPStan.
The strength of this package lies in its flexibility. With PHP Views, you’re free to use it in a way that best fits your project:
- As a Views provider: Combine a model-driven approach with the built-in Blade engine for clean, dynamic templates.
- As a standalone Blade engine: Use its custom Blade implementation for your Blade-based templates.
- For Model-Driven Flexibility: Leverage the model-driven approach with any template engine, such as Twig or even pure PHP.
- As a Template Connector: Integrate and unify templates that utilize different engines under one system.
We conducted a PHP performance benchmark to compare this package with the Laravel's Blade (mocked using jenssegers/blade) and Twig. Here are the results for 1000x renders:
Contestant | First Rendering, MS | Cached Rendering, MS |
---|---|---|
prosopo/views (without models) |
19.75 | 19.75 (no cache atm) |
prosopo/views (with models) |
43.78 | 43.78 (no cache atm) |
illuminate/view (Blade from Laravel) |
181.24 | 56.77 ms |
twig/twig |
441.13 | 9.47 ms |
The numbers speak for themselves. In uncached rendering, even with model class-related overhead, PHP Views’ implementation significantly outperforms all competitors. While Twig offers a robust caching solution, PHP Views still delivers better performance even than Laravel's Blade engine with caching.
We used the following package versions:
- illuminate/view
11.7.0
- twig/twig
3.17.1
- jenssegers/blade
2.0.1
Since the benchmark is included in this repository, you can easily run it locally to verify the results.
git clone https://github.com/prosopo/php-views.git
composer install; cd benchmark; composer install
php benchmark {1000}
- pass your renders count
PHP Views is distributed as a Composer package, making installation straightforward:
composer require prosopo/views
After installation, ensure that your application includes the Composer autoloader (if it hasn’t been included already):
require __DIR__ . '/vendor/autoload.php';
To get started, you’ll need to create three instances: ViewTemplateRenderer
, ViewNamespaceConfig
, and
ViewsManager
.
The main configuration takes place in ViewNamespaceConfig
, where you define the folder for your templates and the root
namespace for the associated models.
use Prosopo\Views\View\ViewNamespaceConfig;
use Prosopo\Views\View\ViewTemplateRenderer;
use Prosopo\Views\ViewsManager;
require __DIR__ . '/vendor/autoload.php';
// 1. Make the Template Renderer.
// (By default it uses the built-in Blade, but you can connect any)
$viewTemplateRenderer = new ViewTemplateRenderer();
// 2. Make the namespace config
$namespaceConfig = (new ViewNamespaceConfig($viewTemplateRenderer))
->setTemplatesRootPath(__DIR__ . './templates')
->setTemplateFileExtension('.blade.php');
// 3. Make the Views Manager instance:
$viewsManager = new ViewsManager();
// 4. Add the root namespace of your Template Models
$viewsManager->registerNamespace('MyPackage\Views', $namespaceConfig);
Now you're ready to create your first model. Similar to many frameworks, such as Laravel, this package embraces a model-driven approach to templates.
Each template is paired with its own Model, where the Model's public properties and methods act as arguments available within the template.
Model class must extend the BaseTemplateModel
class or implement the TemplateModelInterface
:
namespace MyPackage\Views;
use Prosopo\Views\BaseTemplateModel;
class EmployeeTemplateModel extends BaseTemplateModel
{
public int $salary;
public int $bonus;
public CompanyTemplateModel $company;
public function total(): int
{
return $this->salary + $this->bonus;
}
}
Model template (Blade is used in this example):
<p>
Your month income is {{ $total() }},
from which {{ $salary }} is a salary, and {{ $bonus }} is a bonus.
Est. taxes: {{ $company->calcTaxes($salary) }}
</p>
<p>Company info:</p>
{!! $company !!}
As you can see, all the public properties of the model are accessible within the template, including nested models like
$company
. This also enables you to call their public methods directly within the template.
The BaseTemplateModel
class which we inherited overrides the __toString()
method, allowing inner models to be
rendered as strings using Blade echo statements with the HTML support. For
instance:
{!! $innerModel !!}
part of the template will render the model and print the result.
Naming clarification: this package
does not require
theModel suffix
in the names of model classes. In this document, we use the Model suffix for class names purely for demonstration purposes.
The built-in ModelTemplateResolver
automatically matches templates based on the Model names and their relative
namespaces. This automates the process of associating templates with their corresponding Models.
Example:
- src/
- Views/
Homepage.php
- Settings
GeneralSettings.php
- templates/
homepage{.blade.php}
- settings/
general-settings{.blade.php}
- Views/
We define the root templates folder along with the models root namespace in the ViewsNamespaceConfig
(as we already
did above):
$namespaceConfig = (new ViewNamespaceConfig($viewTemplateRenderer))
->setTemplatesRootPath(__DIR__ . './templates')
->setTemplateFileExtension('.blade.php');
// 3. Make the Views Manager instance:
$viewsManager = new ViewsManager();
// 4. Add the root namespace of your Template Models
$viewsManager->registerNamespace('MyPackage\Views', $namespaceConfig);
Naming Note: Use dashes in template names, as camelCase
in Model names is automatically converted to
dash-separated
names.
The ViewsManager
instance (which we created during the setup) provides the createModel
and renderModel
methods.
You can create, set values, and render a Model in a single step using the callback argument of the renderView
method,
as shown below:
echo $viewsManager->renderModel(
EmployeeTemplateModel::class,
function (EmployeeTemplateModel $employee) use ($salary, $bonus) {
$employee->salary = $salary;
$employee->bonus = $bonus;
}
);
This approach enables a functional programming style when working with Models.
Multi-step creation and rendering
When you need split creation, call the makeModel
method to create the model, and then render later when you need it.
$employee = $viewsManager->createModel(EmployeeTemplateModel::class);
// ...
$employee->salary = $salary;
$employee->bonus = $bonus;
// ...
echo $views->renderModel($employee);
// Tip: you can pass the callback as the second argument for both createModel() and renderModel() models
// to customize the Model properties before returning/rendering.
References Advice
The ViewsManager
class implements three interfaces: ViewNamespaceManagerInterface
for registerNamespace
,
ModelFactoryInterface
for createModel
, and ModelRendererInterface
for renderModel
.
When passing the ViewsManager
instance to your methods, use one of these interfaces as the argument type instead of
the ViewsManager
class itself.
This approach ensures that only the specific actions you expect are accessible, promoting cleaner and more maintainable code.
That's it! You’re now ready to start using the package. In the sections below, we'll dive deeper into its configuration and implementation details.
This package embraces a model-driven approach to templates. Each template is paired with its own Model, where the Model's public properties and methods act as arguments available within the template.
Model class must extend the BaseTemplateModel
class or implement the TemplateModelInterface
:
namespace MyPackage\Views;
use Prosopo\Views\BaseTemplateModel;
class EmployeeTemplateModel extends BaseTemplateModel
{
public int $salary;
public int $bonus;
}
Public model properties intended for use in the template must have a defined type. By default, any public properties without a type will be ignored.
Naming clarification: This package
does not require
theModel suffix
in the names of model classes. In this document, we use the Model suffix for class names purely for demonstration purposes.
- Typed variables: Eliminate the hassle of type matching and renaming associated with array-driven variables.
- Reduced Routine: During object creation, public fields of the model without default values are automatically initialized with default values.
- Enhanced Access: Public methods are made available to the template alongside the variables.
- Unified Interface: Use the
TemplateModelInterface
in your application when accepting or returning a Model to maintain flexibility and avoid specifying the exact component.
Model can contain one or more nested models or even an array of models. To allow flexibility in deciding the model
type later, define its type as TemplateModelInterface
.
This approach enables you to assign any model to this field during data loading. However, if you want to restrict the field to a specific model, you can define its exact class type:
namespace MyPackage\Views;
use Prosopo\Views\BaseTemplateModel;
use Prosopo\Views\Interfaces\Model\TemplateModelInterface;
class EmployeeTemplateModel extends BaseTemplateModel
{
// Use the abstract interface to accept any Model.
public TemplateModelInterface $innerModel;
// Use a specific class only when you want to restrict the usage to it.
public CompanyTemplateModel $company;
}
By default, inner models are passed to templates as objects, enabling you to call their public methods directly. For example:
{{ $$innerModel->calc($localVar) }}
.
The BaseTemplateModel
class overrides the __toString()
method, allowing inner models to be rendered as strings using
echo statements, which supports HTML output. For instance:
{!! $innerModel !!}
will render the model and print the result.
If you prefer, you can configure the ViewsManager
to pass models as strings instead of objects. Refer to the
ViewsManager section for details.
Note: In the TemplateModel
class, in order to satisfy the Model factory, the constructor is marked as final. If you
need to
set custom default values, consider using one of the following approaches:
namespace MyPackage\Views;
use Prosopo\Views\BaseTemplateModel;
class EmployeeTemplateModel extends BaseTemplateModel
{
// approach for plain field types.
public int $varWithCustomDefaultValue = 'custom default value';
public Company $company;
protected function setCustomDefaults(): void
{
// approach for object field types.
$this->company = new Company();
}
}
Tip: If your app Models require additional object dependencies, you can override the
PropertyValueProvider
module to integrate with a Dependency Injection container like PHP-DI. This allows model properties to be automatically resolved while object creation by your application's DI system.
The only requirement for a Model is to implement the TemplateModelInterface
. This means you can transform any class
into a Model without needing to extend a specific base class, or even define public properties:
namespace MyPackage\Views;
use Prosopo\Views\Interfaces\Model\TemplateModelInterface;
class AnyClass implements TemplateModelInterface
{
public function getTemplateArguments(): array
{
// you can fill out arguments from any source or define manually.
return [
'name' => 'value',
];
}
}
Note: If you plan to define inner models, remember that the default __toString()
implementation is not provided. You
will need to either implement it yourself or enable the option to pass models as strings in the ViewsManager
configuration:
$namespaceConfig->setModelsAsStringsInTemplates(true);
When this option is enabled, the renderer will automatically convert objects implementing TemplateModelInterface
into
strings before passing them to the template.
The ViewsManager
class provides the registerNamespace
, createModel
and renderModel
methods. It acts as a
namespace manager and brings together different namespace configurations.
Each ViewNamespace
has its own independent setup and set of modules. E.g. among these modules is the
ModelTemplateProvider
, which automates the process of linking models to their
corresponding templates.
use Prosopo\Views\View\ViewNamespaceConfig;
use Prosopo\Views\View\ViewTemplateRenderer;
use Prosopo\Views\ViewsManager;
// 1. Make the Template Renderer.
// (By default it uses the built-in Blade, but you can connect any)
$viewTemplateRenderer = new ViewTemplateRenderer();
// 2. Make the namespace config
$namespaceConfig = (new ViewNamespaceConfig($viewTemplateRenderer))
// required settings:
->setTemplatesRootPath(__DIR__ . './templates')
->setTemplateFileExtension('.blade.php')
// optional setting:
->setTemplateErrorHandler(function (array $eventDetails) {
// logging, notifying, whatever.
})
// this option enables inner models rendering before passing them into the template
->setModelsAsStringsInTemplates(true);
// (This line is necessary only if you defined the templateErrorHandler)
$namespaceConfig->getModules()
->setEventDispatcher($viewTemplateRenderer->getModules()->getEventDispatcher());
// 3. Make the Views instance:
$viewsManager = new ViewsManager();
// 4. Add the root namespace of your Template Models
$viewsManager->registerNamespace('MyPackage\Views', $namespaceConfig);
// Tip: you can have multiple namespaces, and mix their Models.
You can create, set values, and render a Model in a single step using the callback argument of the renderView
method,
as shown below:
echo $viewsManager->renderModel(
EmployeeModel::class,
function (EmployeeModel $employee) use ($salary, $bonus) {
$employee->salary = $salary;
$employee->bonus = $bonus;
}
);
This approach enables a functional programming style when working with Models.
When you need split creation, use the factory to create the model, and then render later when you need it.
$employee = $viewsManager->createModel(EmployeeModel::class);
// ...
$employee->salary = $salary;
$employee->bonus = $bonus;
// ...
echo $views->renderModel($employee);
// Tip: you can still pass the callback as the second renderModel() argument
// to customize the Model properties before rendering.
Advice: The ViewsManager
class implements three interfaces: ViewNamespaceManagerInterface
(for registerNamespace
),
ModelFactoryInterface
(for createModel
), and ModelRendererInterface
(for renderModel
).
When passing the ViewsManager
instance to your methods, use one of these interfaces as the argument type instead of
the ViewsManager
class itself.
This approach ensures that only the specific actions you expect are accessible, promoting cleaner and more maintainable code.
The built-in ModelTemplateResolver
automatically matches templates based on the Model names and their relative
namespaces. This automates the process of associating templates with their corresponding Models.
Example:
- src/
- Views/
Homepage.php
- Settings
GeneralSettings.php
- templates/
homepage{.blade.php}
- settings/
general-settings{.blade.php}
- Views/
We define the root templates folder along with the models root namespace in the ViewsNamespaceConfig
:
$namespaceConfig = (new ViewNamespaceConfig($viewTemplateRenderer))
->setTemplatesRootPath(__DIR__ . './templates')
->setTemplateFileExtension('.blade.php');
// 3. Make the Views Manager instance:
$viewsManager = new ViewsManager();
// 4. Add the root namespace of your Template Models
$viewsManager->registerNamespace('MyPackage\Views', $namespaceConfig);
Naming Note: Use dashes in template names, as camelCase
in Model names is automatically converted to
dash-separated
names.
Tip: In case this approach doesn't work for your setup, you can override the
ModelTemplateResolver
module to implement your own logic. In case the reason is the name-specific only, consider overriding theModelNameResolver
module instead.
By default, the registerNamespace
class creates module instances for the namespace using classes from the current
package.
If you need to override the default module behavior, you can define a custom implementation in the configuration and the package will use the specified implementation.
Tip: You can see the full list of the modules in the
ViewNamespaceModules
class.
// 1. Make a facade (for Twig or another template engine)
use Prosopo\Views\Interfaces\Template\TemplateRendererInterface;
use Prosopo\Views\View\ViewNamespaceConfig;
use Prosopo\Views\ViewsManager;
class TwigDecorator implements TemplateRendererInterface
{
private $twig;
public function __construct()
{
// todo init Twig or another engine.
}
public function renderTemplate(string $template, array $variables = []): string
{
return $this->twig->render($template, $variables);
}
}
// 2. Define the namespace config with the facade instance
$twigDecorator = new TwigDecorator();
$namespaceConfig = (new ViewNamespaceConfig($twigDecorator))
->setTemplatesRootPath(__DIR__ . './templates')
->setTemplateFileExtension('.twig');
// 3. Make the Views:
$viewsManager = new ViewsManager();
// 4. Add the namespace (you can have multiple namespaces)
$viewsManager->registerNamespace('MyPackage\Views', $namespaceConfig);
// ...
$viewsManager->renderModel(MyTwigModel::class);
You can override any namespace module in the following way:
$namespaceConfig->getModules()
// override any available module, like TemplateRenderer or Factory:
->setModelFactory(new MyFactory());
Note: The package includes only the Blade implementation. If you wish to use a different template engine, like Twig, you need to install its Composer package and create a facade object, as demonstrated above.
The ViewsManager
class not only supporting multiple namespaces, but also enabling you to use Models from
one namespace within another, preserving their individual setup.
Example of multi-namespace usage:
Suppose you have registered a namespace for Twig templates:
$viewsManager->registerNamespace('App\Twig',$configForTwigNamespace);
This namespace includes a Button
model and a button.twig
template:
Button's model:
namespace App\Twig;
use Prosopo\Views\BaseTemplateModel;
class ButtonModel extends BaseTemplateModel {
public string $label;
}
Button's template:
<button>{{ label|trim }}</button>
Additionally, you registered a namespace for Blade templates:
$viewsManager->registerNamespace('App\Blade',$configForBladeNamespace);
with a Popup
model:
Popup's model:
namespace App\Blade;
use Prosopo\Views\BaseTemplateModel;
class PopupModel extends BaseTemplateModel {
}
Now is the cool part: you can safely use the App\Twig\ButtonModel
class as a property of the
App\Blade\PopupModel
class, so it looks like this:
Popup's model:
namespace App\Blade;
use Prosopo\Views\BaseTemplateModel;
use App\Twig\ButtonModel;
class PopupModel extends BaseTemplateModel {
public ButtonModel $buttonModel;
}
Now you can use the buttonModel
in the popup's template:
<div>
I'm a popup!
To close, click here - {!! $buttonModel !!}
</div>
When you call ViewsManager->makeModel()
for the PopupModel
class:
- The
App\Blade\PopupModel
instance will be created using theModelFactory
from theApp\Blade
namespace. - During automated property initialization, an instance of
App\Twig\ButtonModel
will be created using theModelFactory
from for theApp\Twig
namespace.
This design allows you to seamlessly reuse models across different namespaces while respecting the specific configuration of each namespace.
Namespace resolution also occurs when you call ViewsManager->renderModel()
. In this example:
App\Twig\ButtonModel
will be rendered using theViewTemplateRenderer
from theApp\Twig
namespace, which is configured for Twig.App\Blade\PopupModel
will be rendered using theViewTemplateRenderer
from theApp\Blade
namespace, which is configured for Blade.
ViewTemplateRenderer
is the class responsible for rendering templates in this package. By default, it integrates the
Blade compiler, but it is fully customizable. You can replace the Blade compiler with your own implementation or use a
simple stub to enable support for plain PHP template files.
Blade is an elegant and powerful template engine originally developed for Laravel. Unlike Twig, Blade embraces PHP usage rather than restricting it. It enhances templates with syntax sugar (which we all love), making them clean and easy to read.
Blade introduces special shorthand tokens that simplify the most cumbersome syntax constructions, while still being fully-fledged PHP with access to all its functions and capabilities.
Unfortunately, Blade isn't available as a standalone package, so this package includes its own Blade compiler. It provides full support for Blade's key features while remaining completely independent of Laravel.
The following Blade tokens are supported:
- Displaying:
{{ $var }}
and{!! $var }}
- IF Conditions:
@if
,@else
,@elseif
,@endif
- Switch conditions:
@switch
,@case
,@break
,@default
,@endswitch
. - Loops:
@foreach
,@endforeach
,@for
,@endfor
,@break
. - Helpers:
@selected
,@checked
,@class
. - PHP-related:
@use
,@php
,@endphp
.
Visit the official Blade docs to learn about their usage.
You may have come across packages that attempt to adapt the official Blade engine by creating stubs for its Laravel dependencies, such as the jenssegers/blade package. However, we chose not to adopt this approach for several reasons:
- PHP Version Requirements: It mandates PHP 8.2 or higher.
- External Dependencies: It introduces additional external dependencies.
- Potential Breakage: It can become unstable with future Laravel updates (as demonstrated by past incidents.
- Limited Flexibility: Since it wasn’t designed as a standalone component, it lacks some of the customization abilities.
- Global functions: Laravel's implementation includes global helper functions, which becomes a problem when you need to scope the package.
Thanks to great Blade's conceptual design, our compiler implementation required fewer than 200 lines of code.
use Prosopo\Views\View\ViewTemplateRenderer;
$viewTemplateRenderer = new ViewTemplateRenderer();
echo $viewTemplateRenderer->renderTemplate('/my-template.blade.php', [
'var' => true
]);
Tip #1: by default,
BladeTemplateRenderer
is configured to work with files, but you can also switch it to work with plain strings:
use Prosopo\Views\View\ViewTemplateRenderer;
use Prosopo\Views\View\ViewTemplateRendererConfig;
$viewRendererConfig = new ViewTemplateRendererConfig();
$viewRendererConfig->setFileBasedTemplates(false);
$viewTemplateRenderer = new ViewTemplateRenderer($viewRendererConfig);
echo $viewTemplateRenderer->renderTemplate('@if($var)The variable is set.@endif', [
'var' => true
]);
Tip #2: As you see, the built-in TemplateRenderer implementation is fully standalone and independent of the
Views
class. This means that even if you can't or don't want to use the model-driven approach, you can still utilize it as an independent Blade compiler.
The ViewTemplateRenderer
supports a variety of settings that let you customize features such as
escaping, error handling, and more:
use Prosopo\Views\View\ViewTemplateRenderer;
use Prosopo\Views\View\ViewTemplateRendererConfig;
$viewRendererConfig = (new ViewTemplateRendererConfig())
// By default, the Renderer expect a file name.
// Set to false if to work with strings
->setFileBasedTemplates(true)
->setTemplateErrorHandler(function (array $eventDetails): void {
// Can be used for logging, notifying, etc.
})
->setCustomOutputEscapeCallback(function ($variable): string {
if (
false === is_string($variable) &&
false === is_numeric($variable)
) {
return '';
}
// htmlspecialchars is the default one.
return htmlentities((string)$variable, ENT_QUOTES, 'UTF-8', false);
})
->setGlobalVariables([
'sum' => function (int $a, int $b): string {
return (string)($a + $b);
},
'variable' => 'value',
])
->setEscapeVariableName('escape')
->setCompilerExtensionCallback(function (string $template): string {
// note: just an example, @use is supported by default.
return (string)preg_replace('/@use\s*\((["\'])(.*?)\1\)/s', '<?php use $2; ?>', $template);
});
$viewTemplateRenderer = new ViewTemplateRenderer($viewRendererConfig);
By default, the ViewTemplateRenderer
creates module instances using classes from the current package, including the
Blade compiler.
If you need to override the default module behavior, you can define a custom implementation in the
configuration. The ViewTemplateRenderer
will use the specified implementation.
Tip: You can see the full list of the modules in the
ViewTemplateRendererModules
.
use Prosopo\Views\Interfaces\Template\TemplateCompilerInterface;
use Prosopo\Views\View\ViewNamespaceConfig;
use Prosopo\Views\View\ViewTemplateRenderer;
use Prosopo\Views\View\ViewTemplateRendererConfig;
use Prosopo\Views\ViewsManager;
class CompilerStubForPlainPhpSupport implements TemplateCompilerInterface
{
public function compileTemplate(string $template): string
{
return $template;
}
}
// ...
$viewTemplateRendererConfig = new ViewTemplateRendererConfig();
$viewTemplateRendererConfig->getModules()
->setTemplateCompiler(new CompilerStubForPlainPhpSupport());
$viewTemplateRenderer = new ViewTemplateRenderer($viewTemplateRendererConfig);
$views = new ViewsManager();
$viewNamespaceConfig = new ViewNamespaceConfig($viewTemplateRenderer);
$viewNamespaceConfig
->setTemplatesRootPath(__DIR__ . './templates')
->setTemplateFileExtension('.php');
$views->registerNamespace('MyApp\Models', $viewNamespaceConfig);
Now this namespace is configured to deal with plain PHP template files, while having all the package features, including model-driven approach and template error handling.
We would be excited if you decide to contribute 🤝
Please read the for-devs.md file for project guidelines and agreements.
This package was created by Maxim Akimov during the development of the WordPress integration for Prosopo Procaptcha.
Procaptcha is a privacy-friendly and cost-effective alternative to Google reCaptcha.
Consider using the Procaptcha service to protect your privacy and support the Prosopo team.