feat(api): organisation email branding and shared mail layout

- Add email branding columns to organisations table (logo, color, reply-to, sender name, footer)
- Create MailBrandingService for resolving per-org branding with defaults
- Create CrewliMailable abstract base class with branded from/reply-to
- Create shared Blade layout (mail.layouts.crewli) with inline CSS
- Refactor Registration*Mail and InvitationMail to extend CrewliMailable
- Add config/crewli.php for platform-wide defaults (portal_url, app_url, logo)
- Add dev-only /mail-preview/{type} route for browser email previewing
- Update Organisation model, resource, and form requests with branding fields

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-04-13 00:44:34 +02:00
parent de8ebf724b
commit ec4ba8733d
21 changed files with 739 additions and 60 deletions

View File

@@ -0,0 +1,90 @@
<?php
declare(strict_types=1);
namespace Tests\Feature\Mail;
use App\Models\Organisation;
use App\Services\MailBrandingService;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
class MailBrandingServiceTest extends TestCase
{
use RefreshDatabase;
private MailBrandingService $service;
protected function setUp(): void
{
parent::setUp();
$this->service = new MailBrandingService();
}
public function test_returns_org_branding_when_configured(): void
{
$org = Organisation::factory()->create([
'name' => 'Festival BV',
'email_logo_url' => 'https://example.com/logo.png',
'email_primary_color' => '#ff5500',
'email_reply_to' => 'info@festival.nl',
'email_sender_name' => 'Festival Team',
'email_footer_text' => 'Stichting Festival, Amsterdam',
]);
$branding = $this->service->getBranding($org);
$this->assertEquals('Festival BV', $branding['organisation_name']);
$this->assertEquals('https://example.com/logo.png', $branding['logo_url']);
$this->assertEquals('#ff5500', $branding['primary_color']);
$this->assertEquals('info@festival.nl', $branding['reply_to']);
$this->assertEquals('Festival Team', $branding['sender_name']);
$this->assertEquals('Stichting Festival, Amsterdam', $branding['footer_text']);
}
public function test_falls_back_to_defaults_when_not_configured(): void
{
$org = Organisation::factory()->create([
'name' => 'Lege Org',
'email_logo_url' => null,
'email_primary_color' => null,
'email_reply_to' => null,
'email_sender_name' => null,
'email_footer_text' => null,
]);
$branding = $this->service->getBranding($org);
$this->assertEquals('Lege Org', $branding['organisation_name']);
$this->assertNull($branding['logo_url']);
$this->assertEquals('#6366f1', $branding['primary_color']);
$this->assertNull($branding['reply_to']);
$this->assertEquals('Lege Org', $branding['sender_name']);
$this->assertNull($branding['footer_text']);
}
public function test_sender_name_falls_back_to_org_name(): void
{
$org = Organisation::factory()->create([
'name' => 'Mijn Organisatie',
'email_sender_name' => null,
]);
$branding = $this->service->getBranding($org);
$this->assertEquals('Mijn Organisatie', $branding['sender_name']);
}
public function test_logo_url_falls_back_to_config_default(): void
{
config(['crewli.default_logo_url' => 'https://crewli.app/logo.png']);
$org = Organisation::factory()->create([
'email_logo_url' => null,
]);
$branding = $this->service->getBranding($org);
$this->assertEquals('https://crewli.app/logo.png', $branding['logo_url']);
}
}

View File

