Find this useful? Enter your email to receive occasional updates for securing PHP code.

Signing you up...

Thank you for signing up!

PHP Decode

<?php namespace Livewire\Features\SupportNavigate; use Laravel\Dusk\Browser; use Livewir..

Decoded Output download

<?php

namespace Livewire\Features\SupportNavigate;

use Laravel\Dusk\Browser;
use Livewire\Attributes\On;
use Illuminate\Support\Facades\Blade;
use Illuminate\Support\Facades\Route;
use Illuminate\Support\Facades\View;
use Livewire\Attributes\Layout;
use Livewire\Attributes\Url;
use Livewire\Component;
use Livewire\Drawer\Utils;
use Livewire\Livewire;

class BrowserTest extends \Tests\BrowserTestCase
{
    public static function tweakApplicationHook()
    {
        return function () {
            View::addNamespace('test-views', __DIR__ . '/test-views');

            Livewire::component('query-page', QueryPage::class);
            Livewire::component('first-page', FirstPage::class);
            Livewire::component('first-page-child', FirstPageChild::class);
            Livewire::component('first-page-with-link-outside', FirstPageWithLinkOutside::class);
            Livewire::component('second-page', SecondPage::class);
            Livewire::component('third-page', ThirdPage::class);
            Livewire::component('first-asset-page', FirstAssetPage::class);
            Livewire::component('second-asset-page', SecondAssetPage::class);
            Livewire::component('third-asset-page', ThirdAssetPage::class);
            Livewire::component('first-tracked-asset-page', FirstTrackedAssetPage::class);
            Livewire::component('second-tracked-asset-page', SecondTrackedAssetPage::class);
            Livewire::component('second-remote-asset', SecondRemoteAsset::class);
            Livewire::component('first-scroll-page', FirstScrollPage::class);
            Livewire::component('second-scroll-page', SecondScrollPage::class);
            Livewire::component('parent-component', ParentComponent::class);
            Livewire::component('child-component', ChildComponent::class);
            Livewire::component('script-component', ScriptComponent::class);

            Livewire::component('nav-bar-component', NavBarComponent::class);

            Route::get('/navbar/{page}', NavBarComponent::class)->middleware('web');

            Route::get('/query-page', QueryPage::class)->middleware('web');
            Route::get('/first', FirstPage::class)->middleware('web');
            Route::get('/first-hide-progress', function () {
                config(['livewire.navigate.show_progress_bar' => false]);

                return (new FirstPage)();
            })->middleware('web');
            Route::get('/first-outside', FirstPageWithLinkOutside::class)->middleware('web');
            Route::get('/redirect-to-second', fn () => redirect()->to('/second'));
            Route::get('/second', SecondPage::class)->middleware('web');
            Route::get('/third', ThirdPage::class)->middleware('web');
            Route::get('/fourth', FourthPage::class)->middleware('web');
            Route::get('/first-asset', FirstAssetPage::class)->middleware('web');
            Route::get('/second-asset', SecondAssetPage::class)->middleware('web');
            Route::get('/third-asset', ThirdAssetPage::class)->middleware('web');
            Route::get('/first-scroll', FirstScrollPage::class)->middleware('web');
            Route::get('/second-scroll', SecondScrollPage::class)->middleware('web');
            Route::get('/second-remote-asset', SecondRemoteAsset::class)->middleware('web');

            Route::get('/first-tracked-asset', FirstTrackedAssetPage::class)->middleware('web');
            Route::get('/second-tracked-asset', SecondTrackedAssetPage::class)->middleware('web');

            Route::get('/test-navigate-asset.js', function () {
                return Utils::pretendResponseIsFile(__DIR__ . '/test-views/test-navigate-asset.js');
            });

            Route::get('/parent', ParentComponent::class)->middleware('web');
            Route::get('/page-with-link-to-page-without-livewire', PageWithLinkAway::class);
            Route::get('/page-without-livewire-component', fn () => Blade::render(<<<'HTML'
                <html>
                    <head>
                        <meta name="empty-layout" content>

                        <script src="/test-navigate-asset.js" data-navigate-track></script>
                    </head>
                    <body>
                        <div dusk="non-livewire-page">This is a page without a livewire component</div>
                    </body>
                </html>
            HTML));

            Route::get('/page-with-alpine-for-loop', PageWithAlpineForLoop::class);
            Route::get('/script-component', ScriptComponent::class);
        };
    }

    public function test_back_button_works_with_teleports()
    {
        $this->registerComponentTestRoutes([
            '/second' => new class extends Component {
                public function render(){ return <<<'HTML'
                    <div>
                        On second page
                    </div>
                HTML; }
            },
        ]);

        Livewire::visit(new class extends Component {
            public function render(){
                return <<<'HTML'
                    <div x-data="{ outerScopeCount: 0 }">
                        Livewire component...

                        <template x-teleport="body">
                            <div>
                                <span x-text="outerScopeCount" dusk="target"></span>
                                <button x-on:click="outerScopeCount++" dusk="button">inc</button>
                            </div>
                        </template>

                        <a href="/second" wire:navigate dusk="link">Go to second page</a>
                    </div>
                HTML;
            }
        })
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ->click('@link')
        ->waitForText('On second page')
        ->back()
        ->assertDontSee('On second page')
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ->forward()
        ->back()
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ;
    }

    public function test_back_button_works_with_teleports_inside_persist()
    {
        $this->registerComponentTestRoutes([
            '/second' => new class extends Component {
                public function render(){ return <<<'HTML'
                    <div>
                        <div>
                            On second page
                        </div>

                        @persist('header')
                            <div x-data="{ outerScopeCount: 0 }">
                                <template x-teleport="body">
                                    <div>
                                        <span x-text="outerScopeCount" dusk="target"></span>
                                        <button x-on:click="outerScopeCount++" dusk="button">inc</button>
                                    </div>
                                </template>
                            </div>
                        @endpersist
                    </div>
                HTML; }
            },
        ]);

        Livewire::visit(new class extends Component {
            public function render(){
                return <<<'HTML'
                    <div>
                        <div>
                            On first page
                        </div>

                        @persist('header')
                            <div x-data="{ outerScopeCount: 0 }">
                                <template x-teleport="body">
                                    <div>
                                        <span x-text="outerScopeCount" dusk="target"></span>
                                        <button x-on:click="outerScopeCount++" dusk="button">inc</button>
                                    </div>
                                </template>
                            </div>
                        @endpersist

                        <a href="/second" wire:navigate dusk="link">Go to second page</a>
                    </div>
                HTML;
            }
        })
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ->click('@link')
        ->waitForText('On second page')
        ->assertSeeIn('@target', '1')
        ->click('@button')
        ->assertSeeIn('@target', '2')
        ->back()
        ->assertSeeIn('@target', '2')
        ->click('@button')
        ->assertSeeIn('@target', '3')
        ->forward()
        ->assertSeeIn('@target', '3')
        ->click('@button')
        ->assertSeeIn('@target', '4')
        ;
    }

