Files
InvoiceShelf/app/Providers/AppServiceProvider.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

192 lines
6.5 KiB
PHP

<?php
namespace App\Providers;
use App\Policies\CompanyPolicy;
use App\Policies\CustomerPolicy;
use App\Policies\DashboardPolicy;
use App\Policies\EstimatePolicy;
use App\Policies\ExpensePolicy;
use App\Policies\InvoicePolicy;
use App\Policies\ItemPolicy;
use App\Policies\ModulesPolicy;
use App\Policies\NotePolicy;
use App\Policies\OwnerPolicy;
use App\Policies\PaymentPolicy;
use App\Policies\RecurringInvoicePolicy;
use App\Policies\ReportPolicy;
use App\Policies\RolePolicy;
use App\Policies\SettingsPolicy;
use App\Policies\UserPolicy;
use App\Services\Setup\InstallUtils;
use App\Support\BouncerDefaultScope;
use App\Support\InstallWizardAuth;
use Gate;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Broadcast;
use Illuminate\Support\Facades\Mail;
use Illuminate\Support\Facades\Notification;
use Illuminate\Support\ServiceProvider;
use Laravel\Sanctum\Sanctum;
use Silber\Bouncer\Database\Models as BouncerModels;
use Silber\Bouncer\Database\Role;
use View;
class AppServiceProvider extends ServiceProvider
{
/**
* The path to your application's "home" route.
*
* Typically, users are redirected here after authentication.
*
* @var string
*/
public const HOME = '/admin/dashboard';
/**
* The path to the "customer home" route for your application.
*
* This is used by Laravel authentication to redirect customers after login.
*
* @var string
*/
public const CUSTOMER_HOME = '/customer/dashboard';
/**
* Bootstrap any application services.
*/
public function boot(): void
{
$this->configureInstallWizardTokenAuth();
if (InstallUtils::isDbCreated()) {
$this->addMenus();
}
Gate::policy(Role::class, RolePolicy::class);
View::addNamespace('pdf_templates', storage_path('app/templates/pdf'));
$this->bootAuth();
$this->bootBroadcast();
// In demo mode, prevent all outgoing emails and notifications
if (config('app.env') === 'demo') {
Mail::fake();
Notification::fake();
}
}
/**
* Register any application services.
*/
public function register(): void
{
BouncerModels::scope(new BouncerDefaultScope);
}
public function addMenus()
{
// main menu
\Menu::make('main_menu', function ($menu) {
foreach (config('invoiceshelf.main_menu') as $data) {
$this->generateMenu($menu, $data);
}
});
// admin menu (super admin mode)
\Menu::make('admin_menu', function ($menu) {
foreach (config('invoiceshelf.admin_menu') as $data) {
$this->generateMenu($menu, $data);
}
});
// setting menu
\Menu::make('setting_menu', function ($menu) {
foreach (config('invoiceshelf.setting_menu') as $data) {
$this->generateMenu($menu, $data);
}
});
\Menu::make('customer_portal_menu', function ($menu) {
foreach (config('invoiceshelf.customer_menu') as $data) {
$this->generateMenu($menu, $data);
}
});
}
public function generateMenu($menu, $data)
{
$menu->add($data['title'], $data['link'])
->data('icon', $data['icon'])
->data('name', $data['name'])
->data('owner_only', $data['owner_only'])
->data('super_admin_only', $data['super_admin_only'] ?? false)
->data('ability', $data['ability'])
->data('model', $data['model'])
->data('group', $data['group'])
->data('group_label', $data['group_label'] ?? '')
->data('priority', $data['priority'] ?? 100);
}
public function bootAuth()
{
Gate::define('create company', [CompanyPolicy::class, 'create']);
Gate::define('transfer company ownership', [CompanyPolicy::class, 'transferOwnership']);
Gate::define('delete company', [CompanyPolicy::class, 'delete']);
Gate::define('manage modules', [ModulesPolicy::class, 'manageModules']);
Gate::define('manage settings', [SettingsPolicy::class, 'manageSettings']);
Gate::define('manage company', [SettingsPolicy::class, 'manageCompany']);
Gate::define('manage backups', [SettingsPolicy::class, 'manageBackups']);
Gate::define('manage file disk', [SettingsPolicy::class, 'manageFileDisk']);
Gate::define('manage email config', [SettingsPolicy::class, 'manageEmailConfig']);
Gate::define('manage pdf config', [SettingsPolicy::class, 'managePDFConfig']);
Gate::define('manage notes', [NotePolicy::class, 'manageNotes']);
Gate::define('view notes', [NotePolicy::class, 'viewNotes']);
Gate::define('send invoice', [InvoicePolicy::class, 'send']);
Gate::define('send estimate', [EstimatePolicy::class, 'send']);
Gate::define('send payment', [PaymentPolicy::class, 'send']);
Gate::define('delete multiple items', [ItemPolicy::class, 'deleteMultiple']);
Gate::define('delete multiple customers', [CustomerPolicy::class, 'deleteMultiple']);
Gate::define('delete multiple users', [UserPolicy::class, 'deleteMultiple']);
Gate::define('delete multiple invoices', [InvoicePolicy::class, 'deleteMultiple']);
Gate::define('delete multiple estimates', [EstimatePolicy::class, 'deleteMultiple']);
Gate::define('delete multiple expenses', [ExpensePolicy::class, 'deleteMultiple']);
Gate::define('delete multiple payments', [PaymentPolicy::class, 'deleteMultiple']);
Gate::define('delete multiple recurring invoices', [RecurringInvoicePolicy::class, 'deleteMultiple']);
Gate::define('view dashboard', [DashboardPolicy::class, 'view']);
Gate::define('view report', [ReportPolicy::class, 'viewReport']);
Gate::define('owner only', [OwnerPolicy::class, 'managedByOwner']);
}
public function bootBroadcast()
{
Broadcast::routes(['middleware' => 'api.auth']);
}
private function configureInstallWizardTokenAuth(): void
{
Sanctum::authenticateAccessTokensUsing(function ($accessToken, bool $isValid): bool {
if (! $isValid) {
return false;
}
$request = request();
if (! $request instanceof Request || ! $request->attributes->get('install_wizard', false)) {
return $isValid;
}
return $accessToken->can(InstallWizardAuth::TOKEN_ABILITY);
});
}
}