Files
InvoiceShelf/tests/Feature/Company/Modules/CompanyModulesIndexTest.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

106 lines
2.8 KiB
PHP

<?php
use App\Models\Module;
use App\Models\User;
use Illuminate\Support\Facades\Artisan;
use InvoiceShelf\Modules\Registry;
use Laravel\Sanctum\Sanctum;
use function Pest\Laravel\getJson;
beforeEach(function () {
Artisan::call('db:seed', ['--class' => 'DatabaseSeeder', '--force' => true]);
Artisan::call('db:seed', ['--class' => 'DemoSeeder', '--force' => true]);
$user = User::find(1);
$this->withHeaders([
'company' => $user->companies()->first()->id,
]);
Sanctum::actingAs($user, ['*']);
Registry::flush();
});
afterEach(function () {
Registry::flush();
});
test('returns only enabled modules', function () {
Module::create([
'name' => 'sales-tax-us',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Module::create([
'name' => 'archived-module',
'version' => '0.5.0',
'installed' => true,
'enabled' => false,
]);
$response = getJson('api/v1/company-modules')->assertOk();
$response->assertJsonCount(1, 'data');
$response->assertJsonPath('data.0.slug', 'sales-tax-us');
});
test('reports has_settings flag based on Registry', function () {
Module::create([
'name' => 'with-settings',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Module::create([
'name' => 'without-settings',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Registry::registerSettings('with-settings', [
'sections' => [
['title' => 'general', 'fields' => [
['key' => 'foo', 'type' => 'text'],
]],
],
]);
$response = getJson('api/v1/company-modules')->assertOk();
$rows = collect($response->json('data'))->keyBy('slug');
expect($rows['with-settings']['has_settings'])->toBeTrue();
expect($rows['without-settings']['has_settings'])->toBeFalse();
});
test('includes registered menu entry for active modules', function () {
Module::create([
'name' => 'menu-module',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Registry::registerMenu('menu-module', [
'title' => 'menu_module::menu.title',
'link' => '/admin/modules/menu-module/settings',
'icon' => 'CalculatorIcon',
]);
$response = getJson('api/v1/company-modules')->assertOk();
$response->assertJsonPath('data.0.display_name', 'Menu Module');
$response->assertJsonPath('data.0.menu.title', 'menu_module::menu.title');
$response->assertJsonPath('data.0.menu.icon', 'CalculatorIcon');
});
test('returns empty array when no modules are enabled', function () {
getJson('api/v1/company-modules')
->assertOk()
->assertJsonPath('data', []);
});