    public function test_can_configure_progress_bar()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.third')
                ->waitFor('#nprogress')
                ->waitForText('Done loading...');
        });

        $this->browse(function ($browser) {
            $browser
                ->visit('/first-hide-progress')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.third')
                ->pause(500)
                ->assertMissing('#nprogress')
                ->waitForText('Done loading...');
        });
    }

    public function test_can_navigate_to_page_without_reloading()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                ->click('@link.to.first')
                ->waitFor('@link.to.second')
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first');
        });
    }

    public function test_can_navigate_to_page_without_reloading_by_hitting_the_enter_key()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->keys('@link.to.second', '{enter}')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_can_navigate_to_another_page_with_hash_fragment()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->waitForNavigate()->click('@link.to.hashtag')
                ->assertFragmentIs('foo');
        });
    }

    public function test_navigate_is_not_triggered_on_cmd_and_enter()
    {
        $key = PHP_OS_FAMILY === 'Darwin' ? \Facebook\WebDriver\WebDriverKeys::COMMAND : \Facebook\WebDriver\WebDriverKeys::CONTROL;

        $this->browse(function (Browser $browser) use ($key) {
            $currentWindowHandles = count($browser->driver->getWindowHandles());

            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->keys('@link.to.second', $key, '{enter}')
                ->pause(500) // Let navigate run if it was going to (it should not)
                ->assertSee('On first')
                ->assertScript('return window._lw_dusk_test');

            $this->assertCount($currentWindowHandles + 1, $browser->driver->getWindowHandles());
        });
    }

    public function test_can_navigate_to_page_from_child_via_parent_component_without_reloading()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                ->click('@redirect.to.second.from.child')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->click('@link.to.first')
                ->waitFor('@redirect.to.second.from.child')
                ->assertSee('On first')
                ->click('@redirect.to.second.from.child')
                ->waitFor('@link.to.first')
                ->assertSee('On second');
        });
    }

    public function test_can_redirect_with_reloading_from_a_page_that_was_loaded_by_wire_navigate()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                ->click('@redirect.to.first')
                ->waitFor('@link.to.second')
                ->assertScript('return window._lw_dusk_test', false)
                ->assertSee('On first');
        });
    }

    public function test_can_redirect_without_reloading_using_the_helper_from_a_page_that_was_loaded_normally()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@redirect.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_can_redirect_to_a_page_after_destorying_session()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@redirect.to.second.and.destroy.session')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                ->assertConsoleLogMissingWarning('Detected multiple instances of Livewire')
                ->assertConsoleLogMissingWarning('Detected multiple instances of Alpine');
        });
    }

    public function test_can_persist_elements_across_pages()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSeeIn('@count', '1')
                ->click('@increment')
                ->assertSeeIn('@count', '2')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertSeeIn('@count', '2')
                ->click('@increment')
                ->assertSeeIn('@count', '3')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_new_assets_in_head_are_loaded_and_old_ones_are_not()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-asset')
                ->assertScript('return _lw_dusk_asset_count', 1)
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitForText('On second')
                ->assertScript('return _lw_dusk_asset_count', 1)
                ->click('@link.to.third')
                ->waitForText('On third')
                ->assertScript('return _lw_dusk_asset_count', 2);
        });
    }

    public function test_tracked_assets_reload_the_page_when_they_change()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-tracked-asset')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertScript('return _lw_dusk_asset_count', 1)
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitForText('On second')
                ->assertScript('return window._lw_dusk_test', false)
                ->assertScript('return _lw_dusk_asset_count', 1);
        });
    }

    public function test_can_use_wire_navigate_outside_of_a_livewire_component()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-outside')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@outside.link.to.second')
                ->waitForText('On second')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_script_runs_on_initial_page_visit()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('window.foo', 'bar')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_can_navigate_to_component_with_url_attribute_and_update_correctly()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/query-page')
                ->assertSee('Query: 0')
                ->click('@link.with.query.1')
                ->assertSee('Query: 1')
                ->waitForNavigate()->click('@link.with.query.2')
                ->assertSee('Query: 2');
        });
    }

    public function test_navigate_scrolls_to_top_and_back_preserves_scroll()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-scroll')
                ->assertVisible('@first-target')
                ->assertNotInViewPort('@first-target')
                ->scrollTo('@first-target')
                ->assertInViewPort('@first-target')

                ->click('@link.to.second')
                ->waitForText('On second')
                ->assertNotInViewPort('@second-target')
                ->scrollTo('@second-target')

                ->back()
                ->waitForText('On first')
                ->assertInViewPort('@first-target')

                ->forward()
                ->waitForText('On second')
                ->assertInViewPort('@second-target')
            ;
        });
    }

    public function test_navigate_back_works_from_page_without_a_livewire_component_that_has_a_script_with_data_navigate_track()
    {
        // When using `@vite` on the page without a Livewire component,
        // it injects a script tag with `data-navigate-track`,
        // which causes Livewire to be unloaded and the back button no longer work.
        $this->browse(function ($browser) {
            $browser
                ->visit('/page-with-link-to-page-without-livewire')
                ->assertSee('Link to page without Livewire component')
                ->assertDontSee('This is a page without a livewire component')
                ->click('@link.away')
                ->waitFor('@non-livewire-page')
                ->assertSee('This is a page without a livewire component')
                ->assertDontSee('Link to page without Livewire component')
                ->back()
                ->waitFor('@page-with-link-away')
                ->assertSee('Link to page without Livewire component')
                ->assertDontSee('This is a page without a livewire component')
            ;
        });
    }

    public function test_navigate_is_only_triggered_on_left_click()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->rightClick('@link.to.second')
                ->pause(500) // Let navigate run if it was going to (it should not)
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ;
        });
    }

    public function test_livewire_navigated_event_is_fired_on_first_page_load()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/second')
                ->assertSee('On second')
                ->assertScript('window.foo_navigated', 'bar');
        });
    }

    public function test_livewire_before_navigate_event_is_fired_when_click()
    {
        $this->browse(function($browser) {
            $browser
                ->visit('/fourth')
                ->assertSee('On fourth')
                ->assertScript('window.foo', 'bar')
                ->assertSee('On fourth')
                ->click('@link.to.first') // first attempt bar -> baz
                ->assertScript('window.foo', 'baz')
                ->assertSee('On fourth')
                ->click('@link.to.first') // second attempt baz -> bat
                ->assertScript('window.foo', 'bat')
                ->assertSee('On fourth')
                ->click('@link.to.first') // finally navigate
                ->assertSee('On first')
            ;
        });
    }

    public function test_livewire_navigated_event_is_fired_after_redirect_without_reloading()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('window.foo_navigated', 'bar');
        });
    }

    public function test_navigate_is_not_triggered_on_cmd_click()
    {
        $key = PHP_OS_FAMILY === 'Darwin' ? \Facebook\WebDriver\WebDriverKeys::COMMAND : \Facebook\WebDriver\WebDriverKeys::CONTROL;

        $this->browse(function (Browser $browser) use ($key) {
            $currentWindowHandles = count($browser->driver->getWindowHandles());

            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->tap(function ($browser) use ($key) {
                    $browser->driver->getKeyboard()->pressKey($key);
                })
                ->click('@link.to.second')
                ->tap(function ($browser) use ($key) {
                    $browser->driver->getKeyboard()->releaseKey($key);
                })
                ->pause(500) // Let navigate run if it was going to (it should not)
                ->assertSee('On first')
                ->assertScript('return window._lw_dusk_test')
            ;

            $this->assertCount($currentWindowHandles + 1, $browser->driver->getWindowHandles());
        });
    }

    public function test_events_from_child_components_still_function_after_navigation()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/parent')
                ->assertSeeNothingIn('@text-child')
                ->assertSeeNothingIn('@text-parent')
                ->waitForLivewire()->type('@text-input', 'test')
                ->waitForTextIn('@text-child', 'test')
                ->waitForTextIn('@text-parent', 'test')

                ->waitForNavigate()->click('@home-link')
                ->assertSeeNothingIn('@text-child')
                ->assertSeeNothingIn('@text-parent')
                ->waitForLivewire()->type('@text-input', 'testing')
                ->waitForTextIn('@text-child', 'testing')
                ->waitForTextIn('@text-parent', 'testing')

                ->back()
                ->waitForTextIn('@text-child', 'test')
                ->waitForTextIn('@text-parent', 'test')
                ->waitForLivewire()->type('@text-input', 'testing')
                ->waitForTextIn('@text-child', 'testing')
                ->waitForTextIn('@text-parent', 'testing');
        });
    }

    public function test_alpine_for_loop_still_functions_after_navigation()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/page-with-alpine-for-loop')
                ->assertSeeIn('@text', 'a,b,c')
                ->assertScript('document.getElementById(\'alpine-for-loop\').querySelectorAll(\'p\').length', 3)
                ->assertConsoleLogMissingWarning('value is not defined')

                ->waitForNavigate()->click('@link.to.second')
                ->assertSee('On second')

                ->back()
                ->assertSeeIn('@text', 'a,b,c')
                ->assertScript('document.getElementById(\'alpine-for-loop\').querySelectorAll(\'p\').length', 3)
                ->assertConsoleLogMissingWarning('value is not defined')
            ;
        });
    }

    public function test_injected_assets_such_as_nprogress_styles_are_retained_when_the_page_changes()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                // There should only be two style blocks, livewire styles and nprogress
                ->assertScript('return document.styleSheets.length', 2)
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                // There should only be two style blocks, livewire styles and nprogress
                ->assertScript('return document.styleSheets.length', 2)
                ->click('@link.to.first')
                ->waitFor('@link.to.second')
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                // There should only be two style blocks, livewire styles and nprogress
                ->assertScript('return document.styleSheets.length', 2);
        });
    }

    public function test_remote_assets_loaded_with_the_directive_fully_load_before_component_scripts_and_initialization()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                // There should only be two style blocks, livewire styles and nprogress
                ->click('@link.to.asset')
                ->waitFor('@target')
                ->waitForTextIn('@target', 'bar')
                ;
        });
    }

    public function test_redirects_are_reflected_properly_in_the_url()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                ->click('@redirect.to.second.link')
                ->waitForText('On second')
                ->assertPathIs('/second')
            ;
        });
    }

    public function test_can_programmatically_click_navigate_links()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                ->tap(function ($browser) {
                    $browser->script(<<<'JS'
                        document.querySelector('a[href="/second"]').click()
                    JS);
                })
                ->waitForText('On second')
                ->assertPathIs('/second')
            ;
        });
    }

    public function test_can_binding_class_attribute_when_navigate_back()
    {
        Livewire::visit(new class extends Component {
            public function render(){
                return <<<'HTML'
                    <div>
                        <style>
                            .hidden {
                                display: none;
                            }
                        </style>

                        <div x-data="{ show: false }">
                            <button dusk="show-foo" type="button" @click="show = !show">Show</button>
                            <span :class="show || 'hidden'">foo</span>
                        </div>

                        <a :href="window.location.pathname" wire:navigate dusk="navigate-to-same-page">Go to same page</a>

                    </div>
                HTML;
            }
        })
            ->click('@show-foo')
            ->click('@show-foo')
            ->waitForNavigate()->click('@navigate-to-same-page')
            ->back()
            ->click('@show-foo')
            ->assertSee('foo');
    }

    public function test_can_navigate_links_and_use_snapshot_cache_for_first_10_history_items()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/navbar/one')
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.two')
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.one', 'active')

                ->waitForNavigateRequest()->click('@link.three')
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.four')
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNavigateRequest()->click('@link.five')
                ->assertSeeIn('@title', 'five')
                ->assertHasClass('@link.five', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNavigateRequest()->click('@link.six')
                ->assertSeeIn('@title', 'six')
                ->assertHasClass('@link.six', 'active')
                ->assertClassMissing('@link.five', 'active')

                ->waitForNavigateRequest()->click('@link.seven')
                ->assertSeeIn('@title', 'seven')
                ->assertHasClass('@link.seven', 'active')
                ->assertClassMissing('@link.six', 'active')

                ->waitForNavigateRequest()->click('@link.eight')
                ->assertSeeIn('@title', 'eight')
                ->assertHasClass('@link.eight', 'active')
                ->assertClassMissing('@link.seven', 'active')

                ->waitForNavigateRequest()->click('@link.nine')
                ->assertSeeIn('@title', 'nine')
                ->assertHasClass('@link.nine', 'active')
                ->assertClassMissing('@link.eight', 'active')

                ->waitForNavigateRequest()->click('@link.ten')
                ->assertSeeIn('@title', 'ten')
                ->assertHasClass('@link.ten', 'active')
                ->assertClassMissing('@link.nine', 'active')

                ->waitForNavigateRequest()->click('@link.eleven')
                ->assertSeeIn('@title', 'eleven')
                ->assertHasClass('@link.eleven', 'active')
                ->assertClassMissing('@link.ten', 'active')

                ->waitForNavigateRequest()->click('@link.twelve')
                ->assertSeeIn('@title', 'twelve')
                ->assertHasClass('@link.twelve', 'active')
                ->assertClassMissing('@link.eleven', 'active')

                ->waitForNavigateRequest()->click('@link.thirteen')
                ->assertSeeIn('@title', 'thirteen')
                ->assertHasClass('@link.thirteen', 'active')
                ->assertClassMissing('@link.twelve', 'active')

                // Assert no navigate request as we expect it to come from the cache
                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'twelve')
                ->assertHasClass('@link.twelve', 'active')
                ->assertClassMissing('@link.thirteen', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'eleven')
                ->assertHasClass('@link.eleven', 'active')
                ->assertClassMissing('@link.twelve', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'ten')
                ->assertHasClass('@link.ten', 'active')
                ->assertClassMissing('@link.eleven', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'nine')
                ->assertHasClass('@link.nine', 'active')
                ->assertClassMissing('@link.ten', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'eight')
                ->assertHasClass('@link.eight', 'active')
                ->assertClassMissing('@link.nine', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'seven')
                ->assertHasClass('@link.seven', 'active')
                ->assertClassMissing('@link.eight', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'six')
                ->assertHasClass('@link.six', 'active')
                ->assertClassMissing('@link.seven', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'five')
                ->assertHasClass('@link.five', 'active')
                ->assertClassMissing('@link.six', 'active')

                // Assert a navigate request was triggered as the remaining pages should no longer be in the cache
                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.five', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')

            ;
        });
    }

    public function test_can_navigate_links_and_if_a_refresh_happens_then_make_requests_until_pages_are_cached_again()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/navbar/one')
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.two')
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.one', 'active')

                ->waitForNavigateRequest()->click('@link.three')
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.four')
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForLivewire()->refresh()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.three')
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.four')
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')
            ;
        });
    }

    public function test_navigate_back_reevaluates_scripts()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/script-component')
                ->waitForDialog(seconds: 1)
                ->assertDialogOpened('script was executed?')
                ->acceptDialog()
                ->assertSee('On script component')
                ->click('@link.to.first')
                ->waitForText('On first')
                ->back()
                ->waitForDialog(seconds: 1)
                ->assertDialogOpened('script was executed?')
                ->acceptDialog()
                ->waitForText('On script component')
                ->click('@link.to.first')
                ->waitForText('On first')
                ->back()
                ->waitForDialog(seconds: 1)
                ->assertDialogOpened('script was executed?')
                ->acceptDialog()
                ->waitForText('On script component')
            ;
        });
    }

    protected function registerComponentTestRoutes($routes)
    {
        $registered = 0;

        foreach ($routes as $route => $component) {
            $name = 'route-component-'.$registered++;

            Livewire::component($name, $component);

            Route::get($route, function () use ($name) {
                return app('livewire')->new($name)();
            })->middleware('web');
        }
    }
}

