Laravel TRMNL is a package designed to streamline the development of both public and private plugins for the TRMNL E-ink device. It supports data updates via webhooks or polling. For public plugins, it also provides built-in support for the TRMNL OAuth flow. Additionally, UI prototyping is made easier with the included Blade components.
If you are looking for a Laravel based TRMNL Server implementation, check out this repo: usetrmnl/byos_laravel
Support the development of this package by purchasing a TRMNL device through our referral link: https://usetrmnl.com/?ref=laravel-trmnl.
At checkout, use the code laravel-trmnl
to receive a $15 discount on your purchase.
- trmnl-train-monitor by @bnussbau
- 🔌 Public and private plugin support (docs), (docs)
- 🔄 Support for updates via webhooks or polling (docs)
- 🎨 Blade Components on top of the TRMNL Design System (docs)
- 🎯 OAuth integration support for public plugins (docs)
- 📱 Render Helpers for responsive layouts
You can install the package via composer:
composer require bnussbau/laravel-trmnl
Optionally, publish the config file using:
php artisan vendor:publish --tag="trmnl-config"
Optionally, publish the views using:
php artisan vendor:publish --tag="trmnl-views"
The package can be configured through environment variables:
TRMNL_PLUGIN_TYPE= # private | public
TRMNL_DATA_STRATEGY= # polling | webhook
TRMNL_WEBHOOK_URL= # grab from TRMNL Dashboard [Private Plugins]
TRMNL_OAUTH_CLIENT_ID= # grab from TRMNL Dashboard [Public Plugins]
TRMNL_OAUTH_CLIENT_SECRET= # grab from TRMNL Dashboard [Public Plugins]
By default, plugins are private. To configure your settings, add the following environment variables to your .env
file:
TRMNL_PLUGIN_TYPE=private
TRMNL_DATA_STRATEGY=webhook # or polling
TRMNL_WEBHOOK_URL= # grab from TRMNL Dashboard if using webhook
Laravel TRMNL provides the UpdateScreenContentJob
to facilitate sending data updates to TRMNL servers.
UpdateScreenContentJob::dispatchSync([
'key' => 'value',
// ... other variables
]);
UpdateScreenContentJob::dispatchSync(
Journey::whereNotIn('track', [1, 2])
->whereBetween('timestamp_planned', [now()->setTimezone('Europe/Vienna')
->addMinutes(15), now()->setTimezone('Europe/Vienna')->addHours(2)])
->paginate(8)
->toArray()
);
Use the markup editor on the TRMNL private plugin webapp or use the stripMarkup()
Method on the Trmnl
Facade to render markup which you can copy into the editor. See Section Blade Components.
Refer to the “Plugin Marketplace” section in the TRMNL documentation, paying close attention to the authentication flow.
Always verify the authorization token for incoming requests to prevent security issues. You can use Auth::guard('trmnl')
to verify.
TRMNL_PLUGIN_TYPE=public
TRMNL_OAUTH_CLIENT_ID= # grab from TRMNL Dashboard
TRMNL_OAUTH_CLIENT_SECRET= # grab from TRMNL Dashboard
Publishes the database table required for storing authentication data.
php artisan vendor:publish --tag="trmnl-migrations"
php artisan migrate
Public plugins need to provide a render endpoint, which returns markup for all screen layouts.
You can use the Trmnl::renderScreen()
as helper.
Route::post('/render', function () {
// validate Authorization
if (! Auth::guard('trmnl')->validate()) {
return response()->json(['error' => 'Unauthorized'], 401);
}
return response()->json(
Trmnl::renderScreen(
'trmnl.full',
'trmnl.half_horizontal',
'trmnl.half_vertical',
'trmnl.quadrant'
)
);
})->name('trmnl.render');
// make sure to not verify CSRF Token for this route
// ->withoutMiddleware([Illuminate\Foundation\Http\Middleware\VerifyCsrfToken::class])
Run command php artisan trmnl:plugin:configuration
to print the URLs required for plugin submission.
Blade Compontens can help you generate markup code. Alternatively you can just use the native CSS classes from the TRMNL Design System.
<x-trmnl::view>
<x-trmnl::layout>
<!-- Your content here -->
</x-trmnl::layout>
<x-trmnl::title-bar/>
</x-trmnl::view>
<x-trmnl::view>
<x-trmnl::layout>
<x-trmnl::markdown gapSize="large">
<x-trmnl::title>Motivational Quote</x-trmnl::title>
<x-trmnl::content>“I love inside jokes. I hope to be a part of one someday.”</x-trmnl::content>
<x-trmnl::label variant="underline">Michael Scott</x-trmnl::label>
</x-trmnl::markdown>
</x-trmnl::layout>
<x-trmnl::title-bar/>
</x-trmnl::view>
composer test
The MIT License (MIT). Please see License File for more information.