laravel / wayfinder
Generate TypeScript representations of your Laravel actions and routes.
Installs: 122
Dependents: 0
Suggesters: 0
Security: 0
Stars: 450
Watchers: 13
Forks: 11
Open Issues: 1
Requires
- php: ^8.3
- illuminate/console: ^11.0|^12.0
- illuminate/filesystem: ^11.0|^12.0
- illuminate/routing: ^11.0|^12.0
- illuminate/support: ^11.0|^12.0
Requires (Dev)
- laravel/pint: ^1.21
- orchestra/testbench: ^10.1
This package is auto-updated.
Last update: 2025-04-03 16:09:11 UTC
README
Introduction
Laravel Wayfinder bridges your Laravel backend and TypeScript frontend with zero friction. It automatically generates fully-typed, importable TypeScript functions for your controllers and routes — so you can call your Laravel endpoints directly in your client code just like any other function. No more hardcoding URLs, guessing route parameters, or syncing backend changes manually.
Note: Wayfinder is currently in Beta, the API is subject to change prior to the v1.0.0 release. All notable changes will be documented in the changelog.
Installation
To get started, install Wayfinder via the Composer package manager:
composer require laravel/wayfinder
If you would like to automatically watch your files for changes, you should also install the vite-plugin-run
npm package:
npm i -D vite-plugin-run
Then, update your application's vite.config.js
file to watch for changes to your application's routes and controllers:
import { run } from "vite-plugin-run"; export default defineConfig({ plugins: [ // ... run([ { name: "wayfinder", run: ["php", "artisan", "wayfinder:generate"], pattern: ["routes/*.php", "app/**/Http/**/*.php"], }, ]), ], });
For convenience, you may also wish to register aliases for importing the generated files into your application:
export default defineConfig({ // ... resolve: { alias: { "@actions/": "./resources/js/actions", "@routes/": "./resources/js/routes", }, }, });
Generating TypeScript Definitions
The wayfinder:generate
command can be used to generate TypeScript definitions for your routes and controller methods:
php artisan wayfinder:generate
By default, Wayfinder generates files in three directories (wayfinder
, actions
, and routes
) within resources/js
, but you can configure the base path:
php artisan wayfinder:generate --path=resources/js/wayfinder
The --skip-actions
and --skip-routes
options may be used to skip TypeScript definition generation for controller methods or routes, respectively:
php artisan wayfinder:generate --skip-actions
php artisan wayfinder:generate --skip-routes
You can safely .gitignore
the wayfinder
, actions
, and routes
directories as they are completely re-generated on every build.
Usage
Wayfinder functions return an object that contains the resolved URL and default HTTP method:
import { show } from "@actions/App/Http/Controllers/PostController"; show(1); // { url: "/posts/1", method: "get" }
If you just need the URL, or would like to choose a method from the HTTP methods defined on the server, you can invoke additional methods on the Wayfinder generated function:
import { show } from "@actions/App/Http/Controllers/PostController"; show.url(1); // "/posts/1" show.head(1); // { url: "/posts/1", method: "head" }
Wayfinder functions accept a variety of shapes for their arguments:
import { show, update } from "@actions/App/Http/Controllers/PostController"; // Single parameter action... show(1); show({ id: 1 }); // Multiple parameter action... update([1, 2]); update({ post: 1, author: 2 }); update({ post: { id: 1 }, author: { id: 2 } });
Note: If you have a delete
method on your controller, Wayfinder will rename it to deleteMethod
when generating its functions. This is because delete
is not allowed as a variable declaration.
If you've specified a key for the parameter binding, Wayfinder will detect this and allow you to pass the value in as a property on an object:
import { show } from "@actions/App/Http/Controllers/PostController"; // Route is /posts/{post:slug}... show("my-new-post"); show({ slug: "my-new-post" });
Invokable Controllers
If your controller is an invokable controller, you may simple invoke the imported Wayfinder function directly:
import StorePostController from "@actions/App/Http/Controllers/StorePostController"; StorePostController();
Importing Controllers
You may also import the Wayfinder generated controller definition and invoke its individual methods on the imported object:
import PostController from "@actions/App/Http/Controllers/PostController"; PostController.show(1);
Note: In the example above, importing the entire controller prevents the PostController
from being tree-shaken, so all PostController
actions will be included in your final bundle.
Importing Named Routes
Wayfinder can also generate methods for your application's named routes as well:
import { show } from "@routes/post"; // Named route is `post.show`... show(1); // { url: "/posts/1", method: "get" }
Conventional Forms
If your application uses conventional HTML form submissions, Wayfinder can help you out there as well. First, opt into form variants when generating your TypeScript definitions:
php artisan wayfinder:generate --with-form
Then, you can use the .form
variant to generate <form>
object attributes automatically:
import { store, update } from "@actions/App/Http/Controllers/PostController"; const Page = () => ( <form {...store.form()}> {/* <form action="/posts" method="post"> */} {/* ... */} </form> ); const Page = () => ( <form {...update.form(1)}> {/* <form action="/posts/1?_method=PATCH" method="post"> */} {/* ... */} </form> );
If your form action supports multiple methods and would like to specify a method, you can invoke additional methods on the form
:
import { store, update } from "@actions/App/Http/Controllers/PostController"; const Page = () => ( <form {...update.form.put(1)}> {/* <form action="/posts/1?_method=PUT" method="post"> */} {/* ... */} </form> );
Query Parameters
All Wayfinder methods accept an optional, final options
argument to which you may pass a query
object. This object can be used to append query parameters onto the resulting URL:
import { show } from "@actions/App/Http/Controllers/PostController"; const options = { query: { page: 1, sort_by: "name", }, }; show(1, options); // { url: "/posts/1?page=1&sort_by=name", method: "get" } show.get(1, options); // { url: "/posts/1?page=1&sort_by=name", method: "get" } show.url(1, options); // "/posts/1?page=1&sort_by=name" show.form.head(1, options); // { action: "/posts/1?page=1&sort_by=name&_method=HEAD", method: "get" }
You can also merge with the URL's existing parameters by passing a mergeQuery
object instead:
import { show } from "@actions/App/Http/Controllers/PostController"; // window.location.search = "?page=1&sort_by=category&q=shirt" const options = { mergeQuery: { page: 2, sort_by: "name", }, }; show.url(1, options); // "/posts/1?page=2&sort_by=name&q=shirt"
If you would like to remove a parameter from the resulting URL, define the value as null
or undefined
:
import { show } from "@actions/App/Http/Controllers/PostController"; // window.location.search = "?page=1&sort_by=category&q=shirt" const options = { mergeQuery: { page: 2, sort_by: null, }, }; show.url(1, options); // "/posts/1?page=2&q=shirt"
Wayfinder and Inertia
When using Inertia, you can pass the result of a Wayfinder method directly to the submit
method of useForm
, it will automatically resolve the correct URL and method:
import { useForm } from "@inertiajs/react"; import { store } from "@actions/App/Http/Controllers/PostController"; const form = useForm({ name: "My Big Post", }); form.submit(store()); // Will POST to `/posts`...
You may also use Wayfinder in conjunction with Inertia's Link
component:
import { Link } from "@inertiajs/react"; import { show } from "@actions/App/Http/Controllers/PostController"; const Nav = () => <Link href={show(1)}>Show me the first post</Link>;