class FirstPage extends Component
{
    public function redirectToPageTwoUsingNavigate()
    {
        return $this->redirect('/second', navigate: true);
    }

    public function redirectToPageTwoUsingNavigateAndDestroyingSession()
    {
        session()->regenerate();

        return $this->redirect('/second', navigate: true);
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On first</div>

            <a :href="window.location.pathname + '#foo'" wire:navigate dusk="link.to.hashtag">Go to same page with hashtag</a>
            <a href="/second" wire:navigate.hover dusk="link.to.second">Go to second page</a>
            <a href="/third" wire:navigate.hover dusk="link.to.third">Go to slow third page</a>
            <a href="/second-remote-asset" wire:navigate.hover dusk="link.to.asset">Go to asset page</a>
            <button type="button" wire:click="redirectToPageTwoUsingNavigate" dusk="redirect.to.second">Redirect to second page</button>
            <a href="/redirect-to-second" wire:navigate dusk="redirect.to.second.link">Redirect to second page from link</a>
            <button type="button" wire:click="redirectToPageTwoUsingNavigateAndDestroyingSession" dusk="redirect.to.second.and.destroy.session">Redirect to second page and destroy session</button>

            <livewire:first-page-child />

            @persist('foo')
                <div x-data="{ count: 1 }">
                    <span x-text="count" dusk="count"></span>
                    <button x-on:click="count++" dusk="increment">+</button>
                </div>
            @endpersist
        </div>
        HTML;
    }
}