@@ -0,0 +1,157 @@
<?php
declare(strict_types=1);
namespace Tests\Feature\Mail;
use App\Mail\CrewliMailable;
use App\Mail\InvitationMail;
use App\Mail\RegistrationApprovedMail;
use App\Mail\RegistrationConfirmationMail;
use App\Mail\RegistrationRejectedMail;
use App\Models\CrowdType;
use App\Models\Event;
use App\Models\Organisation;
use App\Models\Person;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
class MailLayoutTest extends TestCase
{
use RefreshDatabase;
private Organisation $organisation;
private Event $event;
private Person $person;
protected function setUp(): void
{
parent::setUp();
$this->organisation = Organisation::factory()->create([
'name' => 'Test Festival BV',
'email_logo_url' => 'https://example.com/logo.png',
'email_primary_color' => '#e11d48',
'email_footer_text' => 'Stichting Test Festival, Utrecht',
]);
$this->event = Event::factory()->create([
'organisation_id' => $this->organisation->id,
'status' => 'registration_open',
]);
$crowdType = CrowdType::factory()->systemType('VOLUNTEER')->create([
'organisation_id' => $this->organisation->id,
]);
$this->person = Person::factory()->create([
'event_id' => $this->event->id,
'crowd_type_id' => $crowdType->id,
'first_name' => 'Jan',
'last_name' => 'Tester',
'email' => 'jan@test.nl',
]);
}
public function test_confirmation_email_renders_with_org_logo(): void
{
$mail = new RegistrationConfirmationMail($this->person, $this->event);
$html = $mail->render();
$this->assertStringContainsString('https://example.com/logo.png', $html);
$this->assertStringContainsString('Test Festival BV', $html);
}
public function test_confirmation_email_renders_with_primary_color(): void
{
$mail = new RegistrationConfirmationMail($this->person, $this->event);
$html = $mail->render();
$this->assertStringContainsString('#e11d48', $html);
}
public function test_confirmation_email_renders_with_custom_footer(): void
{
$mail = new RegistrationConfirmationMail($this->person, $this->event);
$html = $mail->render();
$this->assertStringContainsString('Stichting Test Festival, Utrecht', $html);
}
public function test_confirmation_email_renders_with_defaults_when_no_branding(): void
{
$plainOrg = Organisation::factory()->create([
'name' => 'Plain Org',
'email_logo_url' => null,
'email_primary_color' => null,
'email_footer_text' => null,
]);
$event = Event::factory()->create([
'organisation_id' => $plainOrg->id,
'status' => 'registration_open',
]);
$crowdType = CrowdType::factory()->systemType('VOLUNTEER')->create([
'organisation_id' => $plainOrg->id,
]);
$person = Person::factory()->create([
'event_id' => $event->id,
'crowd_type_id' => $crowdType->id,
]);
$mail = new RegistrationConfirmationMail($person, $event);
$html = $mail->render();
// Default primary color
$this->assertStringContainsString('#6366f1', $html);
// Org name is shown
$this->assertStringContainsString('Plain Org', $html);
// Powered by Crewli always present
$this->assertStringContainsString('Powered by Crewli', $html);
}
public function test_approved_email_renders_with_branding(): void
{
$mail = new RegistrationApprovedMail($this->person, $this->event);
$html = $mail->render();
$this->assertStringContainsString('https://example.com/logo.png', $html);
$this->assertStringContainsString('#e11d48', $html);
$this->assertStringContainsString('Goed nieuws', $html);
}
public function test_rejected_email_renders_with_reason(): void
{
$mail = new RegistrationRejectedMail($this->person, $this->event, 'Geen plek meer.');
$html = $mail->render();
$this->assertStringContainsString('Geen plek meer.', $html);
$this->assertStringContainsString('Test Festival BV', $html);
}
public function test_rejected_email_renders_without_reason(): void
{
$mail = new RegistrationRejectedMail($this->person, $this->event);
$html = $mail->render();
$this->assertStringNotContainsString('Reden:', $html);
}
public function test_all_mailables_extend_crewli_mailable(): void
{
$this->assertInstanceOf(CrewliMailable::class, new RegistrationConfirmationMail($this->person, $this->event));
$this->assertInstanceOf(CrewliMailable::class, new RegistrationApprovedMail($this->person, $this->event));
$this->assertInstanceOf(CrewliMailable::class, new RegistrationRejectedMail($this->person, $this->event));
}
public function test_powered_by_crewli_always_present(): void
{
$mail = new RegistrationConfirmationMail($this->person, $this->event);
$html = $mail->render();
$this->assertStringContainsString('Powered by Crewli', $html);
$this->assertStringContainsString('Je ontvangt deze email omdat je bent aangemeld bij', $html);
}
}