saakiiib/laravel-spa

Zero-reload SPA engine for Laravel Blade — no jQuery, no Livewire, no Inertia.

Maintainers

Package info

github.com/saakiiib/laravel-spa

pkg:composer/saakiiib/laravel-spa

Statistics

Installs: 1

Dependents: 0

Suggesters: 0

Stars: 30

Open Issues: 0

v1.0.0 2026-06-29 07:02 UTC

This package is auto-updated.

Last update: 2026-06-29 07:11:25 UTC


README

By default, every link click in Laravel loads a full new page — HTML, CSS, JS, fonts, everything re-downloads. SPA navigation skips that. Only the content changes, the rest stays. Faster, smoother, no white flash between pages.

Most Laravel SPA solutions force you to either abandon Blade entirely (Inertia) or add jQuery (pjax). This package adds that same speed on top of your existing Blade views — no rewrite, nothing changes except how fast it feels.

Quick Start

composer require saakiiib/laravel-spa
php artisan vendor:publish --tag=spa-assets

Add /public/vendor to your .gitignore to avoid committing published assets.

Updating? When you run composer update, the published spa-engine.js in public/vendor/ does not auto-update — Composer only updates the package source, not previously published assets. Re-publish with the --force flag to pull the latest fix/version in:

php artisan vendor:publish --tag=spa-assets --force

Skipping this means you'll keep running an old spa-engine.js even after upgrading the package version.

Setup — @extends / @section

1. Layout file

Add @spaContent to your content wrapper and @spaEngine before </body>. Make sure @yield('script') comes after @spaEngine.

<main @spaContent>
    @yield('content')
</main>

@spaEngine
@yield('script')

2. Navigation links

Add @spa to links you want SPA navigation on. Links without @spa do a normal full reload.

<a href="{{ route('home') }}" @spa>Home</a>
<a href="{{ route('about') }}" @spa>About</a>
<a href="{{ route('logout') }}">Logout</a>

3. Controller

public function home()
{
    return spa('pages.home');
}

public function about()
{
    return spa('pages.about', compact('data'));
}

4. Page views — no changes needed

@extends('layouts.app')

@section('title', 'Home')

@section('style')
    <style>
        .hero { background: #1a3c6e; color: #fff; padding: 60px; }
    </style>
@endsection

@section('content')
    <div class="hero">
        <h1>Welcome</h1>
    </div>
@endsection

@section('script')
    <script>
        console.log('page loaded');
    </script>
@endsection

Setup — x-layout components

1. Layout component

Add @spaContent to your content wrapper and @spaEngine before </body>.

<head>
    <style>
        /* global styles */
    </style>

    {{ $style ?? '' }}
</head>
<body>
    <main @spaContent>
        {{ $slot }}
    </main>

    <script>
        /* global scripts */
    </script>

    @spaEngine
    {{ $script ?? '' }}
</body>

2. Navigation links — same as above, add @spa:

<a href="{{ route('home') }}" @spa>Home</a>
<a href="{{ route('logout') }}">Logout</a>

3. Controller — same as above:

public function home()
{
    return spa('pages.home');
}

4. Page views

<x-app-layout title="Home">

    <x-slot:style>
        <style>
            .hero { background: #1a3c6e; color: #fff; padding: 60px; }
        </style>
    </x-slot:style>

    <div class="hero">
        <h1>Welcome</h1>
    </div>

    <x-slot:script>
        <script>
            console.log('page loaded');
        </script>
    </x-slot:script>

</x-app-layout>

Global JavaScript & Third-Party Plugins

Page-specific scripts in @section('script') run automatically on every navigation — no extra setup needed.

For global scripts that need to run on every page, place your JS file after @spaEngine:

@spaEngine
<script src="{{ asset('js/app.js') }}"></script>
@yield('script')

In app.js, use spa:loaded to re-initialize plugins after each navigation. Here's an example with Select2:

document.addEventListener('spa:loaded', function () {
    $('.select2').select2();
});

For event listeners, attach them to document once — they work on every page automatically:

document.addEventListener('click', function (e) {
    if (e.target.matches('.delete-btn')) {
        // works on every page
    }
});

What works out of the box

  • URL updates, back/forward button, refresh, direct links — all work
  • Per-page styles and scripts load and unload on every navigation
  • Session expiry redirects cleanly instead of breaking
  • Hover prefetch — pages start loading before you even click
  • Works with both @extends and x-layout
  • spa:loaded event fires after every navigation for global plugin re-initialization

Requirements

  • PHP 8.1+
  • Laravel 10, 11, 12, or 13

Contributing

Found a bug or want to improve something? PRs are welcome on GitHub.

License

MIT — Sakib