Sometimes, your business may need additional functionality that isn't provided by Nova. For this reason, Nova allows you to build custom tools and add them to the Nova sidebar. Nova tools are incredibly customizable, as they primarily consist of a single-file Vue component that is totally under your control. Within your Vue component, you are free to make HTTP requests to any controller within your application.
Custom tools may be generated using the nova:tool
Artisan command. By default, all new tools will be placed in the nova-components
directory of your application. When generating a tool using the nova:tool
command, the tool name you pass to the command should follow the Composer vendor/package
format. So, if we were building a price tracker tool, we might run the following command to generate our tool:
php artisan nova:tool acme/price-tracker
When generating a tool, Nova will prompt you to install the tool's NPM dependencies, compile its assets, and update your application's composer.json
file. All custom tools are registered with your application as a Composer "path" repository.
Nova tools include all of the scaffolding necessary to build your tool. Each tool even contains its own composer.json
file and is ready to be shared with the world on GitHub or the source control provider of your choice.
Nova tools may be registered in your application's App/Providers/NovaServiceProvider
class. Your service provider contains a tools
method, which returns an array of tools. To register your tool, simply add it to the list of tools returned by this method. For example, if you created a Nova tool named acme/price-tracker
, you may register the tool like so:
use Acme\PriceTracker\PriceTracker;
/**
* Get the tools that should be listed in the Nova sidebar.
*
* @return array
*/
public function tools()
{
return [
new PriceTracker,
];
}
If you would like to only expose a given tool to certain users, you may chain the canSee
method onto your tool's registration. The canSee
method accepts a Closure which should return true
or false
. The Closure will receive the incoming HTTP request:
use Acme\PriceTracker\PriceTracker;
/**
* Get the tools that should be listed in the Nova sidebar.
*
* @return array
*/
public function tools()
{
return [
(new PriceTracker)->canSee(function ($request) {
return false;
}),
];
}
Each tool generated by Nova includes its own service provider and "tool" class. Using the price-tracker
tool as an example, the tool class will be located at src/PriceTracker.php
. The tool class must be registered with your application's NovaServiceProvider
as previously noted.
The tool's service provider is also located within the src
directory of the tool, and is registered within the extra
section of your tool's composer.json
file so that it will be auto-loaded by Laravel.
Often, you will need to define Laravel routes that are called by your tool. When Nova generates your tool, it creates routes/inertia.php
and routes/api.php
route files.
The routes/inertia.php
file is tasked with rendering your tool via Inertia, while the routes/api.php
file may be used to define any routes that your Inertia based tool will be making requests to in order to gather additional data or perform additional tasks.
All routes within the routes/api.php
file are automatically defined inside a route group by your tool's ToolServiceProvider
. The route group specifies that all "API routes", which will typically be invoked from the client via Nova.request, should receive a /nova-vendor/tool-name
URL prefix, where tool-name
is the "kebab-case" name of your tool.
Similarly, routes within the routes/inertia.php
file are also placed within a route group that prefixes all of the routes within the file with the name of your tool.
You are free to modify this route group definition, but you should ensure your Nova tool will easily co-exist with other Nova packages.
Your Nova tool is generated with an Authorize
middleware. You should not typically need to modify this middleware, as it automatically determines whether the authenticated user can "see" the tool before it processes any requests to routes within your tool's route group; however, you are free to modify this middleware if needed.
Your Nova tool class contains a menu
method. This method should return a custom menu that renders your tool's left-side navigation links. You are free to customize this method as needed:
use Illuminate\Http\Request;
use Laravel\Nova\Menu\MenuSection;
/**
* Build the menu that renders the navigation links for the tool.
*
* @param \Illuminate\Http\Request $request
* @return mixed
*/
public function menu(Request $request)
{
return MenuSection::make('Price Tracker')
->path('/price-tracker')
->icon('server');
}
Customized Main Menu
If you have customized Nova's main sidebar menu, a link to your tool will not automatically display in Nova's sidebar. You will need to manually define your tool's menu inside your custom Nova::mainMenu
callback.
Nova utilizes the free Heroicons icon set by Steve Schoger. Therefore, you may simply specify the name of one of these icons when providing the icon name to the icon
method.
When Nova generates your tool, resources/js
and resources/css
directories are generated for you. These directories contain your tool's JavaScript and CSS. The primary files of interest in these directories are: resources/js/components/Tool.vue
and resources/css/tool.css
.
The Tool.vue
file is a single-file Vue component that contains your tool's front-end. From this file, you are free to build your tool however you want. Your tool can make HTTP requests using Axios via Nova.request.
Your Nova tool class contains a boot
method. This method is executed when the tool is registered and available. By default, this method registers your tool's compiled assets so that they will be available to the Nova front-end:
use Laravel\Nova\Nova;
use Laravel\Nova\Events\ServingNova;
/**
* Perform any tasks that need to happen on tool registration.
*
* @return void
*/
public function boot()
{
Nova::script('price-tracker', __DIR__.'/../dist/js/tool.js');
Nova::style('price-tracker', __DIR__.'/../dist/css/tool.css');
}
Your component is bootstrapped and Inertia.js components are registered in the resources/js/tool.js
file. You are free to modify this file or register additional components here as needed:
Nova.booting((Vue, store) => {
Vue.component("PriceTrackerHeader", require("./components/Header").default);
});
Your Nova tool contains a webpack.mix.js
file, which is generated when Nova creates your tool. You may build your tool using the NPM dev
and prod
commands:
# Compile your assets for local development...
npm run dev
# Compile and minify your assets...
npm run prod
In addition, you may run the NPM watch
command to auto-compile your assets when they are changed:
npm run watch
Vue page components contained by your tool have access to all of the components and plugins registered by Nova, including v-tooltip
. For example, your tool's resources/js/pages/Tool.vue
stub will contain a default page title which is managed by the Inertia.js Head
component:
<template>
<Head title="PriceTracker" />
</template>