finity-labs/fin-mail

A powerful email template manager and composer for Filament with dynamic token replacement, template versioning, and inline email sending.

Maintainers

Package info

github.com/finity-labs/fin-mail

pkg:composer/finity-labs/fin-mail

Statistics

Installs: 280

Dependents: 0

Suggesters: 0

Stars: 7

Open Issues: 0

v1.3.0 2026-04-01 10:40 UTC

This package is auto-updated.

Last update: 2026-04-01 10:40:53 UTC


README

finity-labs-fin-mail

FILAMENT 4.x FILAMENT 5.x Latest Version on Packagist Tests Code Style Total Downloads License

A powerful email template manager and composer for Filament. Build, manage, and send emails directly from your admin panel — with dynamic token replacement, multilingual templates, customizable themes, template versioning, email logging with status tracking, auth email overrides, and a reusable Send Email action that drops into any resource.

Features

  • Email Composer — Send emails from any resource using templates as starting points, with full editing of subject, body, recipients, and attachments
  • Dynamic Templates — No need for separate Mailable classes per template. One universal TemplateMail handles everything
  • Token Replacement{{ user.name }}, {{ config.app.name }}, conditionals {% if user.is_premium %}, and fallbacks {{ user.name | 'Customer' }}
  • Merge Tags — Tokens are available as merge tags directly in the RichEditor toolbar for easy insertion
  • CTA Button Block — Insert styled call-to-action buttons from the editor with configurable label, URL, and alignment
  • Template Versioning — Automatic version history with preview and one-click restore
  • Email Logging — Every sent email is logged with status tracking, rendered body storage, and polymorphic model association
  • Translatable — Templates support multiple languages via spatie/laravel-translatable, all locales stored in a single record
  • Theme System — Create color themes and apply them to templates, with live preview that updates as you change colors
  • Swappable Editor — Ships with Filament's RichEditor by default, with Tiptap and TinyMCE supported via the EditorContract
  • Categories & Tags — Organize templates as they grow
  • Reusable ActionsSendEmailAction and SentEmailsRelationManager drop into any Filament resource
  • Preview & Test Send — Preview templates inline and send test emails from the admin
  • Admin Settings — Manage sender defaults, branding, logging, and attachment rules from the UI
  • Full Navigation Control — Configure navigation groups, sort order, and visibility per resource from the plugin
  • Shield Support — Built-in policies and permission setup for Filament Shield

Requirements

  • PHP 8.2+
  • Laravel 11, 12, or 13
  • Filament 4 or 5

Installation

composer require finity-labs/fin-mail

Dependency conflict? If you see an error about phpdocumentor/type-resolver, it means your project has phpdocumentor/reflection-docblock 6.x which conflicts with spatie/laravel-settings. Fix it by allowing Composer to resolve all dependencies:

composer require finity-labs/fin-mail phpdocumentor/reflection-docblock:^5.6 -W
php artisan fin-mail:install

The install command will:

  • Publish config and migrations
  • Configure supported locales (auto-detects from your lang/ directory)
  • Optionally run migrations and seed default templates
  • Optionally register the plugin in your Filament panel
  • Optionally register FinMail styles in your custom Filament theme
  • Optionally configure Filament Shield permissions
  • Optionally configure scheduled cleanup of old sent emails

Non-interactive install

Pass --locales to skip the interactive locale prompt (useful for CI or scripted setups):

php artisan fin-mail:install --locales=en,hu,de --seed

During interactive install, choose "Other" from the locale list to manually enter any of the 59 supported locale codes.

Custom Filament theme

If you are using a custom Filament theme, add the FinMail view path to your theme CSS so Tailwind can scan the plugin's styles:

/* resources/css/filament/{panel}/theme.css */
@source '../../../../vendor/finity-labs/fin-mail/resources/views/**/*';

The install command will do this automatically if it detects a custom theme CSS file for the selected panel.

Register the plugin

use FinityLabs\FinMail\FinMailPlugin;

public function panel(Panel $panel): Panel
{
    return $panel
        ->plugins([
            FinMailPlugin::make(),
        ]);
}

Plugin options

FinMailPlugin::make()
    ->enableSentEmails()    // Show the Sent Emails resource (default: true)
    ->enableThemes()        // Show the Themes resource (default: true)
    ->deleteActionOnEditPage() // Show delete button on edit pages (default: false)
    ->policyNamespace('App\\Policies') // Where Shield policies live (default: App\Policies)

Navigation customization

All navigation group settings accept strings, enums, closures, or null:

