hxgf / slime-render
PHP abstraction functions to help more easily render views for Slim Framework with Handlebars (LightnCandy).
README
PHP abstraction functions to help more easily render views for Slim Framework (v4) with plain text, HTML, JSON, and Handlebars (using LightnCandy)
These functions aim to provide a simplified and standardized interface for rendering various types of data-driven responses as PSR-7 objects for use with Slim.
Included with the Slime boilerplate for Slim applications.
Installation
Easy install with composer:
composer require jyoungblood/slime-render
use Slime\render; require __DIR__ . '/vendor/autoload.php';
Requirements
- Slim Framework 4
- LightnCandy >= 1.2.6
- PHP >= 7.4
Usage
render::html($request, $response, $string, $status = 200)
Renders a string as HTML. Returns a standard Slim (PSR-7) response object with optional HTTP status code (200 by default).
$app->get('/', function ($req, $res, $args) { return render::html($req, $res, '<h2>Hey whats up</h2>'); });
Additionally, a path to an HTML file can be specified to load and render instead of a string:
$app->get('/', function ($req, $res, $args) { return render::html($req, $res, '/hey/whats-up.html'); });
render::text($request, $response, $string, $status = 200)
Renders a string as plain text. Returns a standard Slim (PSR-7) response object with optional HTTP status code (200 by default).
$app->get('/', function ($req, $res, $args) { return render::text($req, $res, 'Hey whats up'); });
render::hbs($request, $response, $parameters, $status = 200)
Renders a specific Handlebars template with an array of data, including any partials and global locals
variables array. Returns a standard Slim (PSR-7) response object with optional HTTP status code (200 by default).
$app->get('/', function ($req, $res, $args) { return render::hbs($req, $res, [ 'template' => 'index', 'layout' => '_layouts/base', // optional "wrapper" layout template 'title' => 'Page title', // for HTML <title> tag 'data' => [ 'name' => 'Ringo', 'friends' => [ 'Paul', 'George', 'John' ] ], ], 200); // optional status code, 200 by default });
The parser function expects templates to be in a templates
directory with html
file extension. This can be customized by defining these variables in a global settings
array:
$GLOBALS['settings']['templates']['path'] = 'pages'; $GLOBALS['settings']['templates']['extension'] = 'hbs';
Additionally, an array of locals
can be added to make variables available across all templates:
$GLOBALS['locals'] = [ 'year' => date('Y'), 'site_title' => 'Web Site Title', 'site_code' => 'WST', 'site_domain' => 'example.com', ];
Welcome to {{locals.site_title}}, the year is {{locals.year}}!
Parameters from PHP $_GET and $_POST variables are automatically made available to templates rendered with this function, using the variables {{GET}}
and {{POST}}
:
<!-- assuming a url like /hello/?name=Delilah&location=New%20York%20City --> Hey there, {{GET.name}}, what's it like in {{GET.location}}?
Check out the Handlebars Cookbook to see everything you can do with LightnCandy and Handlebars.
Additionally, we've included a few helper functions.
The date
helper applies the PHP date()
function to a given variable or string (or now
keyword for the current time)
Date from unix timestamp: {{date unix_ts_var "d/m/Y"}} Current date: {{date "now" "d/m/Y"}} <!-- use the "now" keyword instead of a variable to use the current time --> Date from non-unix timestamp: {{date non_unix_ts_var "d/m/Y" "convert"}} <!-- add the "convert" parameter to convert the variable to unix time using strtotime() -->
The #is
block helper allows for basic conditional logic:
Is it 1981? {{#is locals.year "==" "1981"}} Yes! {{else}} No! {{/is}}
Custom helpers are easy to create. Take a look at how these helpers are defined in initialize_handlebars_helpers(). The Handlebars cookbook also has a reference for creating custom helpers and custom block helpers.
render::handlebars($parameters)
Renders a specicific Handlebars template with data array the same as render::hbs()
, but returns raw html instead of a PSR-7 response.
$app->get('/', function ($req, $res, $args) { echo render::handlebars([ 'template' => 'email/test', 'data' => [ 'link' => 'https://jy.hxgf.io', ] ]); return $res; });
render::redirect($request, $response, $string, $status = 302)
Renders a redirect as standard Slim (PSR-7) response object with optional HTTP status code.
return render::redirect($req, $res, 'https://google.com/');
render::json($request, $response, $data, $status = 200)
Renders an array or data as standard Slim (PSR-7) response object with application/json
content type and optional HTTP status code.
$app->get('/json/', function ($req, $res, $args) { $data = [ 'name' => 'Ringo', 'friends' => [ 'Paul', 'George', 'John' ] ]; return render::json($req, $res, $data); });
render::lightncandy_html($parameters)($data)
Prepares and compiles a specific Handlebars template with an array of data, including any partials and global locals
variables array.
This is automatically called by render::hbs()
but can be used as a standalone function if desired.
$args = [ 'template' => 'index', 'layout' => '_layouts/base', 'title' => 'Page title', ]; $data = [ 'name' => 'Ringo', 'friends' => [ 'Paul', 'George', 'John' ] ]; echo render::lightncandy_html($args)($data);
render::initialize_handlebars_helpers()
For internal use by lightncandy_html()
. Defines a couple custom Handlebars helper functions to be used by the LightnCandy compiler.
Components
We've also created a component
helper, which allows you to define components that accept props and handle asset management. Components are stored in the templates/_components
directory and can include both a template file ([name].html
) and an optional assets file ([name].assets.html
).
Here are some examples of how to use components:
Simple Component
In the primary template:
{{{component "input" label="Username" type="text" width="w-80" padding="pa3" name="username" value="j_lennon" }}}
In templates/_components/input.html
:
<div class="mb2 {{width}} {{padding}}"> <label class="db mb2 fw7">{{label}}</label> <input class="db w-100 pa2 f5" type="{{type}}" name="{{name}}" value="{{value}}" /> </div>
In templates/_components/input.assets.html
:
<!-- will only be included once, no matter how many instances of `input` component are included on the page --> <script src="/js/lib/uppy.min.js"></script> <link href="/css/lib/uppy.min.css" rel="stylesheet"> <script src="/js/lib/sortable.min.js"></script>
Slot Component
In the primary template:
{{{component "card" title="My Card Title"}}} <p>This is the slot content that will appear inside the card.</p> <p>You can put any HTML content here.</p> {{{/component}}}
In templates/_components/card.html
:
<div class="pa3 ba b--black-20 br2 mb4 shadow-4"> <h3 class="mt0 mb2">{{title}}</h3> <div class="card-content"> {{slot}} </div> </div>
Nested Components (Card containing Inputs)
{{{component "card" title="User Profile"}}} {{{component "input" label="Full Name" type="text" width="w-100" name="fullname" value=user.fullname }}} {{{component "input" label="Email" type="email" width="w-100" name="email" value=user.email }}} {{{/component}}}
Key features of the component helper:
- Use triple curly braces
{{{component}}}
to ensure HTML is not escaped - The first argument is the component name (e.g., "card" or "input")
- All other attributes are passed as hash parameters
- For components that support slots (like the card), content between the opening and closing tags will be placed where
{{slot}}
appears in the template - Assets (like CSS and JS) are automatically loaded if they exist in the component's assets file
- The component helper caches templates and assets for better performance