class FirstPageChild extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>First Child</div>

            <button type="button" wire:click="$parent.redirectToPageTwoUsingNavigate" dusk="redirect.to.second.from.child">Redirect to second page from child</button>
            <button type="button" x-on:click="console.log($wire.$parent.__instance.id)">shmump up</button>
        </div>
        HTML;
    }
}

class FirstPageWithLinkOutside extends Component
{
    #[Layout('test-views::layout-with-navigate-outside')]
    public function render()
    {
        return '<div>On first</div>';
    }
}

class SecondPage extends Component
{
    public function redirectToPageOne()
    {
        return redirect('/first');
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On second</div>

            <a href="/first" wire:navigate dusk="link.to.first">Go to first page</a>
            <button type="button" wire:click="redirectToPageOne" dusk="redirect.to.first">Redirect to first page</button>

            @persist('foo')
                <div x-data="{ count: 1 }">
                    <span x-text="count" dusk="count"></span>
                    <button x-on:click="count++" dusk="increment">+</button>
                </div>
            @endpersist

            <script data-navigate-once>window.foo = 'bar';</script>

            <script>
                document.addEventListener('livewire:navigated', () => {
                    window.foo_navigated = 'bar'
                })
            </script>
        </div>
        HTML;
    }
}

class ThirdPage extends Component
{
    public function mount()
    {
        sleep(1);
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            Done loading...
        </div>
        HTML;
    }
}

class FourthPage extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On fourth</div>

            <a href="/first" wire:navigate dusk="link.to.first">Go to first page</a>

            <script data-navigate-once>window.foo = 'bar';</script>

            <script>
                document.addEventListener('livewire:navigate', (event) => {
                    event.preventDefault();
                    if (window.foo === 'bar') {
                        window.foo = 'baz'
                    }
                    else if (window.foo === 'baz') {
                        window.foo ='bat'
                    } else {
                        Alpine.navigate(event.detail.url)
                    }
                })
            </script>
        </div>
        HTML;
    }
}

class FirstAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::layout')]
    public function render()
    {
        return '<div>On first asset page <a href="/second-asset" wire:navigate dusk="link.to.second">Go to second page</a></div>';
    }
}

class SecondAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::layout')]
    public function render()
    {
        return '<div>On second asset page <a href="/third-asset" wire:navigate dusk="link.to.third">Go to third page</a></div>';
    }
}

class ThirdAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::changed-layout')]
    public function render()
    {
        return '<div>On third asset page</div>';
    }
}

class FirstTrackedAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::tracked-layout')]
    public function render()
    {
        return '<div>On first asset page <a href="/second-tracked-asset" wire:navigate dusk="link.to.second">Go to second page</a></div>';
    }
}

class SecondTrackedAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::changed-tracked-layout')]
    public function render()
    {
        return '<div>On second asset page</div>';
    }
}

class SecondRemoteAsset extends Component
{
    public function render()
    {
        return <<<'HTML'
            <div>
                <div>On second asset page</div>
                <div dusk="target">foo</div>
            </div>

            @assets
            <script src="https://cdn.jsdelivr.net/npm/pikaday/pikaday.js" defer></script>
            @endassets

            @script
            <script>
                window.datePicker = new Pikaday({ field: $wire.$el.querySelector('[data-picker]') })

                if (window.datePicker) {
                    document.querySelector('[dusk="target"]').textContent = 'bar'
                }
            </script>
            @endscript
        HTML;
    }
}

class QueryPage extends Component
{
    #[Url]
    public $query = 0;

    public function render()
    {
        return <<<'HTML'
            <div>
                <div>Query: {{ $query }}</div>
                <a href="/query-page?query=1" dusk="link.with.query.1">Link with query 1</a>
                <a href="/query-page?query=2" wire:navigate dusk="link.with.query.2">Link with query 2</a>
            </div>
        HTML;
    }
}

class FirstScrollPage extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On first</div>

            <div style="height: 100vh;">spacer</div>

            <div dusk="first-target">below the fold</div>

            <a href="/second-scroll" wire:navigate.hover dusk="link.to.second">Go to second page</a>

            <div style="height: 100vh;">spacer</div>
        </div>
        HTML;
    }
}

class SecondScrollPage extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On second</div>

            <div style="height: 100vh;">spacer</div>

            <div dusk="second-target">below the fold</div>

            <div style="height: 100vh;">spacer</div>
        </div>
        HTML;
    }
}

class ParentComponent extends Component
{
    public $text = '';

    #[On('my-event')]
    public function change_text($text)
    {
        $this->text = $text;
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <a href="/parent" wire:navigate dusk="home-link">Home</a>

            <p dusk="text-parent">{{ $text }}</p>

            <livewire:child-component key="child" />
        </div>
        HTML;
    }
}

class ChildComponent extends Component
{
    public $text = '';

    public function updated()
    {
        $this->dispatch('my-event', text: $this->text);
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <p dusk="text-child">{{ $text }}</p>

            <input type="text" wire:model.live="text" dusk="text-input">
        </div>
        HTML;
    }
}

class PageWithLinkAway extends Component
{
    #[Layout('test-views::layout')]
    public function render()
    {
        return <<<'HTML'
        <div dusk="page-with-link-away">
            <a wire:navigate dusk="link.away" href="/page-without-livewire-component">
                Link to page without Livewire component
            </a>
        </div>
        HTML;
    }
}

class PageWithAlpineForLoop extends Component
{
    #[Layout('test-views::layout')]
    public function render()
    {
        return <<<'HTML'
        <div dusk="page-with-alpine-for-loop" x-data="{ items: ['a', 'b', 'c'] }">
            <a href="/second" wire:navigate dusk="link.to.second">Go to second page</a>
            <div dusk="text" x-text="items"></div>
            <div id="alpine-for-loop">
                <template x-for="(value, index) in items" :key="index">
                    <p x-text="value"></p>
                </template>
            </div>
        </div>
        HTML;
    }
}

class NavBarComponent extends Component
{
    public $page;

    #[Layout('test-views::navbar-sidebar')]
    public function render()
    {
        return <<<'HTML'
            <div>
                <div>Page: <span dusk="title">{{ $page }}</span></div>
            </div>
        HTML;
    }
}

class ScriptComponent extends Component
{
    public function render()
    {
        return <<<'HTML'
            @script
            <script>
                confirm('script was executed?')
            </script>
            @endscript
            <div>
                <div>On script component</div>
                <a href="/first" wire:navigate dusk="link.to.first">Go to first page</a>
            </div>
        HTML;
    }
}
 ?>

Did this file decode correctly?

Original Code

<?php

namespace Livewire\Features\SupportNavigate;

use Laravel\Dusk\Browser;
use Livewire\Attributes\On;
use Illuminate\Support\Facades\Blade;
use Illuminate\Support\Facades\Route;
use Illuminate\Support\Facades\View;
use Livewire\Attributes\Layout;
use Livewire\Attributes\Url;
use Livewire\Component;
use Livewire\Drawer\Utils;
use Livewire\Livewire;