FinMailPlugin::make()
    // Set a shared navigation group for all resources and settings
    ->navigationGroup('Communications')

    // Or configure each resource independently
    ->emailTemplateNavigationGroup('Email')
    ->emailThemeNavigationGroup('Email')
    ->sentEmailNavigationGroup('Logs')
    ->settingsNavigationGroup('Administration')

    // Set sort order for all resources at once (auto-increments: base, +1, +2, +3)
    ->navigationSort(10)

    // Or configure each resource independently
    ->emailTemplateNavigationSort(10)
    ->emailThemeNavigationSort(20)
    ->sentEmailNavigationSort(30)
    ->settingsNavigationSort(40)

Usage

Sending emails programmatically

use FinityLabs\FinMail\Mail\TemplateMail;

// Simple
Mail::to($user->email)->send(
    TemplateMail::make('welcome-email')
        ->models(['user' => $user])
);

// With locale, attachments, and overrides
Mail::to($customer->email)->send(
    TemplateMail::make('invoice-sent', locale: 'hu')
        ->models(['customer' => $customer, 'invoice' => $invoice])
        ->attachFile($invoice->getPdfPath(), "Invoice-{$invoice->number}.pdf")
);

TemplateMail is automatically queued. Configure the queue connection and name in config/fin-mail.php.

Adding "Send Email" to any resource

use FinityLabs\FinMail\Actions\SendEmailAction;

// In your table actions, header actions, or anywhere Filament actions are used
SendEmailAction::make()
    ->template('invoice-sent')
    ->recipient(fn (Invoice $record) => $record->customer->email)
    ->models(fn (Invoice $record) => [
        'invoice'  => $record,
        'customer' => $record->customer,
    ])
    ->attachments(fn (Invoice $record) => [
        ['path' => $record->getPdfPath(), 'name' => "Invoice-{$record->number}.pdf"],
    ])
    ->onSent(fn (Invoice $record) => $record->update(['emailed_at' => now()]))

A SendEmailAction (page header action) is also available with the same API.

Showing sent emails on any resource

Add the HasEmailTemplates trait to your model:

use FinityLabs\FinMail\Traits\HasEmailTemplates;

class Invoice extends Model
{
    use HasEmailTemplates;
}

Then add the relation manager to your resource:

use FinityLabs\FinMail\Resources\RelationManagers\SentEmailsRelationManager;

public static function getRelations(): array
{
    return [
        SentEmailsRelationManager::class,
    ];
}

The trait provides helpers on your model:

$invoice->sentEmails;                         // All sent emails
$invoice->latestSentEmail();                  // Most recent
$invoice->hasBeenEmailed('invoice-sent');     // Check if a specific template was sent
$invoice->sentEmailsCount();                  // Count

Token syntax

Syntax Example Description
{{ model.attr }} {{ user.name }} Model attribute
{{ model.rel.attr }} {{ order.customer.name }} Nested relation
{{ config.key }} {{ config.app.name }} Config value
{{ token | 'fallback' }} {{ user.name | 'Customer' }} With fallback
{% if token %}...{% endif %} {% if user.is_premium %}...{% endif %} Conditional
{% if token %}...{% else %}...{% endif %} If/else

Merge tags

When editing a template, any tokens defined in the Tokens tab are available as merge tags in the RichEditor toolbar. Click the merge tags button to browse and insert them directly into the email body.

CTA Button block

The editor includes a built-in Button custom block. Click the custom blocks button (squares-plus icon) in the toolbar, select "Button", and configure:

  • Button Text — The label displayed on the button
  • URL — The link destination
  • Alignment — Left, center, or right

The button automatically uses your theme's button colors (button_bg and button_text) in both preview and sent emails, with full inline styling for email client compatibility.

To add your own custom blocks, register them on the RichEditor via the EditorContract or by extending DefaultEditor.

Events

FinMail dispatches events at key points in the email lifecycle so your application can react — e.g., log analytics, trigger webhooks, or update related models.

Event When Payload
EmailSending Before the email is sent SentEmail $sentEmail, ?EmailTemplate $template
EmailSent After the email was sent successfully SentEmail $sentEmail, ?EmailTemplate $template
EmailFailed When sending fails SentEmail $sentEmail, string $error, ?EmailTemplate $template
TemplateUpdated When a template is saved (new version) EmailTemplate $template, int $newVersion

Listening to events

use FinityLabs\FinMail\Events\EmailSent;
use FinityLabs\FinMail\Events\EmailFailed;

// In a service provider or listener
Event::listen(EmailSent::class, function (EmailSent $event) {
    // $event->sentEmail — the SentEmail model
    // $event->template — the EmailTemplate used (nullable)
    logger()->info("Email sent to {$event->sentEmail->recipients_display}");
});

Event::listen(EmailFailed::class, function (EmailFailed $event) {
    // $event->error — the error message
    logger()->error("Email failed: {$event->error}");
});

All event properties are readonly. Events use SerializesModels so they are safe to dispatch from queued jobs.

