Laravel アプリケーションが成長するにつれて、特にロールベースのアクセス制御などの動的要素の場合、ナビゲーション メニューの管理が困難になることがあります。このブログ投稿では、メニュー ビルダー システムを使用してメニューを簡素化し、構造化し、メンテナンス、拡張、拡張を容易にする方法について説明します。
多くの Laravel プロジェクトでは、Blade テンプレートは条件文を使用してメニューの可視性を処理します。
@can('viewAdmin') <a href="{{ route('administration.index') }}"> {{ __('Administration') }} </a> @endcan
このアプローチは単純なアプリケーションでは機能しますが、メニューの数が増えると乱雑になり管理できなくなります。
メニュー ビルダー システム は、メニュー ロジックを再利用可能なクラスにカプセル化し、以下を改善します。
私の仕事を後援することで、開発者コミュニティに力を与えるという私の使命をサポートしてください。あなたの貢献は、貴重なツール、洞察、リソースを構築して共有するのに役立ちます。詳細については、こちらをご覧ください。
管理メニューへのアクセスを制御するには、AuthServiceProvider で viewAdmin ゲートを定義します。
use Illuminate\Support\Facades\Gate; use App\Models\User; class AuthServiceProvider extends ServiceProvider { public function boot() { $this->registerPolicies(); Gate::define('viewAdmin', function (User $user) { return $user->hasRole('admin'); // Replace with your app's role-checking logic }); } }
MenuItem クラスは、ラベル、URL、アイコン、可視性などのメニュー項目のすべての属性を定義します。
<?php namespace App\Actions\Builder; use CleaniqueCoders\Traitify\Contracts\Builder; use InvalidArgumentException; class MenuItem implements Builder { private string $label; private string $url; private string $target = '_self'; private array $attributes = []; private array $children = []; private string $icon = 'o-squares-2x2'; private ?string $description = null; private ?string $tooltip = null; private $visible = true; private array $output = []; public function setLabel(string $label): self { $this->label = $label; return $this; } public function setUrl(string $url): self { $this->url = $url; return $this; } public function setTarget(string $target): self { $this->target = $target; return $this; } public function addAttribute(string $key, string $value): self { $this->attributes[$key] = $value; return $this; } public function addChild(MenuItem $child): self { $this->children[] = $child; return $this; } public function setIcon(string $icon): self { $this->icon = $icon; return $this; } public function setDescription(string $description): self { $this->description = $description; return $this; } public function setTooltip(string $tooltip): self { $this->tooltip = $tooltip; return $this; } public function setVisible($visible): self { if (! is_bool($visible) && ! is_callable($visible)) { throw new InvalidArgumentException('The visible property must be a boolean or a callable.'); } $this->visible = $visible; return $this; } public function isVisible(): bool { return is_callable($this->visible) ? call_user_func($this->visible) : $this->visible; } public function build(): self { $this->output = [ 'label' => $this->label, 'url' => $this->url, 'target' => $this->target, 'attributes' => $this->attributes, 'icon' => $this->icon, 'description' => $this->description, 'tooltip' => $this->tooltip, 'children' => array_filter( array_map(fn (MenuItem $child) => $child->build()->toArray(), $this->children), fn (array $child) => ! empty($child) ), ]; return $this; } public function toArray(): array { return $this->output; } public function toJson(int $options = 0): string { return json_encode($this->toArray(), $options, 512); } }
メニュー ビルダーはメニューを動的に解決して構築します。
namespace App\Actions\Builder; class Menu { public static function make() { return new self; } public function build(string $builder) { $class = match ($builder) { 'navbar' => Navbar::class, 'sidebar' => Sidebar::class, 'administration' => Administration::class, default => Navbar::class, }; $builder = new $class; return $builder->build(); } }
ヘルパー関数を使用してメニューにアクセスします:
<?php use App\Actions\Builder\Menu; if (! function_exists('menu')) { function menu(string $builder) { return Menu::make()->build($builder)->menus(); } }
Administration クラスで管理固有のメニュー項目を定義します。
<?php namespace App\Actions\Builder\Menu; use App\Actions\Builder\MenuItem; use CleaniqueCoders\Traitify\Contracts\Builder; use CleaniqueCoders\Traitify\Contracts\Menu; use Illuminate\Support\Collection; use Illuminate\Support\Facades\Gate; class Administration implements Builder, Menu { private Collection $menus; public function menus(): Collection { return $this->menus; } public function build(): self { $this->menus = collect([ (new MenuItem) ->setLabel(__('Issues')) ->setUrl(url(config('telescope.path'))) ->setTarget('_blank') ->setVisible(fn () => Gate::allows('viewTelescope')) ->setTooltip(__('View Telescope issues')) ->setDescription(__('Access application issues using Laravel Telescope')) ->setIcon('o-bug'), // Heroicon outline for a bug (new MenuItem) ->setLabel(__('Queues')) ->setUrl(url(config('horizon.path'))) ->setTarget('_blank') ->setVisible(fn () => Gate::allows('viewHorizon')) ->setTooltip(__('Manage queues')) ->setDescription(__('Access Laravel Horizon to monitor and manage queues')) ->setIcon('o-cog'), // Heroicon outline for settings/tasks (new MenuItem) ->setLabel(__('Access Control')) ->setUrl(route('security.access-control.index')) ->setVisible(fn () => Gate::allows('viewAccessControl')) ->setTooltip(__('Manage access control')) ->setDescription(__('Define and manage access control rules')) ->setIcon('o-lock-closed'), (new MenuItem) ->setLabel(__('Users')) ->setUrl(route('security.users.index')) ->setVisible(fn () => Gate::allows('viewUser')) ->setTooltip(__('Manage users')) ->setDescription(__('View and manage user accounts')) ->setIcon('o-user-group'), (new MenuItem) ->setLabel(__('Audit Trail')) ->setUrl(route('security.audit-trail.index')) ->setVisible(fn () => Gate::allows('viewAudit')) ->setTooltip(__('View audit trails')) ->setDescription(__('Audit logs for security and activity tracking')) ->setIcon('o-document-text'), ])->reject(fn (MenuItem $menu) => ! $menu->isVisible()) ->map(fn (MenuItem $menu) => $menu->build()->toArray()); return $this; } }
管理ページに次のルート構成を追加します:
<?php use Illuminate\Support\Facades\Route; Route::middleware(['auth:sanctum', 'verified', 'can:viewAdmin']) ->as('administration.') ->prefix('administration') ->group(function () { Route::view('/', 'administration.index')->name('index'); });
ナビゲーション メニュー (navigation-menu.blade.php):
@can('viewAdmin') <a href="{{ route('administration.index') }}"> <x-icon name="o-computer-desktop" /> {{ __('Administration') }} </a> @endcan
管理メニュー (administration/index.blade.php):
<x-app-layout> <x-スロット名="ヘッダー">{{ __('管理') }}</x-スロット> <div> <hr> <p><strong>出力</strong></p> <p>ここで得られる最終出力は次のとおりです:</p> <p><img src="https://img.php.cn/upload/article/000/000/000/173418649412401.jpg" alt="Building Dynamic and Maintainable Menus in Laravel"></p> <blockquote> <p>私の仕事を後援することで、開発者コミュニティに力を与えるという私の使命をサポートしてください。あなたの貢献は、貴重なツール、洞察、リソースを構築して共有するのに役立ちます。詳細については、こちらをご覧ください。</p> </blockquote> <hr> <h3> <strong>結論</strong> </h3> <p>この<strong>メニュービルダーシステム</strong>は、次の方法でLaravelのナビゲーション管理を簡素化します。</p> <ol> <li>メニュー定義を一元化して保守性を向上させます。</li> <li>役割または権限を使用してメニューの表示を動的に制御します。</li> <li>ビューとレイアウト間でメニュー ロジックを再利用します。</li> </ol> <p>このアプローチを採用すると、複雑なアプリケーションであってもナビゲーション システムをシームレスに拡張できます。 </p> <p>データベースからメニューの詳細をロードし、必要なメニューを構築することもできます。しかし、私にとってはこれで十分です。データベース主導のメニュー構成を使用する必要があるプロジェクトはありません。</p> <p>コードはここにあります。 </p> <p>試して感想をシェアしてください! ?</p> <hr> <p>Unsplash の LinedPhoto による写真</p> </div> </x-app-layout>
以上がLaravel で動的で保守可能なメニューを構築するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。