class BrowserTest extends \Tests\BrowserTestCase
{
    public static function tweakApplicationHook()
    {
        return function () {
            View::addNamespace('test-views', __DIR__ . '/test-views');

            Livewire::component('query-page', QueryPage::class);
            Livewire::component('first-page', FirstPage::class);
            Livewire::component('first-page-child', FirstPageChild::class);
            Livewire::component('first-page-with-link-outside', FirstPageWithLinkOutside::class);
            Livewire::component('second-page', SecondPage::class);
            Livewire::component('third-page', ThirdPage::class);
            Livewire::component('first-asset-page', FirstAssetPage::class);
            Livewire::component('second-asset-page', SecondAssetPage::class);
            Livewire::component('third-asset-page', ThirdAssetPage::class);
            Livewire::component('first-tracked-asset-page', FirstTrackedAssetPage::class);
            Livewire::component('second-tracked-asset-page', SecondTrackedAssetPage::class);
            Livewire::component('second-remote-asset', SecondRemoteAsset::class);
            Livewire::component('first-scroll-page', FirstScrollPage::class);
            Livewire::component('second-scroll-page', SecondScrollPage::class);
            Livewire::component('parent-component', ParentComponent::class);
            Livewire::component('child-component', ChildComponent::class);
            Livewire::component('script-component', ScriptComponent::class);

            Livewire::component('nav-bar-component', NavBarComponent::class);

            Route::get('/navbar/{page}', NavBarComponent::class)->middleware('web');

            Route::get('/query-page', QueryPage::class)->middleware('web');
            Route::get('/first', FirstPage::class)->middleware('web');
            Route::get('/first-hide-progress', function () {
                config(['livewire.navigate.show_progress_bar' => false]);

                return (new FirstPage)();
            })->middleware('web');
            Route::get('/first-outside', FirstPageWithLinkOutside::class)->middleware('web');
            Route::get('/redirect-to-second', fn () => redirect()->to('/second'));
            Route::get('/second', SecondPage::class)->middleware('web');
            Route::get('/third', ThirdPage::class)->middleware('web');
            Route::get('/fourth', FourthPage::class)->middleware('web');
            Route::get('/first-asset', FirstAssetPage::class)->middleware('web');
            Route::get('/second-asset', SecondAssetPage::class)->middleware('web');
            Route::get('/third-asset', ThirdAssetPage::class)->middleware('web');
            Route::get('/first-scroll', FirstScrollPage::class)->middleware('web');
            Route::get('/second-scroll', SecondScrollPage::class)->middleware('web');
            Route::get('/second-remote-asset', SecondRemoteAsset::class)->middleware('web');

            Route::get('/first-tracked-asset', FirstTrackedAssetPage::class)->middleware('web');
            Route::get('/second-tracked-asset', SecondTrackedAssetPage::class)->middleware('web');

            Route::get('/test-navigate-asset.js', function () {
                return Utils::pretendResponseIsFile(__DIR__ . '/test-views/test-navigate-asset.js');
            });

            Route::get('/parent', ParentComponent::class)->middleware('web');
            Route::get('/page-with-link-to-page-without-livewire', PageWithLinkAway::class);
            Route::get('/page-without-livewire-component', fn () => Blade::render(<<<'HTML'
                <html>
                    <head>
                        <meta name="empty-layout" content>

                        <script src="/test-navigate-asset.js" data-navigate-track></script>
                    </head>
                    <body>
                        <div dusk="non-livewire-page">This is a page without a livewire component</div>
                    </body>
                </html>
            HTML));

            Route::get('/page-with-alpine-for-loop', PageWithAlpineForLoop::class);
            Route::get('/script-component', ScriptComponent::class);
        };
    }

    public function test_back_button_works_with_teleports()
    {
        $this->registerComponentTestRoutes([
            '/second' => new class extends Component {
                public function render(){ return <<<'HTML'
                    <div>
                        On second page
                    </div>
                HTML; }
            },
        ]);

        Livewire::visit(new class extends Component {
            public function render(){
                return <<<'HTML'
                    <div x-data="{ outerScopeCount: 0 }">
                        Livewire component...

                        <template x-teleport="body">
                            <div>
                                <span x-text="outerScopeCount" dusk="target"></span>
                                <button x-on:click="outerScopeCount++" dusk="button">inc</button>
                            </div>
                        </template>

                        <a href="/second" wire:navigate dusk="link">Go to second page</a>
                    </div>
                HTML;
            }
        })
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ->click('@link')
        ->waitForText('On second page')
        ->back()
        ->assertDontSee('On second page')
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ->forward()
        ->back()
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ;
    }

    public function test_back_button_works_with_teleports_inside_persist()
    {
        $this->registerComponentTestRoutes([
            '/second' => new class extends Component {
                public function render(){ return <<<'HTML'
                    <div>
                        <div>
                            On second page
                        </div>

                        @persist('header')
                            <div x-data="{ outerScopeCount: 0 }">
                                <template x-teleport="body">
                                    <div>
                                        <span x-text="outerScopeCount" dusk="target"></span>
                                        <button x-on:click="outerScopeCount++" dusk="button">inc</button>
                                    </div>
                                </template>
                            </div>
                        @endpersist
                    </div>
                HTML; }
            },
        ]);

        Livewire::visit(new class extends Component {
            public function render(){
                return <<<'HTML'
                    <div>
                        <div>
                            On first page
                        </div>

                        @persist('header')
                            <div x-data="{ outerScopeCount: 0 }">
                                <template x-teleport="body">
                                    <div>
                                        <span x-text="outerScopeCount" dusk="target"></span>
                                        <button x-on:click="outerScopeCount++" dusk="button">inc</button>
                                    </div>
                                </template>
                            </div>
                        @endpersist

                        <a href="/second" wire:navigate dusk="link">Go to second page</a>
                    </div>
                HTML;
            }
        })
        ->assertSeeIn('@target', '0')
        ->click('@button')
        ->assertSeeIn('@target', '1')
        ->click('@link')
        ->waitForText('On second page')
        ->assertSeeIn('@target', '1')
        ->click('@button')
        ->assertSeeIn('@target', '2')
        ->back()
        ->assertSeeIn('@target', '2')
        ->click('@button')
        ->assertSeeIn('@target', '3')
        ->forward()
        ->assertSeeIn('@target', '3')
        ->click('@button')
        ->assertSeeIn('@target', '4')
        ;
    }

