Where available, Laravel Mix provides seamless support for hot module replacement.
{tip} Hot Module Replacement (or Hot Reloading) allows you to, not just refresh the page when a piece of JavaScript is changed, but it will also maintain the current state of the component in the browser. As an example, consider a simple counter component. When you press a button, the count goes up. Imagine that you click this button a number of times, and then update the component file. Once you do, the webpage will refresh to reflect your change, but the count will remain the same. It won't reset. This is the beauty of hot reloading!
Both Laravel and Laravel Mix work together to abstract away the complexities in getting hot reloading to work.
Have a look at your Laravel install's included package.json
file.
Within the scripts
object, you'll see:
"scripts": {
"dev": "cross-env NODE_ENV=development webpack --progress --hide-modules",
"watch": "cross-env NODE_ENV=development webpack --watch --progress --hide-modules",
"hot": "cross-env NODE_ENV=development webpack-dev-server --inline --hot",
"production": "cross-env NODE_ENV=production webpack --progress --hide-modules"
}
Take note of the hot
option; this is the one you want.
From the command line, run npm run hot
to boot up a Node server and monitor your bundle for changes.
Next, load your Laravel app in the browser, as you normally would. Perhaps, http://my-app.test
.
The key to making hot reloading work within a Laravel application is ensuring that all script sources reference the Node server URL that we just booted up. This will be http://localhost:8080. Now you could of course manually update your HTML/Blade files, like so:
<body>
<div id="app">...</div>
<script src="http://localhost:8080/js/bundle.js"></script>
</body>
And this would work. Give it a try. Assuming you have some demo components to work with, try changing the state in the browser, and then modifying the component's template. You should see your browser instantly refresh to reflect the change, without losing your state.
However, it can be a burden to manually change this URL for production deploys.
As such, Laravel offers a useful mix()
function, which will build up your script or stylesheet imports dynamically, and echo them out.
As such, the code snippet above may be changed to:
<body>
<div id="app"></div>
<script src="{{ mix('js/bundle.js') }}"></script>
</body>
With this adjustment, Laravel will do the work for you.
If you run 'npm run hot' to enable hot reloading, the function will set the necessary http://localhost:8080
base url.
If, instead, you use npm run dev
or npm run production
, it'll use your domain as the base.
If you develop your app on an HTTPS connection, your hot reloading scripts and styles must also be served via HTTPS.
To achieve this, add the --https
flag to the hot
option command within package.json
:
"scripts": {
"hot": "NODE_ENV=development webpack-dev-server --inline --hot --https",
}
With the above setting, the webpack-dev-server
will generate a self-signed certificate for you.
If you wish to use your own certificate, you may use these settings:
"hot": "NODE_ENV=development webpack-dev-server --inline --hot --https --key /path/to/server.key --cert /path/to/server.crt --cacert /path/to/ca.pem",
Now, in your HTML/Blade files you can use either:
<script src="https://localhost:8080/js/bundle.js"></script>
or:
<script src="{{ mix('js/bundle.js') }}"></script>
Laravel Mix includes the popular vue-loader
package, which means, for SPAs, there's nothing for you to do.
It'll all work seamlessly out of the box!
If you wish to run Hot Module Replacement on a different host or port you may override these within mix.options()
:
mix.options({
hmrOptions: {
host: 'example.com'
port: 8080
}
})