A quick way to create menus in Laravel 4.x
##Documentation
- Installation
- Getting Started
- Routing
- Sub-items
- Referring to Items
- Referring to Menu Objects
- HTML Attributes
- Maniuplating Links
- Active Item
- Inserting a Separator
- Append and Prepend
- Raw Items
- Menu Groups
- URL Prefixing
- Nested Groups
- Meta Data
- Filtering the Items
- Sorting the Items
- Rendering Methods
- Advanced Usage
- Configuration
- If You Need Help
- License
In the require
key of composer.json
file add lavary/laravel-menu": "dev-master
:
...
"require": {
"laravel/framework": "4.2.*",
"lavary/laravel-menu": "dev-master"
}
Run the composer update command:
composer update
Now append Laravel Menu service provider to providers
array in app/config/app.php
.
<?php
'providers' => array(
'Illuminate\Foundation\Providers\ArtisanServiceProvider',
'Illuminate\Auth\AuthServiceProvider',
...
'Lavary\Menu\ServiceProvider',
),
?>
At the end of config/app.php
add 'Menu' => 'Lavary\Menu\Facade'
to the $aliases
array:
<?php
'aliases' => array(
'App' => 'Illuminate\Support\Facades\App',
'Artisan' => 'Illuminate\Support\Facades\Artisan',
...
'Menu' => 'Lavary\Menu\Facade',
),
?>
This registers the package with Laravel and creates an alias called Menu
.
Menus can be defined in app/routes.php
or start/global.php
or any other place you wish as long as it is auto loaded when a request hits your application.
Here is a basic usage:
<?php
Menu::make('MyNavBar', function($menu){
$menu->add('Home');
$menu->add('About', 'about');
$menu->add('services', 'services');
$menu->add('Contact', 'contact');
});
?>
Attention $MyNavBar
is just a hypothetical name I used in these examples; You can name your menus whatever you please.
In the above example Menu::make()
creates a menu named MyNavBar
, Adds the menu instance to the Menu::collection
and ultimately makes $myNavBar
object available across all views.
This method accepts a callable inside which you can define your items. add
method defines a new item. It receives two parameters, the first one is the item title and the second one is options.
options can be a simple string representing a URL or an associative array of options and HTML attributes which we'll discuss shortly.
To render the menu in your view:
Laravel-menu
provides three rendering methods out of the box. However you can create your own rendering method using the right methods and attributes.
As noted earlier, laravel-menu
provides three rendering formats out of the box, asUl()
, asOl()
and asDiv()
. You can read about the details here.
{{ $MyNavBar->asUl() }}
You can also access the menu object via the menu collection:
{{ Menu::get('MyNavBar')->asUl() }}
This will render your menu like so:
<ul>
<li><a href="http://yourdomain.com">Home</a></li>
<li><a href="http://yourdomain.com/about">About</a></li>
<li><a href="http://yourdomain.com/services">Services</a></li>
<li><a href="http://yourdomain.com/contact">Contact</a></li>
</ul>
And that's all about it!
You can simply assign a URL to your menu item by passing the URL as the second argument to add
method:
<?php
// ...
$menu->add('About Us', 'about-us');
// ...
laravel-menu
supports named routes as well:
This time instead of passing a simple string to add()
, we pass an associative with key route
and a named route as value:
<?php
// Suppose we have these routes defined in our app/routes.php file
//...
Route::get('/', array('as' => 'home.page', function(){...}));
Route::get('about', array('as' => 'page.about', function(){...}));
//...
// Now we make the menu:
Menu::make('MyNavBar', function($menu){
$menu->add('Home', array('route' => 'home.page'));
$menu->add('About', array('route' => 'page.about'));
// ...
});
?>
Laravel Menu supports controller actions as well.
You will just need to set action
key of your options array to a controller action:
<?php
// Suppose we have these routes defined in our app/routes.php file
// ...
Route::get('services', 'ServiceController@index');
//...
// ...
$menu->add('services', array('action' => 'ServicesController@index'));
// ...
?>
Note: if you need to send some data to routes, URLs or controller actions as a query string, you can simply include them in an array along with the route, action or URL value:
<?php
Menu::make('MyNavBar', function($menu){
$menu->add('Home', array('route' => 'home.page'));
$menu->add('About', array('route' => array('page.about', 'template' => 1)));
$menu->add('services', array('action' => array('ServicesController@index', 'id' => 12)));
$menu->add('Contact', 'contact');
});
?>
If you need to serve the route over HTTPS, call secure()
on the item's link
attribute or alternatively add key secure
to the options array and set it to true
:
<?php
// ...
$menu->add('Members', 'members')->link->secure();
// or alternatively use this shortcut
$menu->add('Members', array('url' => 'members', 'secure' => true));
// ...
?>
The output as <ul>
would be:
<ul>
...
<li><a href="https://yourdomain.com/members">Members</a></li>
...
</ul>
Items can have sub-items too:
<?php
Menu::make('MyNavBar', function($menu){
//...
$menu->add('About', array('route' => 'page.about'));
// these items will go under Item 'About'
// refer to about as a property of $menu object then call `add()` on it
$menu->about->add('Who We are', 'who-we-are');
// or
$menu->get('about')->add('What We Do', 'what-we-do');
// or
$menu->item('about')->add('Our Goals', 'our-goals');
//...
});
?>
You can also chain the item definitions and go as deep as you wish:
<?php
// ...
$menu->add('About', array('route' => 'page.about'))
->add('Level2', 'link address')
->add('level3', 'Link address')
->add('level4', 'Link address');
// ...
?>
It is possible to add sub items directly using parent
attribute:
<?php
//...
$menu->add('About', array('route' => 'page.about'));
$menu->add('Level2', array('url' => 'Link address', 'parent' => $menu->about->id));
//...
?>
You can access defined items throughout your code using the methods described below.
along with item's title in camel case:
<?php
// ...
$menu->itemTitleInCamelCase ...
// or
$menu->get('itemTitleInCamelCase') ...
// or
$menu->item('itemTitleInCamelCase') ...
// ...
?>
As an example, let's insert a divider after About us
item after we've defined it:
<?php
// ...
$menu->add('About us', 'about-us')
$menu->aboutUs->divide();
// or
$menu->get('aboutUs')->divide();
// or
$menu->item('aboutUs')->divide();
// ...
?>
If you're not comfortable with the above method you can store the item's object reference in a variable for further reference:
<?php
// ...
$about = $menu->add('About', 'about');
$about->add('Who We Are', 'who-we-are');
$about->add('What We Do', 'what-we-do');
// ...
You can also get an item by Id if needed:
<?php
// ...
$menu->find(12) ...
// ...
?>
<?php
// ...
$menu->all();
// or outside of the builder context
Menu::get('MyNavBar')->all();
// ...
?>
all()
returns a Laravel Collection.
<?php
// ...
$menu->first();
// or outside of the builder context
Menu::get('MyNavBar')->first();
// ...
?>
<?php
// ...
$menu->last();
// or outside of the builder context
Menu::get('MyNavBar')->last();
// ...
?>
First of all you need to get the item using the methods described above then call children()
on it.
To get children of About
item:
<?php
// ...
$aboutSubs = $menu->about->children();
// or outside of the builder context
$aboutSubs = Menu::get('MyNavBar')->about->children();
// Or
$aboutSubs = Menu::get('MyNavBar')->item('about')->children();
// ...
?>
children()
returns a Laravel Collection.
To check if an item has any children or not, you can use hasChildren()
<?php
// ...
if( $menu->about->hasChildren() ) {
// Do something
}
// or outside of the builder context
Menu::get('MyNavBar')->about->hasChildren();
// Or
Menu::get('MyNavBar')->item('about')->hasChildren();
// ...
?>
To get all descendants of an item you may use all
:
<?php
// ...
$aboutSubs = $menu->about->all();
// ...
You can also search the items collection by magic where methods.
These methods are consisted of a where
concatenated with a property (object property or even meta data)
For example to get an item with parent equal to 12, you can use it like so:
<?php
// ...
$subs = $menu->whereParent(12);
// ...
?>
Or to get item's with a specific meta data:
<?php
// ...
$menu->add('Home', '#')->data('color', 'red');
$menu->add('About', '#')->data('color', 'blue');
$menu->add('Services', '#')->data('color', 'red');
$menu->add('Contact', '#')->data('color', 'green');
// ...
// Fetch all the items with color set to red:
$reds = $menu->whereColor('red');
?>
This method returns a Laravel collection.
If you need to fetch descendants of the matched items as well, Just set the second argument as true.
<?php
$reds = $menu->whereColor('red', true);
This will give all items with color red and their decsendants.
You might encounter situations when you need to refer to menu instances out of the builder context.
To get a specific menu by name:
<?php
// ...
$menu = Menu::get('MyNavBar');
// ...
?>
Or to get all menus instances:
<?php
// ...
$menus = Menu::all();
// ...
?>
You can also call getCollection()
to get the same result:
<?php
// ...
$menus = Menu::getCollection();
// ...
?>
Both methods return a Laravel Collection
Since all menu items would be rendered as HTML entities like list items or divs, you can define as many HTML attributes as you need for each item:
<?php
Menu::make('MyNavBar', function($menu){
// As you see, you need to pass the second parameter as an associative array:
$menu->add('Home', array('route' => 'home.page', 'class' => 'navbar navbar-home', 'id' => 'home'));
$menu->add('About', array('route' => 'page.about', 'class' => 'navbar navbar-about dropdown'));
$menu->add('services', array('action' => 'ServicesController@index'));
$menu->add('Contact', 'contact');
});
?>
If we choose HTML lists as our rendering format like ul
, the result would be something similar to this:
<ul>
<li class="navbar navbar-home" id="home"><a href="http://yourdomain.com">Home</a></li>
<li class="navbar navbar-about dropdown"><a href="http://yourdomain.com/about">About</a></li>
<li><a href="http://yourdomain.com/services">Services</a></li>
<li><a href="http://yourdomain.com/contact">Contact</a></li>
</ul>
It is also possible to set or get HTML attributes after the item has been defined using attr()
method.
If you call attr()
with one argument, it will return the attribute value for you.
If you call it with two arguments, It will consider the first and second parameters as a key/value pair and sets the attribute.
You can also pass an associative array of attributes if you need to add a group of HTML attributes in one step; Lastly if you call it without any arguments it will return all the attributes as an array.
<?php
//...
$menu->add('About', array('url' => 'about', 'class' => 'about-item'));
echo $menu->about->attr('class'); // output: about-item
$menu->about->attr('class', 'another-class');
echo $menu->about->attr('class'); // output: about-item another-class
$menu->about->attr(array('class' => 'yet-another', 'id' => 'about'));
echo $menu->about->attr('class'); // output: about-item another-class yet-another
echo $menu->about->attr('id'); // output: id
print_r($menu->about->attr());
/* Output
Array
(
[class] => about-item another-class yet-another
[id] => id
)
*/
//...
?>
You can use attr
on a collection, if you need to target a group of items:
<?php
// ...
$menu->add('About', 'about');
$menu->about->add('Who we are', 'about/whoweare');
$menu->about->add('What we do', 'about/whatwedo');
// add a class to children of About
$menu->about->children()->attr('class', 'about-item');
// ...
All the HTML attributes will go to the wrapping tags(li, div, etc); You might encounter situations when you need to add some HTML attributes to <a>
tags as well.
Each Item
instance has an attribute which stores an instance of Link
object. This object is provided for you to manipulate <a>
tags.
Just like each item, Link
also has an attr()
method which functions exactly like item's:
<?php
Menu::make('MyNavBar', function($menu){
// ...
$about = $menu->add('About', array('route' => 'page.about', 'class' => 'navbar navbar-about dropdown'));
$about->link->attr(array('class' => 'dropdown-toggle', 'data-toggle' => 'dropdown'));
// ...
});
?>
If you don't want to use the routing feature of laravel-menu
or you don't want the builder to prefix your URL with anything (your host address for example), you can explicitly set your link's href property:
<?php
// ...
$menu->add('About')->link->href('#');
// ...
?>
You can mark an item as activated using active()
on that item:
<?php
// ...
$menu->add('Home', '#')->active();
// ...
/* Output
<li class="active"><a href="#">#</a></li>
*/
?>
You can also add class active
to the anchor element instead of the wrapping element:
<?php
// ...
$menu->add('Home', '#')->link->active();
// ...
/* Output
<li><a class="active" href="#">#</a></li>
*/
?>
Laravel Menu does this for you automatically according to the current URI the time you reigster the item.
You can also choose the element to be activated (item or the link) in options.php
which resides in package's config directory:
// ...
'active_element' => 'item', // item|link
// ...
RESTful URLs are also supported as long as restful
option is set as true
in config/settings.php
file, E.g. item with url resource
will be activated by resource/slug
or resource/slug/edit
.
You might encounter situations where your app is in a directory instead of the root directory or your resources have a common prefix; In such case you need to set rest_base
option to a proper prefix for a better restful activation support. rest_base
can take a simple string, array of string or a function call as value.
You can also able to define a pattern for a certain item, if the automatic activation can't help:
<?php
// ...
$menu->add('Articles', 'articles')->active('this-is-another-url/*');
// ...
So this-is-another-url
, this-is-another-url/and-another
will both activate Articles
item.
You can insert a separator after each item using divide()
method:
<?php
//...
$menu->add('Separated Item', 'item-url')->divide()
// You can also use it this way:
$menu->('Another Separated Item', 'another-item-url');
// This line will insert a divider after the last defined item
$menu->divide()
//...
/*
Output as <ul>:
<ul>
...
<li><a href="item-url">Separated Item</a></li>
<li class="divider"></li>
<li><a href="another-item-url">Another Separated Item</a></li>
<li class="divider"></li>
...
</ul>
*/
?>
divide()
also gets an associative array of attributes:
<?php
//...
$menu->add('Separated Item', 'item-url')->divide( array('class' => 'my-divider') );
//...
/*
Output as <ul>:
<ul>
...
<li><a href="item-url">Separated Item</a></li>
<li class="my-divider divider"></li>
...
</ul>
*/
?>
You can append
or prepend
HTML or plain-text to each item's title after it is defined:
<?php
Menu::make('MyNavBar', function($menu){
// ...
$about = $menu->add('About', array('route' => 'page.about', 'class' => 'navbar navbar-about dropdown'));
$menu->about->attr(array('class' => 'dropdown-toggle', 'data-toggle' => 'dropdown'))
->append(' <b classs="caret"></b>')
->prepend('<span classs="glyphicon glyphicon-user"></span> ');
// ...
});
?>
The above code will result:
<ul>
...
<li class="navbar navbar-about dropdown">
<a href="about" class="dropdown-toggle" data-toggle="dropdown">
<span class="glyphicon glyphicon-user"></span> About <b classs="caret"></b>
</a>
</li>
</ul>
You can call prepend
and append
on collections as well.
To insert items as plain text instead of hyper-links you can use raw()
:
<?php
// ...
$menu->raw('Item Title', array('class' => 'some-class'));
$menu->add('About', 'about');
$menu->About->raw('Another Plain Text Item')
// ...
/* Output as an unordered list:
<ul>
...
<li class="some-class">Item's Title</li>
<li>
About
<ul>
<li>Another Plain Text Item</li>
</ul>
</li>
...
</ul>
*/
?>
Sometimes you may need to share attributes between a group of items. Instead of specifying the attributes and options for each item, you may use a menu group feature:
PS: This feature works exactly like Laravel group routes.
<?php
Menu::make('MyNavBar', function($menu){
$menu->add('Home', array('route' => 'home.page', 'class' => 'navbar navbar-home', 'id' => 'home'));
$menu->group(array('style' => 'padding: 0', 'data-role' => 'navigation') function($m){
$m->add('About', array('route' => 'page.about', 'class' => 'navbar navbar-about dropdown'));
$m->add('services', array('action' => 'ServicesController@index'));
}
$menu->add('Contact', 'contact');
});
?>
Attributes style
and data-role
would be applied to both About
and Services
items:
<ul>
<li class="navbar navbar-home" id="home"><a href="http://yourdomain.com">Home</a></li>
<li style="padding: 0" data-role="navigation" class="navbar navbar-about dropdown"><a href="http://yourdomain.com/about"About</a></li>
<li style="padding: 0" data-role="navigation"><a href="http://yourdomain.com/services">Services</a></li>
</ul>
Just like Laravel route prefixing feature, a group of menu items may be prefixed by using the prefix
option in the array being passed to the group.
Attention: Prefixing only works on the menu items addressed with url
but not route
or action
.
<?php
Menu::make('MyNavBar', function($menu){
$menu->add('Home', array('route' => 'home.page', 'class' => 'navbar navbar-home', 'id' => 'home'));
$menu->add('About', array('url' => 'about', 'class' => 'navbar navbar-about dropdown')); // URL: /about
$menu->group(array('prefix' => 'about'), function($m){
$about->add('Who we are?', 'who-we-are'); // URL: about/who-we-are
$about->add('What we do?', 'what-we-do'); // URL: about/what-we-do
});
$menu->add('Contact', 'contact');
});
?>
This will generate:
<ul>
<li class="navbar navbar-home" id="home"><a href="/">Home</a></li>
<li data-role="navigation" class="navbar navbar-about dropdown"><a href="http://yourdomain.com/about/summary"About</a>
<ul>
<li><a href="http://yourdomain.com/about/who-we-are">Who we are?</a></li>
<li><a href="http://yourdomain.com/about/who-we-are">What we do?</a></li>
</ul>
</li>
<li><a href="services">Services</a></li>
<li><a href="contact">Contact</a></li>
</ul>
Laravel Menu supports nested grouping feature as well. A menu group merges its own attribute with its parent group then shares them between its wrapped items:
<?php
Menu::make('MyNavBar', function($menu){
// ...
$menu->group(array('prefix' => 'pages', 'data-info' => 'test'), function($m){
$m->add('About', 'about');
$m->group(array('prefix' => 'about', 'data-role' => 'navigation'), function($a){
$a->add('Who we are', 'who-we-are?');
$a->add('What we do?', 'what-we-do');
$a->add('Our Goals', 'our-goals');
});
});
});
?>
If we render it as a ul:
<ul>
...
<li data-info="test">
<a href="http://yourdomain.com/pages/about">About</a>
<ul>
<li data-info="test" data-role="navigation"><a href="http://yourdomain.com/pages/about/who-we-are"></a></li>
<li data-info="test" data-role="navigation"><a href="http://yourdomain.com/pages/about/what-we-do"></a></li>
<li data-info="test" data-role="navigation"><a href="http://yourdomain.com/pages/about/our-goals"></a></li>
</ul>
</li>
</ul>
You might encounter situations when you need to attach some meta data to each item; This data can be anything from item placement order to permissions required for accessing the item; You can do this by using data()
method.
data()
method works exactly like attr()
method:
If you call data()
with one argument, it will return the data value for you.
If you call it with two arguments, It will consider the first and second parameters as a key/value pair and sets the data.
You can also pass an associative array of data if you need to add a group of key/value pairs in one step; Lastly if you call it without any arguments it will return all data as an array.
<?php
Menu::make('MyNavBar', function($menu){
// ...
$menu->add('Users', array('route' => 'admin.users'))
->data('permission', 'manage_users');
});
?>
You can also access a data as if it's a property:
<?php
//...
$menu->add('Users', '#')->data('placement', 12);
// you can refer to placement as if it's a public property of the item object
echo $menu->users->placement; // Output : 12
//...
?>
Meta data don't do anything to the item and won't be rendered in HTML either. It is the developer who would decide what to do with them.
You can use data
on a collection, if you need to target a group of items:
<?php
// ...
$menu->add('Users', 'users');
$menu->users->add('New User', 'users/new');
$menu->users->add('Uses', 'users');
// add a meta data to children of Users
$menu->users->children()->data('anything', 'value');
// ...
We can filter menu items by a using filter()
method.
Filter()
receives a closure which is defined by you.It then iterates over the items and run your closure on each of them.
You must return false for items you want to exclude and true for those you want to keep.
Let's proceed with a real world scenario:
I suppose your User
model can check whether the user has an specific permission or not:
<?php
Menu::make('MyNavBar', function($menu){
// ...
$menu->add('Users', array('route' => 'admin.users'))
->data('permission', 'manage_users');
})->filter(function($item){
if(User::get()->can( $item->data('permission'))) {
return true;
}
return false;
});
?>
As you might have noticed we attached the required permission for each item using data()
.
As result, Users
item will be visible to those who has the manage_users
permission.
laravel-menu
can sort the items based on either a user defined function or a key which can be item properties like id,parent,etc or meta data stored with each item.
To sort the items based on a property and or meta data:
<?php
Menu::make('main', function($m){
$m->add('About', '#') ->data('order', 2);
$m->add('Home', '#') ->data('order', 1);
$m->add('Services', '#') ->data('order', 3);
$m->add('Contact', '#') ->data('order', 5);
$m->add('Portfolio', '#') ->data('order', 4);
})->sortBy('order');
?>
sortBy()
also receives a second parameter which specifies the ordering direction: Ascending order(asc
) and Descending Order(dsc
).
Default value is asc
.
To sort the items based on Id
in descending order:
<?php
Menu::make('main', function($m){
$m->add('About');
$m->add('Home');
$m->add('Services');
$m->add('Contact');
$m->add('Portfolio');
})->sortBy('id', 'desc');
?>
Sorting the items by passing a closure:
<?php
Menu::make('main', function($m){
$m->add('About') ->data('order', 2);
$m->add('Home') ->data('order', 1);
$m->add('Services') ->data('order', 3);
$m->add('Contact') ->data('order', 5);
$m->add('Portfolio') ->data('order', 4);
})->sortBy(function($items) {
// Your sorting algorithm here...
});
?>
The closure takes the items collection as argument.
Several rendering formats are available out of the box:
{{ $MenuName->asUl() }}
asUl()
will render your menu in an unordered list. it also takes an optional parameter to define attributes for the <ul>
tag itself:
{{ $MenuName->asUl( array('class' => 'awesome-ul') ) }}
Result:
<ul class="awesome-ul">
<li><a href="http://yourdomain.com">Home</a></li>
<li><a href="http://yourdomain.com/about">About</a></li>
<li><a href="http://yourdomain.com/services">Services</a></li>
<li><a href="http://yourdomain.com/contact">Contact</a></li>
</ul>
{{ $MenuName->asOl() }}
asOl()
method will render your menu in an ordered list. it also takes an optional parameter to define attributes for the <ol>
tag itself:
{{ $MenuName->asOl( array('class' => 'awesome-ol') ) }}
Result:
<ol class="awesome-ol">
<li><a href="http://yourdomain.com">Home</a></li>
<li><a href="http://yourdomain.com/about">About</a></li>
<li><a href="http://yourdomain.com/services">Services</a></li>
<li><a href="http://yourdomain.com/contact">Contact</a></li>
</ol>
{{ $MenuName->asDiv() }}
asDiv()
method will render your menu as nested HTML divs. it also takes an optional parameter to define attributes for the parent <div>
tag itself:
{{ $MenuName->asDiv( array('class' => 'awesome-div') ) }}
Result:
<div class="awesome-div">
<div><a href="http://yourdomain.com">Home</a></div>
<div><a href="http://yourdomain.com/about">About</a></div>
<div><a href="http://yourdomain.com/services">Services</a></div>
<div><a href="http://yourdomain.com/contact">Contact</a></div>
</div>
Laravel Menu provides a parital view out of the box which generates menu items in a bootstrap friendly style which you can include in your Bootstrap based navigation bars:
You can access the partial view via Config::get('laravel-menu::views.bootstrap-items')
.
All you need to do is to include the partial view and pass the root level items to it:
{{{...}}}
@include(Config::get('laravel-menu::views.bootstrap-items'), array('items' => $mainNav->roots()))
{{{...}}}
This is how your Bootstrap code is going to look like:
<nav class="navbar navbar-default" role="navigation">
<div class="container-fluid">
<!-- Brand and toggle get grouped for better mobile display -->
<div class="navbar-header">
<button type="button" class="navbar-toggle" data-toggle="collapse" data-target="#bs-example-navbar-collapse-1">
<span class="sr-only">Toggle navigation</span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
</button>
<a class="navbar-brand" href="#">Brand</a>
</div>
<!-- Collect the nav links, forms, and other content for toggling -->
<div class="collapse navbar-collapse" id="bs-example-navbar-collapse-1">
<ul class="nav navbar-nav">
@include(Config::get('laravel-menu::views.bootstrap-items'), array('items' => $mainNav->roots()))
</ul>
<form class="navbar-form navbar-right" role="search">
<div class="form-group">
<input type="text" class="form-control" placeholder="Search">
</div>
<button type="submit" class="btn btn-default">Submit</button>
</form>
<ul class="nav navbar-nav navbar-right">
@include(Config::get('laravel-menu::views.bootstrap-items'), array('items' => $loginNav->roots()))
</ul>
</div><!-- /.navbar-collapse -->
</div><!-- /.container-fluid -->
</nav>
As noted earlier you can create your own rendering formats.
If you'd like to render your menu(s) according to your own design, you should create two views.
View-1
This view contains all the HTML codes likenav
orul
ordiv
tags wrapping your menu items.View-2
This view is actually a partial view responsible for rendering menu items (it is going to be included inView-1
.)
The reason we use two view files here is that View-2
calls itself recursively to render the items to the deepest level required in multi-level menus.
Let's make this easier with an example:
In our routes.php
:
<?php
Menu::make('MyNavBar', function($menu){
$menu->add('Home');
$menu->add('About', array('route' => 'page.about'));
$menu->about->add('Who are we?', 'who-we-are');
$menu->about->add('What we do?', 'what-we-do');
$menu->add('services', 'services');
$menu->add('Contact', 'contact');
});
?>
In this example we name View-1 custom-menu.blade.php
and View-2 custom-menu-items.blade.php
.
custom-menu.blade.php
<nav class="navbar">
<ul class="horizontal-navbar">
@include('custom-menu-items', array('items', $MyNavBar->roots()))
</ul>
</nav><!--/nav-->
custom-menu-items.blade.php
@foreach($items as $item)
<li @if($item->hasChildren()) class="dropdown" @endif>
<a href="{{ $item->url() }}">{{ $item->title }} </a>
@if($item->hasChildren())
<ul class="dropdown-menu">
@include('custom-menu-items', array('items' => $item->children()))
</ul>
@endif
</li>
@endforeach
Let's describe what we did above, In custom-menus.blade.php
we put whatever HTML boilerplate code we had according to our design, then we included custom-menu-items.blade.php
and passed the menu items at root level to custom-menu-items.blade.php
:
...
@include('custom-menu-items', array('items' => $menu->roots()))
...
In custom-menu-items.blade.php
we ran a foreach
loop and called the file recursively in case the current item had any children.
To put the rendered menu in your application template, you can simply include custom-menu
view in your master layout.
Laravel menu extends Blade to handle special layouts.
You might encounter situations when some of your HTML properties are explicitly written inside your view instead of dynamically being defined when adding the item; However you will need to merge these static attributes with your Item's attributes.
@foreach($items as $item)
<li @if($item->hasChildren()) class="dropdown" @endif data-test="test">
<a href="{{ $item->url }}">{{ $item->title }} </a>
@if($item->hasChildren())
<ul class="dropdown-menu">
@include('custom-menu-items', array('items' => $item->children()))
</ul>
@endif
</li>
@endforeach
In the above snippet the li
tag has class dropdown
and data-test
property explicitly defined in the view; Laravel Menu provides a control structure which takes care of this.
Suppose the item has also several attributes dynamically defined when being added:
<?php
// ...
$menu->add('Dropdown', array('class' => 'item item-1', 'id' => 'my-item'));
// ...
The view:
@foreach($items as $item)
<li@lm-attrs($item) @if($item->hasChildren()) class="dropdown" @endif data-test="test" @lm-endattrs>
<a href="{{ $item->url }}">{{ $item->title }} </a>
@if($item->hasChildren())
<ul class="dropdown-menu">
@include('custom-menu-items', array('items' => $item->children()))
</ul>
@endif
</li>
@endforeach
This control structure automatically merges the static HTML properties with the dynamically defined properties.
Here's the result:
...
<li class="item item-1 dropdown" id="my-item" data-test="test">...</li>
...
You can adjust the behavior of the menu builder in config/settings.php
file. Currently it provide a few options out of the box:
- auto_activate Automatically activates menu items based on the current URI
- activate_parents Activates the parents of an active item
- active_class Default CSS class name for active items
- restful Activates RESTful URLS. E.g
resource/slug
will activate item withresource
url. - cascade_data If you need descendants of an item to inherit meta data from their parents, make sure this option is enabled.
- rest_base The base URL that all restful resources might be prefixed with.
- active_element You can choose the HTML element to which you want to add activation classes (anchor or the wrapping element).
You're also able to override the default settings for each menu. To override settings for menu, just add the lower-cased menu name as a key in the settings array and add the options you need to override:
<?php
return array(
'default' => array(
'auto_activate' => true,
'activate_parents' => true,
'active_class' => 'active',
'active_element' => 'item', // item|link
'restful' => true,
),
'yourmenuname' => array(
'auto_activate' => false
),
);
Please submit all issues and questions using GitHub issues and I will try to help you.
Laravel-Menu is free software distributed under the terms of the MIT license.