    public function test_can_configure_progress_bar()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.third')
                ->waitFor('#nprogress')
                ->waitForText('Done loading...');
        });

        $this->browse(function ($browser) {
            $browser
                ->visit('/first-hide-progress')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.third')
                ->pause(500)
                ->assertMissing('#nprogress')
                ->waitForText('Done loading...');
        });
    }

    public function test_can_navigate_to_page_without_reloading()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                ->click('@link.to.first')
                ->waitFor('@link.to.second')
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first');
        });
    }

    public function test_can_navigate_to_page_without_reloading_by_hitting_the_enter_key()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->keys('@link.to.second', '{enter}')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_can_navigate_to_another_page_with_hash_fragment()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->waitForNavigate()->click('@link.to.hashtag')
                ->assertFragmentIs('foo');
        });
    }

    public function test_navigate_is_not_triggered_on_cmd_and_enter()
    {
        $key = PHP_OS_FAMILY === 'Darwin' ? \Facebook\WebDriver\WebDriverKeys::COMMAND : \Facebook\WebDriver\WebDriverKeys::CONTROL;

        $this->browse(function (Browser $browser) use ($key) {
            $currentWindowHandles = count($browser->driver->getWindowHandles());

            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->keys('@link.to.second', $key, '{enter}')
                ->pause(500) // Let navigate run if it was going to (it should not)
                ->assertSee('On first')
                ->assertScript('return window._lw_dusk_test');

            $this->assertCount($currentWindowHandles + 1, $browser->driver->getWindowHandles());
        });
    }

    public function test_can_navigate_to_page_from_child_via_parent_component_without_reloading()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                ->click('@redirect.to.second.from.child')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->click('@link.to.first')
                ->waitFor('@redirect.to.second.from.child')
                ->assertSee('On first')
                ->click('@redirect.to.second.from.child')
                ->waitFor('@link.to.first')
                ->assertSee('On second');
        });
    }

    public function test_can_redirect_with_reloading_from_a_page_that_was_loaded_by_wire_navigate()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                ->click('@redirect.to.first')
                ->waitFor('@link.to.second')
                ->assertScript('return window._lw_dusk_test', false)
                ->assertSee('On first');
        });
    }

    public function test_can_redirect_without_reloading_using_the_helper_from_a_page_that_was_loaded_normally()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@redirect.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_can_redirect_to_a_page_after_destorying_session()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@redirect.to.second.and.destroy.session')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                ->assertConsoleLogMissingWarning('Detected multiple instances of Livewire')
                ->assertConsoleLogMissingWarning('Detected multiple instances of Alpine');
        });
    }

    public function test_can_persist_elements_across_pages()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSeeIn('@count', '1')
                ->click('@increment')
                ->assertSeeIn('@count', '2')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertSeeIn('@count', '2')
                ->click('@increment')
                ->assertSeeIn('@count', '3')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_new_assets_in_head_are_loaded_and_old_ones_are_not()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-asset')
                ->assertScript('return _lw_dusk_asset_count', 1)
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitForText('On second')
                ->assertScript('return _lw_dusk_asset_count', 1)
                ->click('@link.to.third')
                ->waitForText('On third')
                ->assertScript('return _lw_dusk_asset_count', 2);
        });
    }

    public function test_tracked_assets_reload_the_page_when_they_change()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-tracked-asset')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertScript('return _lw_dusk_asset_count', 1)
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitForText('On second')
                ->assertScript('return window._lw_dusk_test', false)
                ->assertScript('return _lw_dusk_asset_count', 1);
        });
    }

    public function test_can_use_wire_navigate_outside_of_a_livewire_component()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-outside')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@outside.link.to.second')
                ->waitForText('On second')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_script_runs_on_initial_page_visit()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('window.foo', 'bar')
                ->assertScript('return window._lw_dusk_test');
        });
    }

    public function test_can_navigate_to_component_with_url_attribute_and_update_correctly()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/query-page')
                ->assertSee('Query: 0')
                ->click('@link.with.query.1')
                ->assertSee('Query: 1')
                ->waitForNavigate()->click('@link.with.query.2')
                ->assertSee('Query: 2');
        });
    }

    public function test_navigate_scrolls_to_top_and_back_preserves_scroll()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first-scroll')
                ->assertVisible('@first-target')
                ->assertNotInViewPort('@first-target')
                ->scrollTo('@first-target')
                ->assertInViewPort('@first-target')

                ->click('@link.to.second')
                ->waitForText('On second')
                ->assertNotInViewPort('@second-target')
                ->scrollTo('@second-target')

                ->back()
                ->waitForText('On first')
                ->assertInViewPort('@first-target')

                ->forward()
                ->waitForText('On second')
                ->assertInViewPort('@second-target')
            ;
        });
    }

    public function test_navigate_back_works_from_page_without_a_livewire_component_that_has_a_script_with_data_navigate_track()
    {
        // When using `@vite` on the page without a Livewire component,
        // it injects a script tag with `data-navigate-track`,
        // which causes Livewire to be unloaded and the back button no longer work.
        $this->browse(function ($browser) {
            $browser
                ->visit('/page-with-link-to-page-without-livewire')
                ->assertSee('Link to page without Livewire component')
                ->assertDontSee('This is a page without a livewire component')
                ->click('@link.away')
                ->waitFor('@non-livewire-page')
                ->assertSee('This is a page without a livewire component')
                ->assertDontSee('Link to page without Livewire component')
                ->back()
                ->waitFor('@page-with-link-away')
                ->assertSee('Link to page without Livewire component')
                ->assertDontSee('This is a page without a livewire component')
            ;
        });
    }

    public function test_navigate_is_only_triggered_on_left_click()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->rightClick('@link.to.second')
                ->pause(500) // Let navigate run if it was going to (it should not)
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ;
        });
    }

    public function test_livewire_navigated_event_is_fired_on_first_page_load()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/second')
                ->assertSee('On second')
                ->assertScript('window.foo_navigated', 'bar');
        });
    }

    public function test_livewire_before_navigate_event_is_fired_when_click()
    {
        $this->browse(function($browser) {
            $browser
                ->visit('/fourth')
                ->assertSee('On fourth')
                ->assertScript('window.foo', 'bar')
                ->assertSee('On fourth')
                ->click('@link.to.first') // first attempt bar -> baz
                ->assertScript('window.foo', 'baz')
                ->assertSee('On fourth')
                ->click('@link.to.first') // second attempt baz -> bat
                ->assertScript('window.foo', 'bat')
                ->assertSee('On fourth')
                ->click('@link.to.first') // finally navigate
                ->assertSee('On first')
            ;
        });
    }

    public function test_livewire_navigated_event_is_fired_after_redirect_without_reloading()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('window.foo_navigated', 'bar');
        });
    }

    public function test_navigate_is_not_triggered_on_cmd_click()
    {
        $key = PHP_OS_FAMILY === 'Darwin' ? \Facebook\WebDriver\WebDriverKeys::COMMAND : \Facebook\WebDriver\WebDriverKeys::CONTROL;

        $this->browse(function (Browser $browser) use ($key) {
            $currentWindowHandles = count($browser->driver->getWindowHandles());

            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                ->tap(function ($browser) use ($key) {
                    $browser->driver->getKeyboard()->pressKey($key);
                })
                ->click('@link.to.second')
                ->tap(function ($browser) use ($key) {
                    $browser->driver->getKeyboard()->releaseKey($key);
                })
                ->pause(500) // Let navigate run if it was going to (it should not)
                ->assertSee('On first')
                ->assertScript('return window._lw_dusk_test')
            ;

            $this->assertCount($currentWindowHandles + 1, $browser->driver->getWindowHandles());
        });
    }

    public function test_events_from_child_components_still_function_after_navigation()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/parent')
                ->assertSeeNothingIn('@text-child')
                ->assertSeeNothingIn('@text-parent')
                ->waitForLivewire()->type('@text-input', 'test')
                ->waitForTextIn('@text-child', 'test')
                ->waitForTextIn('@text-parent', 'test')

                ->waitForNavigate()->click('@home-link')
                ->assertSeeNothingIn('@text-child')
                ->assertSeeNothingIn('@text-parent')
                ->waitForLivewire()->type('@text-input', 'testing')
                ->waitForTextIn('@text-child', 'testing')
                ->waitForTextIn('@text-parent', 'testing')

                ->back()
                ->waitForTextIn('@text-child', 'test')
                ->waitForTextIn('@text-parent', 'test')
                ->waitForLivewire()->type('@text-input', 'testing')
                ->waitForTextIn('@text-child', 'testing')
                ->waitForTextIn('@text-parent', 'testing');
        });
    }

    public function test_alpine_for_loop_still_functions_after_navigation()
    {
        $this->browse(function (Browser $browser) {
            $browser
                ->visit('/page-with-alpine-for-loop')
                ->assertSeeIn('@text', 'a,b,c')
                ->assertScript('document.getElementById(\'alpine-for-loop\').querySelectorAll(\'p\').length', 3)
                ->assertConsoleLogMissingWarning('value is not defined')

                ->waitForNavigate()->click('@link.to.second')
                ->assertSee('On second')

                ->back()
                ->assertSeeIn('@text', 'a,b,c')
                ->assertScript('document.getElementById(\'alpine-for-loop\').querySelectorAll(\'p\').length', 3)
                ->assertConsoleLogMissingWarning('value is not defined')
            ;
        });
    }

    public function test_injected_assets_such_as_nprogress_styles_are_retained_when_the_page_changes()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->tap(fn ($b) => $b->script('window._lw_dusk_test = true'))
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                // There should only be two style blocks, livewire styles and nprogress
                ->assertScript('return document.styleSheets.length', 2)
                ->click('@link.to.second')
                ->waitFor('@link.to.first')
                ->assertSee('On second')
                ->assertScript('return window._lw_dusk_test')
                // There should only be two style blocks, livewire styles and nprogress
                ->assertScript('return document.styleSheets.length', 2)
                ->click('@link.to.first')
                ->waitFor('@link.to.second')
                ->assertScript('return window._lw_dusk_test')
                ->assertSee('On first')
                // There should only be two style blocks, livewire styles and nprogress
                ->assertScript('return document.styleSheets.length', 2);
        });
    }

    public function test_remote_assets_loaded_with_the_directive_fully_load_before_component_scripts_and_initialization()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                // There should only be two style blocks, livewire styles and nprogress
                ->click('@link.to.asset')
                ->waitFor('@target')
                ->waitForTextIn('@target', 'bar')
                ;
        });
    }

    public function test_redirects_are_reflected_properly_in_the_url()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                ->click('@redirect.to.second.link')
                ->waitForText('On second')
                ->assertPathIs('/second')
            ;
        });
    }

    public function test_can_programmatically_click_navigate_links()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/first')
                ->assertSee('On first')
                ->tap(function ($browser) {
                    $browser->script(<<<'JS'
                        document.querySelector('a[href="/second"]').click()
                    JS);
                })
                ->waitForText('On second')
                ->assertPathIs('/second')
            ;
        });
    }

    public function test_can_binding_class_attribute_when_navigate_back()
    {
        Livewire::visit(new class extends Component {
            public function render(){
                return <<<'HTML'
                    <div>
                        <style>
                            .hidden {
                                display: none;
                            }
                        </style>

                        <div x-data="{ show: false }">
                            <button dusk="show-foo" type="button" @click="show = !show">Show</button>
                            <span :class="show || 'hidden'">foo</span>
                        </div>

                        <a :href="window.location.pathname" wire:navigate dusk="navigate-to-same-page">Go to same page</a>

                    </div>
                HTML;
            }
        })
            ->click('@show-foo')
            ->click('@show-foo')
            ->waitForNavigate()->click('@navigate-to-same-page')
            ->back()
            ->click('@show-foo')
            ->assertSee('foo');
    }

    public function test_can_navigate_links_and_use_snapshot_cache_for_first_10_history_items()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/navbar/one')
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.two')
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.one', 'active')

                ->waitForNavigateRequest()->click('@link.three')
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.four')
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNavigateRequest()->click('@link.five')
                ->assertSeeIn('@title', 'five')
                ->assertHasClass('@link.five', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNavigateRequest()->click('@link.six')
                ->assertSeeIn('@title', 'six')
                ->assertHasClass('@link.six', 'active')
                ->assertClassMissing('@link.five', 'active')

                ->waitForNavigateRequest()->click('@link.seven')
                ->assertSeeIn('@title', 'seven')
                ->assertHasClass('@link.seven', 'active')
                ->assertClassMissing('@link.six', 'active')

                ->waitForNavigateRequest()->click('@link.eight')
                ->assertSeeIn('@title', 'eight')
                ->assertHasClass('@link.eight', 'active')
                ->assertClassMissing('@link.seven', 'active')

                ->waitForNavigateRequest()->click('@link.nine')
                ->assertSeeIn('@title', 'nine')
                ->assertHasClass('@link.nine', 'active')
                ->assertClassMissing('@link.eight', 'active')

                ->waitForNavigateRequest()->click('@link.ten')
                ->assertSeeIn('@title', 'ten')
                ->assertHasClass('@link.ten', 'active')
                ->assertClassMissing('@link.nine', 'active')

                ->waitForNavigateRequest()->click('@link.eleven')
                ->assertSeeIn('@title', 'eleven')
                ->assertHasClass('@link.eleven', 'active')
                ->assertClassMissing('@link.ten', 'active')

                ->waitForNavigateRequest()->click('@link.twelve')
                ->assertSeeIn('@title', 'twelve')
                ->assertHasClass('@link.twelve', 'active')
                ->assertClassMissing('@link.eleven', 'active')

                ->waitForNavigateRequest()->click('@link.thirteen')
                ->assertSeeIn('@title', 'thirteen')
                ->assertHasClass('@link.thirteen', 'active')
                ->assertClassMissing('@link.twelve', 'active')

                // Assert no navigate request as we expect it to come from the cache
                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'twelve')
                ->assertHasClass('@link.twelve', 'active')
                ->assertClassMissing('@link.thirteen', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'eleven')
                ->assertHasClass('@link.eleven', 'active')
                ->assertClassMissing('@link.twelve', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'ten')
                ->assertHasClass('@link.ten', 'active')
                ->assertClassMissing('@link.eleven', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'nine')
                ->assertHasClass('@link.nine', 'active')
                ->assertClassMissing('@link.ten', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'eight')
                ->assertHasClass('@link.eight', 'active')
                ->assertClassMissing('@link.nine', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'seven')
                ->assertHasClass('@link.seven', 'active')
                ->assertClassMissing('@link.eight', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'six')
                ->assertHasClass('@link.six', 'active')
                ->assertClassMissing('@link.seven', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'five')
                ->assertHasClass('@link.five', 'active')
                ->assertClassMissing('@link.six', 'active')

                // Assert a navigate request was triggered as the remaining pages should no longer be in the cache
                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.five', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')

            ;
        });
    }

    public function test_can_navigate_links_and_if_a_refresh_happens_then_make_requests_until_pages_are_cached_again()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/navbar/one')
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.two')
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.one', 'active')

                ->waitForNavigateRequest()->click('@link.three')
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.four')
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForLivewire()->refresh()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.three')
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.two', 'active')

                ->waitForNavigateRequest()->click('@link.four')
                ->assertSeeIn('@title', 'four')
                ->assertHasClass('@link.four', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'three')
                ->assertHasClass('@link.three', 'active')
                ->assertClassMissing('@link.four', 'active')

                ->waitForNoNavigateRequest()->back()
                ->assertSeeIn('@title', 'two')
                ->assertHasClass('@link.two', 'active')
                ->assertClassMissing('@link.three', 'active')

                ->waitForNavigateRequest()->back()
                ->assertSeeIn('@title', 'one')
                ->assertHasClass('@link.one', 'active')
                ->assertClassMissing('@link.two', 'active')
            ;
        });
    }

    public function test_navigate_back_reevaluates_scripts()
    {
        $this->browse(function ($browser) {
            $browser
                ->visit('/script-component')
                ->waitForDialog(seconds: 1)
                ->assertDialogOpened('script was executed?')
                ->acceptDialog()
                ->assertSee('On script component')
                ->click('@link.to.first')
                ->waitForText('On first')
                ->back()
                ->waitForDialog(seconds: 1)
                ->assertDialogOpened('script was executed?')
                ->acceptDialog()
                ->waitForText('On script component')
                ->click('@link.to.first')
                ->waitForText('On first')
                ->back()
                ->waitForDialog(seconds: 1)
                ->assertDialogOpened('script was executed?')
                ->acceptDialog()
                ->waitForText('On script component')
            ;
        });
    }

    protected function registerComponentTestRoutes($routes)
    {
        $registered = 0;

        foreach ($routes as $route => $component) {
            $name = 'route-component-'.$registered++;

            Livewire::component($name, $component);

            Route::get($route, function () use ($name) {
                return app('livewire')->new($name)();
            })->middleware('web');
        }
    }
}