Filament Shield Integration

FinMail ships with built-in support for Filament Shield. Shield is entirely optional — without it, all resources are accessible to any authenticated user.

Automatic setup

If Shield is installed, the fin-mail:install command will:

  1. Register FinMail resources in your filament-shield.php config
  2. Generate policies and permission entries via shield:generate

FinMail automatically maps Shield-generated policies (in App\Policies by default) to its models. If your policies live elsewhere, configure the namespace on the plugin:

FinMailPlugin::make()
    ->policyNamespace('App\\Policies\\Admin')

Manual setup

If you prefer to set up Shield manually, or if the automatic setup didn't complete:

php artisan shield:generate --panel=admin --option=policies_and_permissions

Supported permissions

Resources:

Resource Permissions
Email Templates ViewAny, View, Create, Update, Delete
Email Themes ViewAny, View, Create, Update, Delete
Sent Emails ViewAny, View

Settings pages:

Each settings page (General, Branding, Logging, Attachments, Auth Emails) has its own page-level permission managed by Shield.

Cleanup on uninstall

The fin-mail:uninstall command automatically removes Shield config entries and permission records from the database.

Auth Email Overrides

FinMail can replace the application's default authentication emails (verification, password reset) with your custom templates, and optionally send a welcome email on registration.

Enable overrides

Navigate to Settings → Auth Emails in the admin panel and toggle the overrides you want.

Required templates

Create templates with these keys (the seeder includes them by default):

Template Key Purpose Available Tokens
user-verify-email Email verification link {{ user.name }}, {{ user.email }}, {{ url }}
user-password-reset Password reset link {{ user.name }}, {{ user.email }}, {{ url }}
user-welcome Welcome email after registration {{ user.name }}, {{ user.email }}

Locale support

Auth email overrides automatically use the active application locale (app()->getLocale()). If you use a language switcher plugin, the emails will be sent in the user's selected language — provided the template has a translation for that locale.

Configuration

Publish the config:

php artisan vendor:publish --tag=fin-mail-config

Date formatting

By default, dates and datetimes throughout the plugin use Filament's built-in formatting. You can override this globally or per locale in config/fin-mail.php:

// A single format for all locales
'date_format' => 'd/m/Y',
'datetime_format' => 'd/m/Y H:i',

// Or an array keyed by locale
'date_format' => [
    'en' => 'M d, Y',
    'de' => 'd.m.Y',
    'hu' => 'Y. m. d.',
],
'datetime_format' => [
    'en' => 'M d, Y H:i',
    'de' => 'd.m.Y H:i',
    'hu' => 'Y. m. d. H:i',
],

When set to null (or when the current locale isn't in the array), Filament's default formatting kicks in. These formats are standard PHP date format characters.

You can also access the resolved format programmatically:

use FinityLabs\FinMail\Facades\FinMail;

FinMail::dateFormat();     // string|null for current locale
FinMail::dateTimeFormat(); // string|null for current locale

Other publish tags:

Tag Description
fin-mail-config Configuration file
fin-mail-migrations Database migrations
fin-mail-settings-migrations Spatie Settings migrations
fin-mail-views Email template views

Upgrading

When upgrading from a previous version, run the upgrade command to apply any data migrations:

php artisan fin-mail:upgrade

This checks locked templates in the database against the latest seeder definitions and updates any that are outdated. The command is idempotent and safe to run multiple times.

Preview changes without applying them:

php artisan fin-mail:upgrade --dry-run

Uninstalling

Run the uninstall command before removing the package:

php artisan fin-mail:uninstall
composer remove finity-labs/fin-mail

The uninstall command will:

  • Remove FinMailPlugin::make() from your panel provider(s)
  • Remove FinMail @source directive from custom Filament theme CSS files
  • Remove FinMail entries from Shield config and clean up permissions from the database
  • Optionally drop all FinMail database tables and settings entries
  • Optionally delete published migrations, config, views, and translations
  • Clear settings and application caches

Testing

composer test

License

MIT

Screenshots

📝 Template Management

Template List Manage, search, and create email templates with multi-language support. Email Templates List View

Template Editor The editor includes a live preview and a dynamic token selector. Template Editor Editor Tokens View

🎨 Theming

Theme Editor Create a consistent brand look with the visual theme editor — no CSS knowledge required. Theme Editor

⚙️ Plugin Configuration

General Settings Configure senders, localization, and template categories. General Settings

Branding Settings Customize logo, colors, and footer links for your email layout. Branding Settings

Logging Settings Control how sent emails are recorded and cleaned up. Logging Settings

Attachment Settings Set file size limits and allowed extensions. Attachment Settings

Auth Email Overrides Replace default Laravel auth emails with your custom templates. Auth Override Settings