<div>
<b-nav>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Another Link</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Overview
The base <b-nav>
component is built with flexbox and provides a strong foundation for building all types of navigation components. It includes some style overrides (for working with lists), some link padding for larger hit areas, and basic disabled styling. No active states are included in the base nav.
<b-nav>
supports the following child components:
<b-nav-item>
for actionable links (or router-links) <b-nav-item-dropdown>
for dropdowns <b-nav-text>
for plain text content <b-nav-form>
for inline forms
Link appearance
Two style variations are supported: tabs
and pills
, which support active
state styling. These variants are mutually exclusive - use only one style or the other.
Tab style
Make the nav look like tabs by setting the tabs
prop.
<div>
<b-nav tabs>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Another Link</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Pill style
Use the pill style by setting the pills
prop.
<div>
<b-nav pills>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Another Link</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Small
Make the nav smaller by setting the small
prop.
<div>
<b-nav small>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Another Link</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Fill and justify
Force your <b-nav>
content to extend the full available width.
Fill
To proportionately fill all available space with your <b-nav-item>
components, set the fill
prop. Notice that all horizontal space is occupied, but not every nav item has the same width.
<div>
<b-nav tabs fill>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Link with a long name </b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Justified
For equal-width elements, set the justified
prop instead. All horizontal space will be occupied by nav links, but unlike fill
above, every <b-nav-item>
will be the same width.
<div>
<b-nav tabs justified>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Link with a long name </b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Alignment
To align your <b-nav-item>
components, use the align
prop. Available values are left
, center
and right
.
<div>
<b-nav tabs align="center">
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Link with a long name </b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Vertical variation
By default <b-nav>
appear on a horizontal line. Stack your navigation by setting the vertical
prop.
<div>
<b-nav vertical class="w-25">
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item>Another Link</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</div>
Dropdown support
Use <b-nav-item-dropdown>
to place dropdown items within your nav.
<div>
<b-nav pills>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Link</b-nav-item>
<b-nav-item-dropdown
id="my-nav-dropdown"
text="Dropdown"
toggle-class="nav-link-custom"
right
>
<b-dropdown-item>One</b-dropdown-item>
<b-dropdown-item>Two</b-dropdown-item>
<b-dropdown-divider></b-dropdown-divider>
<b-dropdown-item>Three</b-dropdown-item>
</b-nav-item-dropdown>
</b-nav>
</div>
Sometimes you want to add your own class names to the generated dropdown toggle button, that by default have the classes nav-link
and dropdown-toggle
. Use the toggle-class
prop to add them (like above) which will render HTML similar to:
<li id="my-nav-dropdown" class="nav-item b-nav-dropdown dropdown">
<a
role="button"
href="#my-nav-dropdown"
id="my-nav-dropdown__BV_button_"
aria-haspopup="true"
aria-expanded="false"
class="nav-link dropdown-toggle nav-link-custom"
></a>
...
</li>
Refer to <b-dropdown>
for a list of supported sub-components.
Optionally scoped default slot
The dropdown default slot is optionally scoped with the following scope available:
Property or Method | Description |
hide() | Can be used to close the dropdown menu. Accepts an optional boolean argument, which if true returns focus to the toggle button |
Lazy dropdown
By default, <b-nav-item-dropdown>
renders the menu contents in the DOM even when the menu is not shown. When there are a large number of dropdowns rendered on the same page, performance could be impacted due to larger overall memory utilization. You can instruct <b-nav-item-dropdown>
to render the menu contents only when it is shown by setting the lazy
prop to true.
Dropdown placement
Use the dropdown props right
, dropup
, dropright
, dropleft
, no-flip
, and offset
to control the positioning of <b-nav-item-dropdown>
.
Refer to the <b-dropdown>
positioning section for details on the effects and usage of these props.
Dropdown implementation note
Note that the toggle button is actually rendered as a link <a>
tag with role="button"
for styling purposes, and typically has the href
set to #
unless an ID is provided via the id
prop.
The toggle will prevent scroll-top-top behaviour (via JavaScript) when clicking the toggle link. In some cases when using SSR, and the user clicks the toggle button before Vue has had a chance to hydrate the component, the page will scroll to top. In these cases, simply providing a unique ID via the id
prop will prevent the unwanted scroll-to-top behaviour.
Nav text content
Use the <b-nav-text>
child component to place plain text content into the nav:
<div>
<b-nav >
<b-nav-item href="#1">Link 1</b-nav-item>
<b-nav-item href="#2">Link 2</b-nav-item>
<b-nav-text>Plain text</b-nav-text>
</b-nav>
</div>
Use the <b-nav-form>
child component to place an inline form into the nav:
<div>
<b-nav pills>
<b-nav-item href="#1" active>Link 1</b-nav-item>
<b-nav-item href="#2">Link 2</b-nav-item>
<b-nav-form @submit.stop.prevent="alert('Form Submitted')">
<b-form-input aria-label="Input" class="mr-1"></b-form-input>
<b-button type="submit">Ok</b-button>
</b-nav-form>
</b-nav>
</div>
Refer to the <b-form>
inline documentation for additional details on placing form controls.
Tabbed local content support
See the <b-tabs>
component for creating tabbable panes of local content (not suited for navigation).
Card integration
Use a <b-nav>
in a <b-card>
header, by enabling the card-header
prop on <b-nav>
and setting either the pills
or tabs
props:
Tabs style:
<div>
<b-card title="Card Title" no-body>
<b-card-header header-tag="nav">
<b-nav card-header tabs>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Inactive</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</b-card-header>
<b-card-body class="text-center">
<b-card-text>
With supporting text below as a natural lead-in to additional content.
</b-card-text>
<b-button variant="primary">Go somewhere</b-button>
</b-card-body>
</b-card>
</div>
Pill style:
<div>
<b-card title="Card Title" no-body>
<b-card-header header-tag="nav">
<b-nav card-header pills>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Inactive</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</b-card-header>
<b-card-body class="text-center">
<b-card-text>
With supporting text below as a natural lead-in to additional content.
</b-card-text>
<b-button variant="primary">Go somewhere</b-button>
</b-card-body>
</b-card>
</div>
Plain style:
The card-header
prop is only needed when you are applying tabs
or pills
style. Note that Bootstrap v4 SCSS does not have special styling for active
state plain style nav items.
<div>
<b-card title="Card Title" no-body>
<b-card-header header-tag="nav">
<b-nav>
<b-nav-item active>Active</b-nav-item>
<b-nav-item>Inactive</b-nav-item>
<b-nav-item disabled>Disabled</b-nav-item>
</b-nav>
</b-card-header>
<b-card-body class="text-center">
<b-card-text>
With supporting text below as a natural lead-in to additional content.
</b-card-text>
<b-button variant="primary">Go somewhere</b-button>
</b-card-body>
</b-card>
</div>
The card-header
prop has no styling effect if the <b-nav>
is in vertical
mode.
Using with Vue Router
Have your card <b-nav>
control vue router nested routes via <router-view>
or <nuxt-child>
components, to created tabbed content that changes with route URL:
// On page with route `/some/route`
<div>
<b-card title="Card Title" no-body>
<b-card-header header-tag="nav">
<b-nav card-header tabs>
<b-nav-item to="/some/route/" exact exact-active-class="active">Active</b-nav-item>
<b-nav-item to="/some/route/foo" exact exact-active-class="active">Foo</b-nav-item>
<b-nav-item to="/some/route/bar" exact exact-active-class="active">Bar</b-nav-item>
</b-nav>
</b-card-header>
<b-card-body>
<router-view></router-view>
</b-card-body>
</b-card>
</div>
Note: Vue Router does not support defining active routes with hashes (#
), which is why you must define the "tab" content as child routes.
Example router config for above:
const routes = [
{
path: '/some/route',
component: SomeRouteComponent,
children: [
{ path: '', component: DefaultTabComponent, name: 'some-route' },
{ path: 'foo', component: FooTabComponent },
{ path: 'bar', component: BarTabComponent }
]
}
]
One can also use Vue Router named routes and/or route params instead of path based routes.
For more details see:
Accessibility
If you're using <b-nav>
to provide a navigation bar, be sure to add a role="navigation"
to the most logical parent container of <b-nav>
, or wrap a <nav>
element around <b-nav>
. Do not add the role to the <b-nav>
itself, as this would prevent it from being announced as an actual list by assistive technologies.
When using a <b-nav-item-dropdown>
in your <b-nav>
, be sure to assign a unique id
prop value to the <b-nav-item-dropdown>
so that the appropriate aria-*
attributes can be automatically generated.
Tabbed interface accessibility
Note that navigation bars, even if visually styled as tabs, should not be given role="tablist"
, role="tab"
or role="tabpanel"
attributes. These are only appropriate for tabbed interfaces that do not change the URL or $route
, as described in the WAI ARIA Authoring Practices. See <b-tabs>
for dynamic tabbed interfaces that are compliant with WAI ARIA.
Tabbed interfaces should avoid using dropdown menus, as this causes both usability and accessibility issues:
- From a usability perspective, the fact that the currently displayed tab’s trigger element is not immediately visible (as it’s inside the closed dropdown menu) can cause confusion.
- From an accessibility point of view, there is currently no sensible way to map this sort of construct to a standard WAI ARIA pattern, meaning that it cannot be easily made understandable to users of assistive technologies.
See also
<b-tabs>
to create tabbable panes of local content, even via dropdown menus. <b-navbar>
a wrapper that positions branding, navigation, and other elements in a concise header. <b-dropdown>
for sub-components that you can place inside <b-nav-item-dropdown>
- Router Link Support reference for information about router-link specific props available on
<b-nav-item>