Files
InvoiceShelf/tests/Feature/Company/Modules/HelloWorldIntegrationTest.php
Darko Gjorgjijoski 7885bf9d11 feat(menu): priority-sorted menu groups, user-menu items, sidebar appearance toggle
Every main_menu entry moves from numeric group (1/2/3) to string-based group + group_label + priority. Groups now carry their own i18n label and child entries are sorted by an explicit priority field instead of config-array order, so module-contributed menu items can slot into any existing group at any position.

BootstrapController merges module-registered menu items into main_menu (previously they lived in a separate module_menu response key) and introduces a user_menu response key for items modules want to place in the avatar dropdown. The global store follows suit: moduleMenu becomes userMenu, menuGroups is a computed that sorts by priority, and hasActiveModules drops out.

New admin Appearance setting page with a single toggle for whether sidebar group labels render — so instances that prefer a compact sidebar can hide the Documents/Administration/Modules headings without losing the grouping itself. CompanyLayout watches route meta and re-bootstraps when the admin-mode flag flips so the sidebar repaints with the right menu on navigation across the admin boundary.

Test suites updated: module menu merging is asserted against main_menu (name: 'module-{slug}') rather than the old module_menu response; HelloWorldIntegrationTest verifies the schema translation path; CompanyModulesIndexTest covers the display_name attachment.
2026-04-11 00:30:00 +02:00

114 lines
4.5 KiB
PHP

<?php
use App\Models\CompanySetting;
use App\Models\Module;
use App\Models\User;
use Illuminate\Support\Facades\Artisan;
use Laravel\Sanctum\Sanctum;
use function Pest\Laravel\getJson;
use function Pest\Laravel\putJson;
/**
* Integration test that exercises the real Modules/HelloWorld module end-to-end
* — no Registry mocking. Proves that when an active module's ServiceProvider
* registers menu + settings via InvoiceShelf\Modules\Registry, the host app's
* company-modules index and settings controllers surface it consistently.
*
* The HelloWorld module's provider boots automatically because nwidart sees
* it in `storage/app/modules_statuses.json` (set to enabled when the module
* was generated via `php artisan module:make HelloWorld`).
*/
beforeEach(function () {
Artisan::call('db:seed', ['--class' => 'DatabaseSeeder', '--force' => true]);
Artisan::call('db:seed', ['--class' => 'DemoSeeder', '--force' => true]);
$user = User::find(1);
$this->companyId = $user->companies()->first()->id;
$this->withHeaders([
'company' => $this->companyId,
]);
Sanctum::actingAs($user, ['*']);
// Mark the module as activated at the InvoiceShelf instance level so it
// shows up in the company-context Active Modules index.
Module::query()->updateOrCreate(
['name' => 'HelloWorld'],
['version' => '1.0.0', 'installed' => true, 'enabled' => true],
);
});
test('bootstrap merges HelloWorld into main_menu under modules group', function () {
$response = getJson('api/v1/bootstrap')->assertOk();
$mainMenu = collect($response->json('main_menu'));
$helloWorld = $mainMenu->firstWhere('name', 'module-hello-world');
expect($helloWorld)->not->toBeNull();
expect($helloWorld['link'])->toBe('/admin/modules/hello-world/dashboard');
expect($helloWorld['icon'])->toBe('HandRaisedIcon');
expect($helloWorld['group'])->toBe('modules');
});
test('HelloWorld appears in the company Active Modules index with translated display name', function () {
$response = getJson('api/v1/company-modules')->assertOk();
// The DB row stores PascalCase but the controller normalizes to kebab-case
// for the URL/registry slug.
$row = collect($response->json('data'))->firstWhere('slug', 'hello-world');
expect($row)->not->toBeNull();
expect($row['name'])->toBe('HelloWorld');
expect($row['display_name'])->toBe('Hello World');
expect($row['has_settings'])->toBeTrue();
expect($row['menu']['title'])->toBe('Hello World');
expect($row['menu']['icon'])->toBe('HandRaisedIcon');
});
test('GET module settings returns the translated HelloWorld schema with defaults', function () {
$response = getJson('api/v1/modules/hello-world/settings')->assertOk();
$sections = $response->json('schema.sections');
expect($sections)->toHaveCount(2);
expect($sections[0]['title'])->toBe('Greeting');
$fields = collect($sections[0]['fields'])->keyBy('key');
expect($fields)->toHaveKeys(['greeting', 'recipient', 'show_emoji']);
expect($fields['greeting']['type'])->toBe('text');
expect($fields['greeting']['label'])->toBe('Greeting message');
expect($fields['greeting']['rules'])->toContain('required');
// Defaults flow through when nothing has been saved yet
$values = $response->json('values');
expect($values['greeting'])->toBe('Hello, world!');
expect($values['show_emoji'])->toBeTrue();
});
test('PUT module settings persists values per company', function () {
putJson('api/v1/modules/hello-world/settings', [
'greeting' => 'Bonjour!',
'recipient' => 'Marie',
'show_emoji' => false,
'tone' => 'formal',
'note' => 'A custom welcome.',
])->assertOk();
expect(CompanySetting::getSetting('module.hello-world.greeting', $this->companyId))
->toBe('Bonjour!');
expect(CompanySetting::getSetting('module.hello-world.show_emoji', $this->companyId))
->toBe('0');
expect(CompanySetting::getSetting('module.hello-world.tone', $this->companyId))
->toBe('formal');
// Re-fetch and confirm the values round-trip through the show endpoint
$response = getJson('api/v1/modules/hello-world/settings')->assertOk();
expect($response->json('values.greeting'))->toBe('Bonjour!');
expect($response->json('values.tone'))->toBe('formal');
});
test('PUT rejects when required fields are missing', function () {
putJson('api/v1/modules/hello-world/settings', [
'recipient' => 'No greeting given',
])->assertStatus(422)
->assertJsonValidationErrors(['greeting', 'tone']);
});