class FirstPage extends Component
{
    public function redirectToPageTwoUsingNavigate()
    {
        return $this->redirect('/second', navigate: true);
    }

    public function redirectToPageTwoUsingNavigateAndDestroyingSession()
    {
        session()->regenerate();

        return $this->redirect('/second', navigate: true);
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On first</div>

            <a :href="window.location.pathname + '#foo'" wire:navigate dusk="link.to.hashtag">Go to same page with hashtag</a>
            <a href="/second" wire:navigate.hover dusk="link.to.second">Go to second page</a>
            <a href="/third" wire:navigate.hover dusk="link.to.third">Go to slow third page</a>
            <a href="/second-remote-asset" wire:navigate.hover dusk="link.to.asset">Go to asset page</a>
            <button type="button" wire:click="redirectToPageTwoUsingNavigate" dusk="redirect.to.second">Redirect to second page</button>
            <a href="/redirect-to-second" wire:navigate dusk="redirect.to.second.link">Redirect to second page from link</a>
            <button type="button" wire:click="redirectToPageTwoUsingNavigateAndDestroyingSession" dusk="redirect.to.second.and.destroy.session">Redirect to second page and destroy session</button>

            <livewire:first-page-child />

            @persist('foo')
                <div x-data="{ count: 1 }">
                    <span x-text="count" dusk="count"></span>
                    <button x-on:click="count++" dusk="increment">+</button>
                </div>
            @endpersist
        </div>
        HTML;
    }
}

class FirstPageChild extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>First Child</div>

            <button type="button" wire:click="$parent.redirectToPageTwoUsingNavigate" dusk="redirect.to.second.from.child">Redirect to second page from child</button>
            <button type="button" x-on:click="console.log($wire.$parent.__instance.id)">shmump up</button>
        </div>
        HTML;
    }
}

class FirstPageWithLinkOutside extends Component
{
    #[Layout('test-views::layout-with-navigate-outside')]
    public function render()
    {
        return '<div>On first</div>';
    }
}

class SecondPage extends Component
{
    public function redirectToPageOne()
    {
        return redirect('/first');
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On second</div>

            <a href="/first" wire:navigate dusk="link.to.first">Go to first page</a>
            <button type="button" wire:click="redirectToPageOne" dusk="redirect.to.first">Redirect to first page</button>

            @persist('foo')
                <div x-data="{ count: 1 }">
                    <span x-text="count" dusk="count"></span>
                    <button x-on:click="count++" dusk="increment">+</button>
                </div>
            @endpersist

            <script data-navigate-once>window.foo = 'bar';</script>

            <script>
                document.addEventListener('livewire:navigated', () => {
                    window.foo_navigated = 'bar'
                })
            </script>
        </div>
        HTML;
    }
}

class ThirdPage extends Component
{
    public function mount()
    {
        sleep(1);
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            Done loading...
        </div>
        HTML;
    }
}

class FourthPage extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On fourth</div>

            <a href="/first" wire:navigate dusk="link.to.first">Go to first page</a>

            <script data-navigate-once>window.foo = 'bar';</script>

            <script>
                document.addEventListener('livewire:navigate', (event) => {
                    event.preventDefault();
                    if (window.foo === 'bar') {
                        window.foo = 'baz'
                    }
                    else if (window.foo === 'baz') {
                        window.foo ='bat'
                    } else {
                        Alpine.navigate(event.detail.url)
                    }
                })
            </script>
        </div>
        HTML;
    }
}

class FirstAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::layout')]
    public function render()
    {
        return '<div>On first asset page <a href="/second-asset" wire:navigate dusk="link.to.second">Go to second page</a></div>';
    }
}

class SecondAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::layout')]
    public function render()
    {
        return '<div>On second asset page <a href="/third-asset" wire:navigate dusk="link.to.third">Go to third page</a></div>';
    }
}

class ThirdAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::changed-layout')]
    public function render()
    {
        return '<div>On third asset page</div>';
    }
}

class FirstTrackedAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::tracked-layout')]
    public function render()
    {
        return '<div>On first asset page <a href="/second-tracked-asset" wire:navigate dusk="link.to.second">Go to second page</a></div>';
    }
}

class SecondTrackedAssetPage extends Component
{
    #[\Livewire\Attributes\Layout('test-views::changed-tracked-layout')]
    public function render()
    {
        return '<div>On second asset page</div>';
    }
}

class SecondRemoteAsset extends Component
{
    public function render()
    {
        return <<<'HTML'
            <div>
                <div>On second asset page</div>
                <div dusk="target">foo</div>
            </div>

            @assets
            <script src="https://cdn.jsdelivr.net/npm/pikaday/pikaday.js" defer></script>
            @endassets

            @script
            <script>
                window.datePicker = new Pikaday({ field: $wire.$el.querySelector('[data-picker]') })

                if (window.datePicker) {
                    document.querySelector('[dusk="target"]').textContent = 'bar'
                }
            </script>
            @endscript
        HTML;
    }
}

class QueryPage extends Component
{
    #[Url]
    public $query = 0;

    public function render()
    {
        return <<<'HTML'
            <div>
                <div>Query: {{ $query }}</div>
                <a href="/query-page?query=1" dusk="link.with.query.1">Link with query 1</a>
                <a href="/query-page?query=2" wire:navigate dusk="link.with.query.2">Link with query 2</a>
            </div>
        HTML;
    }
}

class FirstScrollPage extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On first</div>

            <div style="height: 100vh;">spacer</div>

            <div dusk="first-target">below the fold</div>

            <a href="/second-scroll" wire:navigate.hover dusk="link.to.second">Go to second page</a>

            <div style="height: 100vh;">spacer</div>
        </div>
        HTML;
    }
}

class SecondScrollPage extends Component
{
    public function render()
    {
        return <<<'HTML'
        <div>
            <div>On second</div>

            <div style="height: 100vh;">spacer</div>

            <div dusk="second-target">below the fold</div>

            <div style="height: 100vh;">spacer</div>
        </div>
        HTML;
    }
}

class ParentComponent extends Component
{
    public $text = '';

    #[On('my-event')]
    public function change_text($text)
    {
        $this->text = $text;
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <a href="/parent" wire:navigate dusk="home-link">Home</a>

            <p dusk="text-parent">{{ $text }}</p>

            <livewire:child-component key="child" />
        </div>
        HTML;
    }
}

class ChildComponent extends Component
{
    public $text = '';

    public function updated()
    {
        $this->dispatch('my-event', text: $this->text);
    }

    public function render()
    {
        return <<<'HTML'
        <div>
            <p dusk="text-child">{{ $text }}</p>

            <input type="text" wire:model.live="text" dusk="text-input">
        </div>
        HTML;
    }
}

class PageWithLinkAway extends Component
{
    #[Layout('test-views::layout')]
    public function render()
    {
        return <<<'HTML'
        <div dusk="page-with-link-away">
            <a wire:navigate dusk="link.away" href="/page-without-livewire-component">
                Link to page without Livewire component
            </a>
        </div>
        HTML;
    }
}

class PageWithAlpineForLoop extends Component
{
    #[Layout('test-views::layout')]
    public function render()
    {
        return <<<'HTML'
        <div dusk="page-with-alpine-for-loop" x-data="{ items: ['a', 'b', 'c'] }">
            <a href="/second" wire:navigate dusk="link.to.second">Go to second page</a>
            <div dusk="text" x-text="items"></div>
            <div id="alpine-for-loop">
                <template x-for="(value, index) in items" :key="index">
                    <p x-text="value"></p>
                </template>
            </div>
        </div>
        HTML;
    }
}

class NavBarComponent extends Component
{
    public $page;

    #[Layout('test-views::navbar-sidebar')]
    public function render()
    {
        return <<<'HTML'
            <div>
                <div>Page: <span dusk="title">{{ $page }}</span></div>
            </div>
        HTML;
    }
}

class ScriptComponent extends Component
{
    public function render()
    {
        return <<<'HTML'
            @script
            <script>
                confirm('script was executed?')
            </script>
            @endscript
            <div>
                <div>On script component</div>
                <a href="/first" wire:navigate dusk="link.to.first">Go to first page</a>
            </div>
        HTML;
    }
}

Function Calls

None

Variables

None

Stats

MD5 dd340c003b2164efeb8dd8bd0843d038
Eval Count 0
Decode Time 90 ms